From 8a73d5cadbe40f7b275ddba37f680a141259756c Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 21 Nov 2023 14:38:19 +0800 Subject: [PATCH 001/637] add increment development function --- metagpt/actions/__init__.py | 1 + metagpt/actions/refine.py | 10 + metagpt/actions/refine_design_api.py | 231 ++++++++++++++++ metagpt/actions/refine_prd.py | 260 +++++++++++++++++++ metagpt/actions/refine_project_management.py | 208 +++++++++++++++ metagpt/actions/write_code.py | 10 +- metagpt/roles/architect.py | 48 +++- metagpt/roles/engineer.py | 74 +++++- metagpt/roles/product_manager.py | 36 ++- metagpt/roles/project_manager.py | 38 ++- metagpt/roles/role.py | 1 + startup.py | 54 +++- 12 files changed, 943 insertions(+), 28 deletions(-) create mode 100644 metagpt/actions/refine.py create mode 100644 metagpt/actions/refine_design_api.py create mode 100644 metagpt/actions/refine_prd.py create mode 100644 metagpt/actions/refine_project_management.py diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index b004bd58e..a9087e822 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -15,6 +15,7 @@ from metagpt.actions.design_api import WriteDesign from metagpt.actions.design_api_review import DesignReview from metagpt.actions.design_filenames import DesignFilenames from metagpt.actions.project_management import AssignTasks, WriteTasks +from metagpt.actions.refine import Refine from metagpt.actions.research import CollectLinks, WebBrowseAndSummarize, ConductResearch from metagpt.actions.run_code import RunCode from metagpt.actions.search_and_summarize import SearchAndSummarize diff --git a/metagpt/actions/refine.py b/metagpt/actions/refine.py new file mode 100644 index 000000000..beea40fc8 --- /dev/null +++ b/metagpt/actions/refine.py @@ -0,0 +1,10 @@ +from metagpt.actions import Action + + +# 增量开发动作的基类 +class Refine(Action): + def __init__(self, name="Refine", context=None, llm=None): + super().__init__(name, context, llm) + + def run(self, *args, **kwargs): + raise NotImplementedError diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py new file mode 100644 index 000000000..591db175a --- /dev/null +++ b/metagpt/actions/refine_design_api.py @@ -0,0 +1,231 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +import shutil +from pathlib import Path +from typing import List + +from metagpt.actions import Action, ActionOutput +from metagpt.config import CONFIG +from metagpt.const import WORKSPACE_ROOT +from metagpt.logs import logger +from metagpt.utils.common import CodeParser +from metagpt.utils.get_template import get_template +from metagpt.utils.json_to_markdown import json_to_markdown +from metagpt.utils.mermaid import mermaid_to_file + +templates = { + "json": { + "PROMPT_TEMPLATE": """ +# Context +{context} + +## Legacy Design +{legacy} + +## Format example +{format_example} +----- +Role: You are an architect; the goal is to design a SOTA PEP8-compliant python system; make the best use of good open source tools +Requirement: Fill in the following missing information based on the context, each section name is a key in json +Max Output: 8192 chars or 2048 tokens. Try to use them up. + +## Difference Description: Provided as Python list[str], the list of differences between the new design and the legacy design + +## Implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. + +## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores + +## File list: Provided as Python list[str], the list of ONLY REQUIRED files needed to write the program(LESS IS MORE!). Only need relative paths, comply with PEP8 standards. ALWAYS write a main.py or app.py here + +## Data structures and interface definitions: Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. + +## Program call flow: Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. + +## Anything UNCLEAR: Provide as Plain text. Make clear here. + +output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, +and only output the json inside this tag, nothing else +""", + "FORMAT_EXAMPLE": """ +[CONTENT] +{ + "Difference Description": ["The ..."], + "Implementation approach": "We will ...", + "Python package name": "snake_game", + "File list": ["main.py"], + "Data structures and interface definitions": ' + classDiagram + class Game{ + +int score + } + ... + Game "1" -- "1" Food: has + ', + "Program call flow": ' + sequenceDiagram + participant M as Main + ... + G->>M: end game + ', + "Anything UNCLEAR": "The requirement is clear to me." +} +[/CONTENT] +""", + }, + "markdown": { + "PROMPT_TEMPLATE": """ +# Context +{context} + +## Legacy Design +{legacy} + +## Format example +{format_example} +----- +Role: You are an architect; the goal is to design a SOTA PEP8-compliant python system; make the best use of good open source tools +Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately +Max Output: 8192 chars or 2048 tokens. Try to use them up. +Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. + +## Difference Description: Provided as Python list[str], the list of differences between the new design and the legacy design + +## Implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. + +## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores + +## File list: Provided as Python list[str], the list of ONLY REQUIRED files needed to write the program(LESS IS MORE!). Only need relative paths, comply with PEP8 standards. ALWAYS write a main.py or app.py here + +## Data structures and interface definitions: Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. + +## Program call flow: Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. + +## Anything UNCLEAR: Provide as Plain text. Make clear here. + +""", + "FORMAT_EXAMPLE": """ +--- +## Difference Description +```python +[ + "The ...", +] +``` + +## Implementation approach +We will ... + +## Python package name +```python +"snake_game" +``` + +## File list +```python +[ + "main.py", +] +``` + +## Data structures and interface definitions +```mermaid +classDiagram + class Game{ + +int score + } + ... + Game "1" -- "1" Food: has +``` + +## Program call flow +```mermaid +sequenceDiagram + participant M as Main + ... + G->>M: end game +``` + +## Anything UNCLEAR +The requirement is clear to me. +--- +""", + }, +} + +OUTPUT_MAPPING = { + "Difference Description": (List[str], ...), + "Implementation approach": (str, ...), + "Python package name": (str, ...), + "File list": (List[str], ...), + "Data structures and interface definitions": (str, ...), + "Program call flow": (str, ...), + "Anything UNCLEAR": (str, ...), +} + + +class RefineDesign(Action): + def __init__(self, name, context=None, llm=None): + super().__init__(name, context, llm) + self.desc = ( + "Based on the PRD, think about the system design, and design the corresponding APIs, " + "data structures, library tables, processes, and paths. Please provide your design, feedback " + "clearly and in detail." + ) + + def recreate_workspace(self, workspace: Path): + try: + shutil.rmtree(workspace) + except FileNotFoundError: + pass # Folder does not exist, but we don't care + workspace.mkdir(parents=True, exist_ok=True) + + async def _save_prd(self, docs_path, resources_path, context): + prd_file = docs_path / "prd.md" + if context[-1].instruct_content and context[-1].instruct_content.dict()["Competitive Quadrant Chart"]: + quadrant_chart = context[-1].instruct_content.dict()["Competitive Quadrant Chart"] + await mermaid_to_file(quadrant_chart, resources_path / "competitive_analysis") + + if context[-1].instruct_content: + logger.info(f"Saving PRD to {prd_file}") + prd_file.write_text(json_to_markdown(context[-1].instruct_content.dict())) + + async def _save_system_design(self, docs_path, resources_path, system_design): + data_api_design = system_design.instruct_content.dict()[ + "Data structures and interface definitions" + ] # CodeParser.parse_code(block="Data structures and interface definitions", text=content) + seq_flow = system_design.instruct_content.dict()[ + "Program call flow" + ] # CodeParser.parse_code(block="Program call flow", text=content) + await mermaid_to_file(data_api_design, resources_path / "data_api_design") + await mermaid_to_file(seq_flow, resources_path / "seq_flow") + system_design_file = docs_path / "system_design.md" + logger.info(f"Saving System Designs to {system_design_file}") + system_design_file.write_text((json_to_markdown(system_design.instruct_content.dict()))) + + async def _save(self, context, system_design): + if isinstance(system_design, ActionOutput): + ws_name = system_design.instruct_content.dict()["Python package name"] + else: + ws_name = CodeParser.parse_str(block="Python package name", text=system_design) + workspace = WORKSPACE_ROOT / ws_name + self.recreate_workspace(workspace) + docs_path = workspace / "docs" + resources_path = workspace / "resources" + docs_path.mkdir(parents=True, exist_ok=True) + resources_path.mkdir(parents=True, exist_ok=True) + await self._save_prd(docs_path, resources_path, context) + await self._save_system_design(docs_path, resources_path, system_design) + + async def run(self, context, legacy, format=CONFIG.prompt_format): + prompt_template, format_example = get_template(templates, format) + prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) + # system_design = await self._aask(prompt) + system_design = await self._aask_v1(prompt, "system_design", OUTPUT_MAPPING, format=format) + # fix Python package name, we can't system_design.instruct_content.python_package_name = "xxx" since "Python package name" contain space, have to use setattr + setattr( + system_design.instruct_content, + "Python package name", + system_design.instruct_content.dict()["Python package name"].strip().strip("'").strip('"'), + ) + await self._save(context, system_design) + return system_design diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py new file mode 100644 index 000000000..1e2709c6b --- /dev/null +++ b/metagpt/actions/refine_prd.py @@ -0,0 +1,260 @@ +from typing import List + +from metagpt.actions import Refine, ActionOutput, SearchAndSummarize +from metagpt.config import CONFIG +from metagpt.logs import logger +from metagpt.utils.get_template import get_template + +increment_template = { + "json": { + "PROMPT_TEMPLATE": """ +# Context +## User's New Requirements +{new_requirements} + +## Difference Description +{difference_description} + +## Legacy PRD +{legacy} + +## Search Information +{search_information} + +## mermaid quadrantChart code syntax example. DONT USE QUOTO IN CODE DUE TO INVALID SYNTAX. Replace the with REAL COMPETITOR NAME +```mermaid +quadrantChart + title Reach and engagement of campaigns + x-axis Low Reach --> High Reach + y-axis Low Engagement --> High Engagement + quadrant-1 We should expand + quadrant-2 Need to promote + quadrant-3 Re-evaluate + quadrant-4 May be improved + "Campaign: A": [0.3, 0.6] + "Campaign B": [0.45, 0.23] + "Campaign C": [0.57, 0.69] + "Campaign D": [0.78, 0.34] + "Campaign E": [0.40, 0.34] + "Campaign F": [0.35, 0.78] + "Our Target Product": [0.5, 0.6] +``` + +## Format example +{format_example} +----- +Role: You are a professional product manager; the goal is to design a concise, usable, efficient product based on the new requirements, the difference description and Legacy PRD. +Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design + +## New Requirements: Provide as Plain text and place the new requirements here + +## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple + +## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple + +## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less + +## Competitive Analysis: Provided as Python list[str], up to 7 competitive product analyses, consider as similar competitors as possible + +## Competitive Quadrant Chart: Use mermaid quadrantChart code syntax. up to 14 competitive products. Translation: Distribute these competitor scores evenly between 0 and 1, trying to conform to a normal distribution centered around 0.5 as much as possible. + +## Requirement Analysis: Provide as Plain text. Be simple. LESS IS MORE. Make your requirements less dumb. Delete the parts unnessasery. + +## Requirement Pool: Provided as Python list[list[str], the parameters are requirement description, priority(P0/P1/P2), respectively, comply with PEP standards; no more than 5 requirements and consider to make its difficulty lower + +## UI Design draft: Provide as Plain text. Be simple. Describe the elements and functions, also provide a simple style description and layout description. +## Anything UNCLEAR: Provide as Plain text. Make clear here. + +output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, +and only output the json inside this tag, nothing else +""", + "FORMAT_EXAMPLE": """ +[CONTENT] +{ + "New Requirements": "", + "Difference Description": "", + "Search Information": "", + "Requirements": "", + "Product Goals": [], + "User Stories": [], + "Competitive Analysis": [], + "Competitive Quadrant Chart": "quadrantChart + title Reach and engagement of campaigns + x-axis Low Reach --> High Reach + y-axis Low Engagement --> High Engagement + quadrant-1 We should expand + quadrant-2 Need to promote + quadrant-3 Re-evaluate + quadrant-4 May be improved + Campaign A: [0.3, 0.6] + Campaign B: [0.45, 0.23] + Campaign C: [0.57, 0.69] + Campaign D: [0.78, 0.34] + Campaign E: [0.40, 0.34] + Campaign F: [0.35, 0.78]", + "Requirement Analysis": "", + "Requirement Pool": [["P0","P0 requirement"],["P1","P1 requirement"]], + "UI Design draft": "", + "Anything UNCLEAR": "", +} +[/CONTENT] +""", + }, + "markdown": { + "PROMPT_TEMPLATE": """ +# Context +You need to refine the requirements based on the new requirements and the existing requirements' output. +## User's New Requirements +{new_requirements} + +## Difference Description +{difference_description} + +## Legacy PRD +{legacy} + +## Search Information +{search_information} + +## mermaid quadrantChart code syntax example. DONT USE QUOTO IN CODE DUE TO INVALID SYNTAX. Replace the with REAL COMPETITOR NAME +```mermaid +quadrantChart + title Reach and engagement of campaigns + x-axis Low Reach --> High Reach + y-axis Low Engagement --> High Engagement + quadrant-1 We should expand + quadrant-2 Need to promote + quadrant-3 Re-evaluate + quadrant-4 May be improved + "Campaign: A": [0.3, 0.6] + "Campaign B": [0.45, 0.23] + "Campaign C": [0.57, 0.69] + "Campaign D": [0.78, 0.34] + "Campaign E": [0.40, 0.34] + "Campaign F": [0.35, 0.78] + "Our Target Product": [0.5, 0.6] +``` + +## Format example +{format_example} +----- +Role: You are a professional product manager; the goal is to design a concise, usable, efficient product +Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format. + +## New Requirements: Provide as Plain text and place the new requirements here + +## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple + +## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple + +## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less + +## Competitive Analysis: Provided as Python list[str], up to 7 competitive product analyses, consider as similar competitors as possible + +## Competitive Quadrant Chart: Use mermaid quadrantChart code syntax. up to 14 competitive products. Translation: Distribute these competitor scores evenly between 0 and 1, trying to conform to a normal distribution centered around 0.5 as much as possible. + +## Requirement Analysis: Provide as Plain text. Be simple. LESS IS MORE. Make your requirements less dumb. Delete the parts unnessasery. + +## Requirement Pool: Provided as Python list[list[str], the parameters are requirement description, priority(P0/P1/P2), respectively, comply with PEP standards; no more than 5 requirements and consider to make its difficulty lower + +## UI Design draft: Provide as Plain text. Be simple. Describe the elements and functions, also provide a simple style description and layout description. +## Anything UNCLEAR: Provide as Plain text. Make clear here. +""", + "FORMAT_EXAMPLE": """ +--- +## New Requirements +The boss ... + +## Difference Description +```python +[ + "...", +] + +## Product Goals +```python +[ + "Create a ...", +] +``` + +## User Stories +```python +[ + "As a user, ...", +] +``` + +## Competitive Analysis +```python +[ + "Python Snake Game: ...", +] +``` + +## Competitive Quadrant Chart +```mermaid +quadrantChart + title Reach and engagement of campaigns + ... + "Our Target Product": [0.6, 0.7] +``` + +## Requirement Analysis +The product should be a ... + +## Requirement Pool +```python +[ + ["End game ...", "P0"] +] +``` + +## UI Design draft +Give a basic function description, and a draft + +## Anything UNCLEAR +There are no unclear points. +--- +""", + }, +} + +INCREMENT_OUTPUT_MAPPING = { + "New Requirements": (str, ...), + # "Major Enhancements": (List[str], ...), + "Difference Description": (List[str], ...), + "Product Goals": (List[str], ...), + "User Stories": (List[str], ...), + "Competitive Analysis": (List[str], ...), + "Competitive Quadrant Chart": (str, ...), + "Requirement Analysis": (str, ...), + "Requirement Pool": (List[List[str]], ...), + "UI Design draft": (str, ...), + "Anything UNCLEAR": (str, ...), +} + + +# 对于产品经理,增量开发的动作是:RefinePDR,输出是结合新需求和已有需求输出的新的PDR +class RefinePRD(Refine): + + def __init__(self, name="RefinePRD", context=None, llm=None): + super().__init__(name, context, llm) + + async def run(self, new_requirements, difference_description, legacy, format=CONFIG.prompt_format, *args, **kwargs): + sas = SearchAndSummarize() + rsp = "" + info = f"### Search Results\n{sas.result}\n\n### Search Summary\n{rsp}" + if sas.result: + logger.info(sas.result) + logger.info(rsp) + + prompt_template, format_example = get_template(increment_template, format) + prompt = prompt_template.format( + new_requirements=new_requirements, difference_description=difference_description, legacy=legacy, search_information=info, + format_example=format_example + ) + logger.debug(prompt) + prd = await self._aask_v1(prompt, "prd", INCREMENT_OUTPUT_MAPPING, format=format) + return prd diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py new file mode 100644 index 000000000..354996f3e --- /dev/null +++ b/metagpt/actions/refine_project_management.py @@ -0,0 +1,208 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +from typing import List + +from metagpt.actions.action import Action +from metagpt.config import CONFIG +from metagpt.const import WORKSPACE_ROOT +from metagpt.utils.common import CodeParser +from metagpt.utils.get_template import get_template +from metagpt.utils.json_to_markdown import json_to_markdown + +templates = { + "json": { + "PROMPT_TEMPLATE": """ +# Context +{context} + +## Legacy Design +{legacy} + +## Format example +{format_example} +----- +Role: You are a project manager; the goal is to break down tasks according to PRD/technical design, give a task list, and analyze task dependencies to start with the prerequisite modules +Requirements: Based on the context, fill in the following missing information, each section name is a key in json. Here the granularity of the task is a file, if there are any missing files, you can supplement them +Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. + +## Required Python third-party packages: Provided in requirements.txt format + +## Required Other language third-party packages: Provided in requirements.txt format + +## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. + +## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first + +## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first + +## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. + +## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. + +## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. + +output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, +and only output the json inside this tag, nothing else +""", + "FORMAT_EXAMPLE": ''' +{ + "Required Python third-party packages": [ + "flask==1.1.2", + "bcrypt==3.2.0" + ], + "Required Other language third-party packages": [ + "No third-party ..." + ], + "Full API spec": """ + openapi: 3.0.0 + ... + description: A JSON object ... + """, + "Logic Analysis": [ + ["game.py","Contains..."] + ], + "Task list": [ + "game.py" + ], + "Shared Knowledge": """ + 'game.py' contains ... + """, + "Difference Description": """ + The ... + """, + "Anything UNCLEAR": "We need ... how to start." +} +''', + }, + "markdown": { + "PROMPT_TEMPLATE": """ +# Context +{context} + +## Legacy Design +{legacy} + +## Format example +{format_example} +----- +Role: You are a project manager; the goal is to break down tasks according to PRD/technical design, give a task list, and analyze task dependencies to start with the prerequisite modules +Requirements: Based on the context, fill in the following missing information, note that all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file, if there are any missing files, you can supplement them +Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. + +## Required Python third-party packages: Provided in requirements.txt format + +## Required Other language third-party packages: Provided in requirements.txt format + +## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. + +## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first + +## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first + +## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. + +## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. + +## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. + +""", + "FORMAT_EXAMPLE": ''' +--- +## Required Python third-party packages +```python +""" +flask==1.1.2 +bcrypt==3.2.0 +""" +``` + +## Required Other language third-party packages +```python +""" +No third-party ... +""" +``` + +## Full API spec +```python +""" +openapi: 3.0.0 +... +description: A JSON object ... +""" +``` + +## Logic Analysis +```python +[ + ["game.py", "Contains ..."], +] +``` + +## Task list +```python +[ + "game.py", +] +``` + +## Shared Knowledge +```python +""" +'game.py' contains ... +""" +``` + +## Difference Description +```python +""" +The ... +""" +``` + +## Anything UNCLEAR +We need ... how to start. +--- +''', + }, +} +OUTPUT_MAPPING = { + "Required Python third-party packages": (List[str], ...), + "Required Other language third-party packages": (List[str], ...), + "Full API spec": (str, ...), + "Logic Analysis": (List[List[str]], ...), + "Task list": (List[str], ...), + "Shared Knowledge": (str, ...), + "Difference Description": (str, ...), + "Anything UNCLEAR": (str, ...), +} + + +class RefineTasks(Action): + def __init__(self, name="CreateTasks", context=None, llm=None): + super().__init__(name, context, llm) + + def _save(self, context, rsp): + if context[-1].instruct_content: + ws_name = context[-1].instruct_content.dict()["Python package name"] + else: + ws_name = CodeParser.parse_str(block="Python package name", text=context[-1].content) + file_path = WORKSPACE_ROOT / ws_name / "docs/api_spec_and_tasks.md" + file_path.write_text(json_to_markdown(rsp.instruct_content.dict())) + + # Write requirements.txt + requirements_path = WORKSPACE_ROOT / ws_name / "requirements.txt" + requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Required Python third-party packages"))) + + async def run(self, context, legacy, format=CONFIG.prompt_format): + prompt_template, format_example = get_template(templates, format) + prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) + rsp = await self._aask_v1(prompt, "task", OUTPUT_MAPPING, format=format) + self._save(context, rsp) + return rsp + + +class AssignTasks(Action): + async def run(self, *args, **kwargs): + # Here you should implement the actual action + pass diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index c000805c5..663ae0929 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -7,6 +7,7 @@ """ from metagpt.actions import WriteDesign from metagpt.actions.action import Action +from metagpt.actions.refine_design_api import RefineDesign from metagpt.const import WORKSPACE_ROOT from metagpt.logs import logger from metagpt.schema import Message @@ -55,7 +56,14 @@ class WriteCode(Action): if self._is_invalid(filename): return - design = [i for i in context if i.cause_by == WriteDesign][0] + # FIXME: 需要适配increment + # design = [i for i in context if i.cause_by == WriteDesign][0] + design = [] + for i in context: + if i.cause_by == WriteDesign: + design.append(i) + elif i.cause_by == RefineDesign: + design.append(i) ws_name = CodeParser.parse_str(block="Python package name", text=design.content) ws_path = WORKSPACE_ROOT / ws_name diff --git a/metagpt/roles/architect.py b/metagpt/roles/architect.py index 15d5fe5b1..dddcd2a8b 100644 --- a/metagpt/roles/architect.py +++ b/metagpt/roles/architect.py @@ -6,9 +6,13 @@ @File : architect.py """ -from metagpt.actions import WritePRD +from metagpt.actions import WritePRD, ActionOutput from metagpt.actions.design_api import WriteDesign +from metagpt.actions.refine_design_api import RefineDesign +from metagpt.actions.refine_prd import RefinePRD +from metagpt.logs import logger from metagpt.roles import Role +from metagpt.schema import Message class Architect(Role): @@ -23,17 +27,43 @@ class Architect(Role): """ def __init__( - self, - name: str = "Bob", - profile: str = "Architect", - goal: str = "Design a concise, usable, complete python system", - constraints: str = "Try to specify good open source tools as much as possible", + self, + name: str = "Bob", + profile: str = "Architect", + goal: str = "Design a concise, usable, complete python system", + constraints: str = "Try to specify good open source tools as much as possible", + legacy: str = "", + increment: bool = False, ) -> None: """Initializes the Architect with given attributes.""" super().__init__(name, profile, goal, constraints) + self.legacy = legacy + self.increment = increment # Initialize actions specific to the Architect role - self._init_actions([WriteDesign]) - # Set events or actions the Architect should watch or be aware of - self._watch({WritePRD}) + if self.increment: + self._init_actions([RefineDesign]) + self._watch({RefinePRD}) + else: + self._init_actions([WriteDesign]) + self._watch({WritePRD}) + + async def _act(self) -> Message: + if self.increment: + logger.info(f"{self._setting}: ready to RefineDesign") + response = await self._rc.todo.run(self._rc.history, self.legacy) + + else: + logger.info(f"{self._setting}: ready to WriteDesign") + response = await self._rc.todo.run(self._rc.history) + + if isinstance(response, ActionOutput): + msg = Message(content=response.content, instruct_content=response.instruct_content, + role=self.profile, cause_by=type(self._rc.todo)) + else: + msg = Message(content=response, role=self.profile, cause_by=type(self._rc.todo)) + self._rc.memory.add(msg) + logger.debug(f"{response}") + + return msg diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 1f6685b38..0a0107636 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -11,6 +11,8 @@ from collections import OrderedDict from pathlib import Path from metagpt.actions import WriteCode, WriteCodeReview, WriteDesign, WriteTasks +from metagpt.actions.refine_design_api import RefineDesign +from metagpt.actions.refine_project_management import RefineTasks from metagpt.const import WORKSPACE_ROOT from metagpt.logs import logger from metagpt.roles import Role @@ -68,14 +70,21 @@ class Engineer(Role): constraints: str = "The code should conform to standards like PEP8 and be modular and maintainable", n_borg: int = 1, use_code_review: bool = False, + legacy: str = "", + increment: bool = False, ) -> None: """Initializes the Engineer role with given attributes.""" super().__init__(name, profile, goal, constraints) self._init_actions([WriteCode]) self.use_code_review = use_code_review - if self.use_code_review: + self.legacy = legacy + self.increment = increment + if self.use_code_review or self.increment: self._init_actions([WriteCode, WriteCodeReview]) - self._watch([WriteTasks]) + if self.increment: + self._watch([RefineTasks]) + else: + self._watch([WriteTasks]) self.todos = [] self.n_borg = n_borg @@ -96,7 +105,10 @@ class Engineer(Role): return CodeParser.parse_str(block="Python package name", text=system_design_msg.content) def get_workspace(self) -> Path: - msg = self._rc.memory.get_by_action(WriteDesign)[-1] + if self.increment: + msg = self._rc.memory.get_by_action(RefineDesign)[-1] + else: + msg = self._rc.memory.get_by_action(WriteDesign)[-1] if not msg: return WORKSPACE_ROOT / "src" workspace = self.parse_workspace(msg) @@ -167,6 +179,55 @@ class Engineer(Role): ) return msg + async def _act_increment(self, legacy) -> Message: + code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later + flag = True + for todo in self.todos: + """ + # Select essential information from the historical data to reduce the length of the prompt (summarized from human experience): + 1. All from Architect + 2. All from ProjectManager + 3. Do we need other codes (currently needed)? + TODO: The goal is not to need it. After clear task decomposition, based on the design idea, you should be able to write a single file without needing other codes. If you can't, it means you need a clearer definition. This is the key to writing longer code. + """ + context = [] + + if self.increment: + msg = self._rc.memory.get_by_actions([RefineDesign, RefineTasks, WriteCode]) + else: + msg = self._rc.memory.get_by_actions([WriteDesign, WriteTasks, WriteCode]) + + for m in msg: + context.append(m.content) + context_str = "\n".join(context) + # Refine code or Write code + if flag and self.increment: + code = legacy + flag = False + else: + code = await WriteCode().run(context=context_str, filename=todo) + + # Code review + if self.use_code_review: + try: + rewrite_code = await WriteCode().run(context=context_str, code=code, filename=todo) + code = rewrite_code + except Exception as e: + logger.error("code review failed!", e) + pass + file_path = self.write_file(todo, code) + msg = Message(content=code, role=self.profile, cause_by=WriteCode) + self._rc.memory.add(msg) + + code_msg = todo + FILENAME_CODE_SEP + str(file_path) + code_msg_all.append(code_msg) + + logger.info(f"Done {self.get_workspace()} generating.") + msg = Message( + content=MSG_SEP.join(code_msg_all), role=self.profile, cause_by=type(self._rc.todo), send_to="QaEngineer" + ) + return msg + async def _act_sp_precision(self) -> Message: code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later for todo in self.todos: @@ -207,7 +268,12 @@ class Engineer(Role): async def _act(self) -> Message: """Determines the mode of action based on whether code review is used.""" - logger.info(f"{self._setting}: ready to WriteCode") + if self.increment: + logger.info(f"{self._setting}: ready to RefineWriteCode") + else: + logger.info(f"{self._setting}: ready to WriteCode") if self.use_code_review: return await self._act_sp_precision() + elif self.increment: + return await self._act_increment(self.legacy) return await self._act_sp() diff --git a/metagpt/roles/product_manager.py b/metagpt/roles/product_manager.py index a58ea5385..0b2d83ed0 100644 --- a/metagpt/roles/product_manager.py +++ b/metagpt/roles/product_manager.py @@ -5,8 +5,11 @@ @Author : alexanderwu @File : product_manager.py """ -from metagpt.actions import BossRequirement, WritePRD +from metagpt.actions import BossRequirement, WritePRD, ActionOutput +from metagpt.actions.refine_prd import RefinePRD +from metagpt.logs import logger from metagpt.roles import Role +from metagpt.schema import Message class ProductManager(Role): @@ -26,6 +29,9 @@ class ProductManager(Role): profile: str = "Product Manager", goal: str = "Efficiently create a successful product", constraints: str = "", + difference_description: str = "", + legacy: str = "", + increment: bool = False, ) -> None: """ Initializes the ProductManager role with given attributes. @@ -37,5 +43,31 @@ class ProductManager(Role): constraints (str): Constraints or limitations for the product manager. """ super().__init__(name, profile, goal, constraints) - self._init_actions([WritePRD]) + self.difference_description = difference_description + self.legacy = legacy + self.increment = increment + + if self.increment: + self._init_actions([RefinePRD]) + else: + self._init_actions([WritePRD]) self._watch([BossRequirement]) + + async def _act(self) -> Message: + if self.increment: + logger.info(f"{self._setting}: ready to RefinePRD") + response = await self._rc.todo.run(self._rc.history, self.difference_description, self.legacy) + + else: + logger.info(f"{self._setting}: ready to WritePRD") + response = await self._rc.todo.run(self._rc.history) + + if isinstance(response, ActionOutput): + msg = Message(content=response.content, instruct_content=response.instruct_content, + role=self.profile, cause_by=type(self._rc.todo)) + else: + msg = Message(content=response, role=self.profile, cause_by=type(self._rc.todo)) + self._rc.memory.add(msg) + logger.debug(f"{response}") + + return msg diff --git a/metagpt/roles/project_manager.py b/metagpt/roles/project_manager.py index 7e7c5699d..21a196d21 100644 --- a/metagpt/roles/project_manager.py +++ b/metagpt/roles/project_manager.py @@ -5,9 +5,13 @@ @Author : alexanderwu @File : project_manager.py """ -from metagpt.actions import WriteTasks +from metagpt.actions import WriteTasks, ActionOutput from metagpt.actions.design_api import WriteDesign +from metagpt.actions.refine_design_api import RefineDesign +from metagpt.actions.refine_project_management import RefineTasks +from metagpt.logs import logger from metagpt.roles import Role +from metagpt.schema import Message class ProjectManager(Role): @@ -27,6 +31,8 @@ class ProjectManager(Role): profile: str = "Project Manager", goal: str = "Improve team efficiency and deliver with quality and quantity", constraints: str = "", + increment: bool = False, + legacy: str = "", ) -> None: """ Initializes the ProjectManager role with given attributes. @@ -38,5 +44,31 @@ class ProjectManager(Role): constraints (str): Constraints or limitations for the project manager. """ super().__init__(name, profile, goal, constraints) - self._init_actions([WriteTasks]) - self._watch([WriteDesign]) + self.increment = increment + self.legacy = legacy + + if self.increment: + self._init_actions([RefineTasks]) + self._watch([RefineDesign]) + else: + self._init_actions([WriteTasks]) + self._watch([WriteDesign]) + + async def _act(self) -> Message: + if self.increment: + logger.info(f"{self._setting}: ready to RefineTasks") + response = await self._rc.todo.run(self._rc.history, self.legacy) + + else: + logger.info(f"{self._setting}: ready to WriteTasks") + response = await self._rc.todo.run(self._rc.history) + + if isinstance(response, ActionOutput): + msg = Message(content=response.content, instruct_content=response.instruct_content, + role=self.profile, cause_by=type(self._rc.todo)) + else: + msg = Message(content=response, role=self.profile, cause_by=type(self._rc.todo)) + self._rc.memory.add(msg) + logger.debug(f"{response}") + + return msg diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 0251176f7..a39ca1001 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -205,6 +205,7 @@ class Role: # history=self.history) logger.info(f"{self._setting}: ready to {self._rc.todo}") + response = await self._rc.todo.run(self._rc.important_memory) # logger.info(response) if isinstance(response, ActionOutput): diff --git a/startup.py b/startup.py index e9fbf94d3..7e8e0a692 100644 --- a/startup.py +++ b/startup.py @@ -1,6 +1,7 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- import asyncio +import os import fire @@ -16,26 +17,57 @@ from metagpt.team import Team async def startup( idea: str, + difference_description: str = "", + path: str = "", investment: float = 3.0, n_round: int = 5, code_review: bool = False, run_tests: bool = False, implement: bool = True, + increment: bool = False, ): """Run a startup. Be a boss.""" company = Team() - company.hire( - [ - ProductManager(), - Architect(), - ProjectManager(), - ] - ) + + if increment: + # 读取文件 + prd_path = os.path.join(path, 'docs/prd.md') + design_path = os.path.join(path, 'docs/system_design.md') + api_spec_path = os.path.join(path, 'docs/api_spec_and_tasks.md') + code_path = os.path.join(path, os.path.basename(path)) + + with open(prd_path, 'r', encoding='utf-8') as f: + legacy_prd = f.read() + + with open(design_path, 'r', encoding='utf-8') as f: + legacy_design = f.read() + + with open(api_spec_path, 'r', encoding='utf-8') as f: + legacy_api_spec = f.read() + + # 遍历文件夹,获取所有代码文件 + legacy_code = '' + for root, dirs, files in os.walk(code_path): + filenames = [filename for filename in files if filename.endswith('.py')] + legacy_code += f'There are {len(files)} scripts in the current folder: {", ".join(filenames)}\n\n' + for file in files: + if file.endswith('.py'): + with open(os.path.join(root, file), 'r', encoding='utf-8') as f: + legacy_code += f.read() + '\n\n' + + company.hire( + [ProductManager(difference_description=difference_description, legacy=legacy_prd, increment=increment), + Architect(legacy=legacy_design, increment=increment), + ProjectManager(legacy=legacy_api_spec, increment=increment)]) + else: + company.hire([ProductManager(), Architect(), ProjectManager()]) # if implement or code_review - if implement or code_review: + if (implement or code_review) and not increment: # developing features: implement the idea company.hire([Engineer(n_borg=5, use_code_review=code_review)]) + elif (implement or code_review) and increment: + company.hire([Engineer(n_borg=5, use_code_review=code_review, legacy=legacy_code, increment=increment)]) if run_tests: # developing features: run tests on the spot and identify bugs @@ -49,11 +81,14 @@ async def startup( def main( idea: str, + difference_description: str = "", + path: str = "", investment: float = 3.0, n_round: int = 5, code_review: bool = True, run_tests: bool = False, implement: bool = True, + increment: bool = False, ): """ We are a software startup comprised of AI. By investing in us, @@ -65,7 +100,8 @@ def main( :param code_review: Whether to use code review. :return: """ - asyncio.run(startup(idea, investment, n_round, code_review, run_tests, implement)) + asyncio.run( + startup(idea, difference_description, path, investment, n_round, code_review, run_tests, implement, increment)) if __name__ == "__main__": From 1ea3c0c9f3149d1ccd4cd36ea8a2bd738c683a52 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 21 Nov 2023 14:42:45 +0800 Subject: [PATCH 002/637] add increment development function --- metagpt/roles/role.py | 1 - 1 file changed, 1 deletion(-) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index a39ca1001..0251176f7 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -205,7 +205,6 @@ class Role: # history=self.history) logger.info(f"{self._setting}: ready to {self._rc.todo}") - response = await self._rc.todo.run(self._rc.important_memory) # logger.info(response) if isinstance(response, ActionOutput): From 89f1dce936f238d1931341c1ff79abce1555fc57 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 21 Nov 2023 18:38:41 +0800 Subject: [PATCH 003/637] feat: add actions: plan, write_code_function, pycode_executor. --- metagpt/actions/__init__.py | 6 + metagpt/actions/code_executor.py | 173 ++++++++++++++++++++ metagpt/actions/plan.py | 20 +++ metagpt/actions/write_code_v2.py | 36 ++++ metagpt/prompts/plan.py | 7 + metagpt/schema.py | 1 + tests/metagpt/actions/test_code_executor.py | 58 +++++++ tests/metagpt/actions/test_plan.py | 12 ++ tests/metagpt/actions/test_write_code_v2.py | 22 +++ 9 files changed, 335 insertions(+) create mode 100644 metagpt/actions/code_executor.py create mode 100644 metagpt/actions/plan.py create mode 100644 metagpt/actions/write_code_v2.py create mode 100644 metagpt/prompts/plan.py create mode 100644 tests/metagpt/actions/test_code_executor.py create mode 100644 tests/metagpt/actions/test_plan.py create mode 100644 tests/metagpt/actions/test_write_code_v2.py diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index b004bd58e..d7afae2fe 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -23,6 +23,9 @@ from metagpt.actions.write_code_review import WriteCodeReview from metagpt.actions.write_prd import WritePRD from metagpt.actions.write_prd_review import WritePRDReview from metagpt.actions.write_test import WriteTest +from metagpt.actions.code_executor import PyCodeExecutor +from metagpt.actions.write_code_v2 import WriteCode as WriteCodeFunction +from metagpt.actions.plan import Plan class ActionType(Enum): @@ -45,6 +48,9 @@ class ActionType(Enum): COLLECT_LINKS = CollectLinks WEB_BROWSE_AND_SUMMARIZE = WebBrowseAndSummarize CONDUCT_RESEARCH = ConductResearch + PYCODE_EXECUTOR = PyCodeExecutor + WRITE_CODE_FUNCTION = WriteCodeFunction + PLAN = Plan __all__ = [ diff --git a/metagpt/actions/code_executor.py b/metagpt/actions/code_executor.py new file mode 100644 index 000000000..c05c00c9c --- /dev/null +++ b/metagpt/actions/code_executor.py @@ -0,0 +1,173 @@ +# -*- encoding: utf-8 -*- +""" +@Date : 2023/11/17 14:22:15 +@Author : orange-crow +@File : code_executor.py +""" +from abc import ABC, abstractmethod +from pathlib import Path +from typing import Dict, List, Tuple, Union + +import nbformat +from nbclient import NotebookClient +from nbformat.v4 import new_code_cell, new_output +from rich.console import Console +from rich.syntax import Syntax + +from metagpt.actions import Action +from metagpt.schema import Message + + +class CodeExecutor(ABC): + @abstractmethod + async def build(self): + """build code executor""" + ... + + @abstractmethod + async def run(self, code: str): + """run code""" + ... + + @abstractmethod + async def terminate(self): + """terminate executor""" + ... + + @abstractmethod + async def reset(self): + """reset executor""" + ... + + +class PyCodeExecutor(CodeExecutor, Action): + """execute code, return result to llm, and display it.""" + + def __init__(self, name: str = "python_executor", context=None, llm=None): + super().__init__(name, context, llm) + self.nb = nbformat.v4.new_notebook() + self.nb_client = NotebookClient(self.nb) + self.console = Console() + self.interaction = "ipython" if self.is_ipython() else "terminal" + + async def build(self): + if self.nb_client.kc is None or not await self.nb_client.kc.is_alive(): + self.nb_client.create_kernel_manager() + self.nb_client.start_new_kernel() + self.nb_client.start_new_kernel_client() + + async def terminate(self): + """kill NotebookClient""" + await self.nb_client._async_cleanup_kernel() + + async def reset(self): + """reset NotebookClient""" + await self.terminate() + self.nb_client = NotebookClient(self.nb) + + def add_code_cell(self, code): + self.nb.cells.append(new_code_cell(source=code)) + + def _display(self, code, language: str = "python"): + if language == "python": + code = Syntax(code, "python", theme="paraiso-dark", line_numbers=True) + self.console.print("\n") + self.console.print(code) + + def add_output_to_cell(self, cell, output): + if "outputs" not in cell: + cell["outputs"] = [] + # TODO: show figures + else: + cell["outputs"].append(new_output(output_type="stream", name="stdout", text=str(output))) + + def parse_outputs(self, outputs: List) -> str: + assert isinstance(outputs, list) + parsed_output = {"text": [], "image": []} + + # empty outputs: such as 'x=1\ny=2' + if not outputs: + return parsed_output + + for output in outputs: + if output["output_type"] == "stream": + parsed_output["text"].append(output["text"]) + elif output["output_type"] == "display_data": + self.show_bytes_figure(output["data"]["image/png"], self.interaction) + parsed_output["image"].append(output["data"]["image/png"]) + return str(parsed_output) + + def show_bytes_figure(self, image_base64: str, interaction_type: str = "ipython"): + import base64 + + image_bytes = base64.b64decode(image_base64) + if interaction_type == "ipython": + from IPython.display import Image, display + + display(Image(data=image_bytes)) + else: + import io + + from PIL import Image + + image = Image.open(io.BytesIO(image_bytes)) + image.show() + + def is_ipython(self) -> bool: + try: + # 如果在Jupyter Notebook中运行,__file__ 变量不存在 + from IPython import get_ipython + + if get_ipython() is not None and "IPKernelApp" in get_ipython().config: + return True + else: + return False + except NameError: + # 如果在Python脚本中运行,__file__ 变量存在 + return False + + def _process_code(self, code: Union[str, Dict, Message], language: str = None) -> Tuple: + if isinstance(code, str) and Path(code).suffix in (".py", ".txt"): + code = Path(code).read_text(encoding="utf-8") + return code, language + + if isinstance(code, str): + return code, language + + if isinstance(code, dict): + assert "code" in code + assert "language" in code + code, language = code["code"], code["language"] + elif isinstance(code, Message): + assert hasattr(code, "language") + code, language = code.content, code.language + else: + raise ValueError(f"Not support code type {type(code).__name__}.") + + return code, language + + async def run(self, code: Union[str, Dict, Message], language: str = "python") -> Message: + code, language = self._process_code(code, language) + + self._display(code, language) + + if language == "python": + # add code to the notebook + self.add_code_cell(code=code) + try: + # build code executor + await self.build() + # run code + # TODO: add max_tries for run code. + cell_index = len(self.nb.cells) - 1 + await self.nb_client.async_execute_cell(self.nb.cells[-1], cell_index) + return Message( + self.parse_outputs(self.nb.cells[-1].outputs), state="done", sent_from=self.__class__.__name__ + ) + except Exception as e: + # FIXME: CellExecutionError is hard to read. for example `1\0` raise ZeroDivisionError: + # CellExecutionError('An error occurred while executing the following cell:\n------------------\nz=1/0\n------------------\n\n\n\x1b[0;31m---------------------------------------------------------------------------\x1b[0m\n\x1b[0;31mZeroDivisionError\x1b[0m Traceback (most recent call last)\nCell \x1b[0;32mIn[1], line 1\x1b[0m\n\x1b[0;32m----> 1\x1b[0m z\x1b[38;5;241m=\x1b[39m\x1b[38;5;241;43m1\x1b[39;49m\x1b[38;5;241;43m/\x1b[39;49m\x1b[38;5;241;43m0\x1b[39;49m\n\n\x1b[0;31mZeroDivisionError\x1b[0m: division by zero\n') + return Message(e, state="error", sent_from=self.__class__.__name__) + else: + # TODO: markdown + raise NotImplementedError(f"Not support this code type : {language}, Only support code!") diff --git a/metagpt/actions/plan.py b/metagpt/actions/plan.py new file mode 100644 index 000000000..d46783ba2 --- /dev/null +++ b/metagpt/actions/plan.py @@ -0,0 +1,20 @@ +# -*- encoding: utf-8 -*- +""" +@Date : 2023/11/20 11:24:03 +@Author : orange-crow +@File : plan.py +""" +from metagpt.actions import Action +from metagpt.prompts.plan import TASK_PLAN_SYSTEM_MSG +from metagpt.schema import Message + + +class Plan(Action): + def __init__(self, llm=None): + super().__init__("", None, llm) + + async def run(self, prompt: str, role: str = None, system_msg: str = None) -> str: + if role: + system_msg = TASK_PLAN_SYSTEM_MSG.format(role=role) + rsp = await self._aask(system_msg + prompt) + return Message(rsp, role="assistant", sent_from=self.__class__.__name__) diff --git a/metagpt/actions/write_code_v2.py b/metagpt/actions/write_code_v2.py new file mode 100644 index 000000000..335e70dc0 --- /dev/null +++ b/metagpt/actions/write_code_v2.py @@ -0,0 +1,36 @@ +# -*- encoding: utf-8 -*- +""" +@Date : 2023/11/20 13:19:39 +@Author : orange-crow +@File : write_code_v2.py +""" +from typing import Dict, List, Union + +from metagpt.actions import Action +from metagpt.schema import Message + + +class WriteCode(Action): + """Use openai function to generate code.""" + + def __init__(self, name: str = "", context=None, llm=None) -> str: + super().__init__(name, context, llm) + + def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): + if isinstance(prompt, str): + return system_msg + prompt if system_msg else prompt + + if isinstance(prompt, Message): + prompt.content = system_msg + prompt.content if system_msg else prompt.content + return prompt + + if isinstance(prompt, list) and system_msg: + prompt.insert(0, {"role": "system", "content": system_msg}) + return prompt + + async def run( + self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None, **kwargs + ) -> Dict: + prompt = self.process_msg(prompt, system_msg) + code_content = await self.llm.aask_code(prompt, **kwargs) + return Message(content=code_content, role="assistant") diff --git a/metagpt/prompts/plan.py b/metagpt/prompts/plan.py new file mode 100644 index 000000000..c4b056ab0 --- /dev/null +++ b/metagpt/prompts/plan.py @@ -0,0 +1,7 @@ +TASK_PLAN_SYSTEM_MSG = """You are a {role}. Write a plan with single digits steps. make sure others can understand what you are doing. +Example: +# plan +1. ...\n\n +2. ...\n\n +... +""" diff --git a/metagpt/schema.py b/metagpt/schema.py index bdca093c2..4bada005a 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -30,6 +30,7 @@ class Message: sent_from: str = field(default="") send_to: str = field(default="") restricted_to: str = field(default="") + state: str = None # None, done, todo, doing, error def __str__(self): # prefix = '-'.join([self.role, str(self.cause_by)]) diff --git a/tests/metagpt/actions/test_code_executor.py b/tests/metagpt/actions/test_code_executor.py new file mode 100644 index 000000000..d1833b48c --- /dev/null +++ b/tests/metagpt/actions/test_code_executor.py @@ -0,0 +1,58 @@ +import pytest + +from metagpt.actions import PyCodeExecutor +from metagpt.schema import Message + + +@pytest.mark.asyncio +async def test_code_running(): + pi = PyCodeExecutor() + output = await pi.run("print('hello world!')") + assert output.state == "done" + output = await pi.run({"code": "print('hello world!')", "language": "python"}) + assert output.state == "done" + code_msg = Message("print('hello world!')") + setattr(code_msg, "language", "python") + output = await pi.run(code_msg) + assert output.state == "done" + + +@pytest.mark.asyncio +async def test_split_code_running(): + pi = PyCodeExecutor() + output = await pi.run("x=1\ny=2") + output = await pi.run("z=x+y") + output = await pi.run("assert z==3") + assert output.state == "done" + + +@pytest.mark.asyncio +async def test_execute_error(): + pi = PyCodeExecutor() + output = await pi.run("z=1/0") + assert output.state == "error" + + +@pytest.mark.asyncio +async def test_plotting_code(): + pi = PyCodeExecutor() + code = """ + import numpy as np + import matplotlib.pyplot as plt + + # 生成随机数据 + random_data = np.random.randn(1000) # 生成1000个符合标准正态分布的随机数 + + # 绘制直方图 + plt.hist(random_data, bins=30, density=True, alpha=0.7, color='blue', edgecolor='black') + + # 添加标题和标签 + plt.title('Histogram of Random Data') + plt.xlabel('Value') + plt.ylabel('Frequency') + + # 显示图形 + plt.show() + """ + output = await pi.run(code) + assert output.state == "done" diff --git a/tests/metagpt/actions/test_plan.py b/tests/metagpt/actions/test_plan.py new file mode 100644 index 000000000..35f8f20cc --- /dev/null +++ b/tests/metagpt/actions/test_plan.py @@ -0,0 +1,12 @@ +import pytest + +from metagpt.actions.plan import Plan + + +@pytest.mark.asyncio +async def test_plan(): + p = Plan() + task_desc = """Here’s some background information on Cyclistic, a bike-sharing company designing a marketing strategy aimed at converting casual riders into annual members: So far, Cyclistic’s marketing strategy has relied on building general awareness and engaging a wide range of consumers. group. One way to help achieve these goals is the flexibility of its pricing plans: one-way passes, full-day passes, and annual memberships. Customers who purchase a one-way or full-day pass are known as recreational riders. Customers purchasing an annual membership are Cyclistic members. I will provide you with a data sheet that records user behavior: '/Users/vicis/Downloads/202103-divvy-tripdata.csv""" + rsp = await p.run(task_desc, role="data analyst") + assert len(rsp.content) > 0 + assert rsp.sent_from == "Plan" diff --git a/tests/metagpt/actions/test_write_code_v2.py b/tests/metagpt/actions/test_write_code_v2.py new file mode 100644 index 000000000..929407051 --- /dev/null +++ b/tests/metagpt/actions/test_write_code_v2.py @@ -0,0 +1,22 @@ +import pytest + +from metagpt.actions.write_code_v2 import WriteCode + + +@pytest.mark.asyncio +async def test_write_code(): + coder = WriteCode() + code = await coder.run("Write a hello world code.") + assert "language" in code.content + assert "code" in code.content + print(code) + + +@pytest.mark.asyncio +async def test_write_code_by_list_prompt(): + coder = WriteCode() + msg = ["a=[1,2,5,10,-10]", "写出求a中最大值的代码python"] + code = await coder.run(msg) + assert "language" in code.content + assert "code" in code.content + print(code) From 50f64ca934d13072910989c20769e740920f7d7d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 21 Nov 2023 19:14:52 +0800 Subject: [PATCH 004/637] doc: add rich. --- requirements.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/requirements.txt b/requirements.txt index f0169d7fa..53176bd0a 100644 --- a/requirements.txt +++ b/requirements.txt @@ -45,3 +45,4 @@ semantic-kernel==0.3.13.dev0 wrapt==1.15.0 websocket-client==0.58.0 zhipuai==1.0.7 +rich==13.6.0 \ No newline at end of file From fa400a0b0d43d59afdea037430cc7fac57e34634 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 21 Nov 2023 19:15:16 +0800 Subject: [PATCH 005/637] chore: rename WriteCode -> WriteCodeFunction. --- metagpt/actions/__init__.py | 2 +- .../actions/{write_code_v2.py => write_code_function.py} | 2 +- .../{test_write_code_v2.py => test_write_code_function.py} | 6 +++--- 3 files changed, 5 insertions(+), 5 deletions(-) rename metagpt/actions/{write_code_v2.py => write_code_function.py} (97%) rename tests/metagpt/actions/{test_write_code_v2.py => test_write_code_function.py} (78%) diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index d7afae2fe..d0163c24e 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -24,7 +24,7 @@ from metagpt.actions.write_prd import WritePRD from metagpt.actions.write_prd_review import WritePRDReview from metagpt.actions.write_test import WriteTest from metagpt.actions.code_executor import PyCodeExecutor -from metagpt.actions.write_code_v2 import WriteCode as WriteCodeFunction +from metagpt.actions.write_code_function import WriteCodeFunction from metagpt.actions.plan import Plan diff --git a/metagpt/actions/write_code_v2.py b/metagpt/actions/write_code_function.py similarity index 97% rename from metagpt/actions/write_code_v2.py rename to metagpt/actions/write_code_function.py index 335e70dc0..2d943176a 100644 --- a/metagpt/actions/write_code_v2.py +++ b/metagpt/actions/write_code_function.py @@ -10,7 +10,7 @@ from metagpt.actions import Action from metagpt.schema import Message -class WriteCode(Action): +class WriteCodeFunction(Action): """Use openai function to generate code.""" def __init__(self, name: str = "", context=None, llm=None) -> str: diff --git a/tests/metagpt/actions/test_write_code_v2.py b/tests/metagpt/actions/test_write_code_function.py similarity index 78% rename from tests/metagpt/actions/test_write_code_v2.py rename to tests/metagpt/actions/test_write_code_function.py index 929407051..0e57b4ced 100644 --- a/tests/metagpt/actions/test_write_code_v2.py +++ b/tests/metagpt/actions/test_write_code_function.py @@ -1,11 +1,11 @@ import pytest -from metagpt.actions.write_code_v2 import WriteCode +from metagpt.actions.write_code_function import WriteCodeFunction @pytest.mark.asyncio async def test_write_code(): - coder = WriteCode() + coder = WriteCodeFunction() code = await coder.run("Write a hello world code.") assert "language" in code.content assert "code" in code.content @@ -14,7 +14,7 @@ async def test_write_code(): @pytest.mark.asyncio async def test_write_code_by_list_prompt(): - coder = WriteCode() + coder = WriteCodeFunction() msg = ["a=[1,2,5,10,-10]", "写出求a中最大值的代码python"] code = await coder.run(msg) assert "language" in code.content From 8ef05bb19f2ebbea8df60dc93813957e19cedbac Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 22 Nov 2023 17:56:43 +0800 Subject: [PATCH 006/637] chore: prompt support Message type. --- metagpt/actions/plan.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/plan.py b/metagpt/actions/plan.py index d46783ba2..ab3963c72 100644 --- a/metagpt/actions/plan.py +++ b/metagpt/actions/plan.py @@ -4,6 +4,8 @@ @Author : orange-crow @File : plan.py """ +from typing import Union + from metagpt.actions import Action from metagpt.prompts.plan import TASK_PLAN_SYSTEM_MSG from metagpt.schema import Message @@ -13,8 +15,8 @@ class Plan(Action): def __init__(self, llm=None): super().__init__("", None, llm) - async def run(self, prompt: str, role: str = None, system_msg: str = None) -> str: + async def run(self, prompt: Union[str, Message], role: str = None, system_msg: str = None) -> str: if role: system_msg = TASK_PLAN_SYSTEM_MSG.format(role=role) - rsp = await self._aask(system_msg + prompt) + rsp = self._aask(system_msg + prompt.content) if isinstance(prompt, Message) else await self._aask(system_msg + prompt) return Message(rsp, role="assistant", sent_from=self.__class__.__name__) From 9f5108a4643bf4b27f6abe1b7543c02be937ec4b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 22 Nov 2023 19:36:21 +0800 Subject: [PATCH 007/637] chore: return plan by list. --- metagpt/actions/plan.py | 4 +++- metagpt/prompts/plan.py | 5 +++-- tests/metagpt/actions/test_plan.py | 1 + 3 files changed, 7 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/plan.py b/metagpt/actions/plan.py index ab3963c72..8bc575992 100644 --- a/metagpt/actions/plan.py +++ b/metagpt/actions/plan.py @@ -9,6 +9,7 @@ from typing import Union from metagpt.actions import Action from metagpt.prompts.plan import TASK_PLAN_SYSTEM_MSG from metagpt.schema import Message +from metagpt.utils.common import CodeParser class Plan(Action): @@ -19,4 +20,5 @@ class Plan(Action): if role: system_msg = TASK_PLAN_SYSTEM_MSG.format(role=role) rsp = self._aask(system_msg + prompt.content) if isinstance(prompt, Message) else await self._aask(system_msg + prompt) - return Message(rsp, role="assistant", sent_from=self.__class__.__name__) + plan = CodeParser.parse_code(None, rsp).split('\n\n') + return Message(plan, role="assistant", sent_from=self.__class__.__name__) diff --git a/metagpt/prompts/plan.py b/metagpt/prompts/plan.py index c4b056ab0..4d3add211 100644 --- a/metagpt/prompts/plan.py +++ b/metagpt/prompts/plan.py @@ -1,7 +1,8 @@ TASK_PLAN_SYSTEM_MSG = """You are a {role}. Write a plan with single digits steps. make sure others can understand what you are doing. -Example: -# plan +Example, must start with ```, and end with ```: +``` 1. ...\n\n 2. ...\n\n ... +``` """ diff --git a/tests/metagpt/actions/test_plan.py b/tests/metagpt/actions/test_plan.py index 35f8f20cc..1b1b90513 100644 --- a/tests/metagpt/actions/test_plan.py +++ b/tests/metagpt/actions/test_plan.py @@ -10,3 +10,4 @@ async def test_plan(): rsp = await p.run(task_desc, role="data analyst") assert len(rsp.content) > 0 assert rsp.sent_from == "Plan" + print(rsp) From 8a0a89e604241187fdd253851678c6a16a8a4bbd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 22 Nov 2023 20:33:57 +0800 Subject: [PATCH 008/637] fix: fix bug about message. --- metagpt/actions/write_code_function.py | 21 ++++++++++++++++++--- 1 file changed, 18 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/write_code_function.py b/metagpt/actions/write_code_function.py index 2d943176a..6fb7f535e 100644 --- a/metagpt/actions/write_code_function.py +++ b/metagpt/actions/write_code_function.py @@ -21,16 +21,31 @@ class WriteCodeFunction(Action): return system_msg + prompt if system_msg else prompt if isinstance(prompt, Message): - prompt.content = system_msg + prompt.content if system_msg else prompt.content + if isinstance(prompt.content, dict): + prompt.content = system_msg + str([(k, v) for k, v in prompt.content.items()])\ + if system_msg else prompt.content + else: + prompt.content = system_msg + prompt.content if system_msg else prompt.content return prompt + if isinstance(prompt, list): + _prompt = [] + for msg in prompt: + if isinstance(msg, Message) and isinstance(msg.content, dict): + msg.content = str([(k, v) for k, v in msg.content.items()]) + if isinstance(msg, Message): + msg = msg.to_dict() + _prompt.append(msg) + prompt = _prompt + if isinstance(prompt, list) and system_msg: - prompt.insert(0, {"role": "system", "content": system_msg}) + if system_msg not in prompt[0]['content']: + prompt[0]['content'] = system_msg + prompt[0]['content'] return prompt async def run( self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None, **kwargs - ) -> Dict: + ) -> Message: prompt = self.process_msg(prompt, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) return Message(content=code_content, role="assistant") From d8ddf1fcb0a516269c8a041ba2ca1a36931af87e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 22 Nov 2023 20:35:47 +0800 Subject: [PATCH 009/637] add new test for list plan. --- .../actions/test_write_code_function.py | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/tests/metagpt/actions/test_write_code_function.py b/tests/metagpt/actions/test_write_code_function.py index 0e57b4ced..cac459380 100644 --- a/tests/metagpt/actions/test_write_code_function.py +++ b/tests/metagpt/actions/test_write_code_function.py @@ -1,6 +1,7 @@ import pytest from metagpt.actions.write_code_function import WriteCodeFunction +from metagpt.actions.code_executor import PyCodeExecutor @pytest.mark.asyncio @@ -20,3 +21,21 @@ async def test_write_code_by_list_prompt(): assert "language" in code.content assert "code" in code.content print(code) + + +@pytest.mark.asyncio +async def test_write_code_by_list_plan(): + coder = WriteCodeFunction() + executor = PyCodeExecutor() + messages = [] + plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] + for task in plan: + print(f"\n任务: {task}\n\n") + messages.append(task) + code = await coder.run(messages) + messages.append(code) + assert "language" in code.content + assert "code" in code.content + output = await executor.run(code) + print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") + messages.append(output) From 7b94c04f51b5cca61143ddad67dbc87a40fc2fc8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 22 Nov 2023 20:36:51 +0800 Subject: [PATCH 010/637] fix: return string in parse_outputs. --- metagpt/actions/code_executor.py | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/metagpt/actions/code_executor.py b/metagpt/actions/code_executor.py index c05c00c9c..0b4f5171f 100644 --- a/metagpt/actions/code_executor.py +++ b/metagpt/actions/code_executor.py @@ -83,7 +83,7 @@ class PyCodeExecutor(CodeExecutor, Action): def parse_outputs(self, outputs: List) -> str: assert isinstance(outputs, list) - parsed_output = {"text": [], "image": []} + parsed_output = "" # empty outputs: such as 'x=1\ny=2' if not outputs: @@ -91,11 +91,12 @@ class PyCodeExecutor(CodeExecutor, Action): for output in outputs: if output["output_type"] == "stream": - parsed_output["text"].append(output["text"]) + parsed_output += output["text"] elif output["output_type"] == "display_data": self.show_bytes_figure(output["data"]["image/png"], self.interaction) - parsed_output["image"].append(output["data"]["image/png"]) - return str(parsed_output) + elif output["output_type"] == "execute_result": + parsed_output += output["data"]["text/plain"] + return parsed_output def show_bytes_figure(self, image_base64: str, interaction_type: str = "ipython"): import base64 @@ -139,8 +140,8 @@ class PyCodeExecutor(CodeExecutor, Action): assert "language" in code code, language = code["code"], code["language"] elif isinstance(code, Message): - assert hasattr(code, "language") - code, language = code.content, code.language + assert "language" in code.content + code, language = code.content["code"], code.content["language"] else: raise ValueError(f"Not support code type {type(code).__name__}.") From a0b13c8e0ff4b4647585780f75644aff3b64471e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 23 Nov 2023 10:45:40 +0800 Subject: [PATCH 011/637] chore: change name. --- metagpt/actions/__init__.py | 8 ++++---- .../{code_executor.py => execute_code.py} | 15 ++++++++++----- metagpt/actions/{plan.py => write_plan.py} | 2 +- ...t_code_executor.py => test_execute_code.py} | 11 +++++------ .../actions/test_write_code_function.py | 18 +++++++++--------- .../{test_plan.py => test_write_plan.py} | 6 +++--- 6 files changed, 32 insertions(+), 28 deletions(-) rename metagpt/actions/{code_executor.py => execute_code.py} (92%) rename metagpt/actions/{plan.py => write_plan.py} (97%) rename tests/metagpt/actions/{test_code_executor.py => test_execute_code.py} (87%) rename tests/metagpt/actions/{test_plan.py => test_write_plan.py} (88%) diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index d0163c24e..ba2170cbd 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -23,9 +23,9 @@ from metagpt.actions.write_code_review import WriteCodeReview from metagpt.actions.write_prd import WritePRD from metagpt.actions.write_prd_review import WritePRDReview from metagpt.actions.write_test import WriteTest -from metagpt.actions.code_executor import PyCodeExecutor +from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.write_code_function import WriteCodeFunction -from metagpt.actions.plan import Plan +from metagpt.actions.write_plan import WritePlan class ActionType(Enum): @@ -48,9 +48,9 @@ class ActionType(Enum): COLLECT_LINKS = CollectLinks WEB_BROWSE_AND_SUMMARIZE = WebBrowseAndSummarize CONDUCT_RESEARCH = ConductResearch - PYCODE_EXECUTOR = PyCodeExecutor + EXECUTE_PYCODE = ExecutePyCode WRITE_CODE_FUNCTION = WriteCodeFunction - PLAN = Plan + WRITE_PLAN = WritePlan __all__ = [ diff --git a/metagpt/actions/code_executor.py b/metagpt/actions/execute_code.py similarity index 92% rename from metagpt/actions/code_executor.py rename to metagpt/actions/execute_code.py index 0b4f5171f..e80886c3e 100644 --- a/metagpt/actions/code_executor.py +++ b/metagpt/actions/execute_code.py @@ -18,7 +18,7 @@ from metagpt.actions import Action from metagpt.schema import Message -class CodeExecutor(ABC): +class ExecuteCode(ABC): @abstractmethod async def build(self): """build code executor""" @@ -40,7 +40,7 @@ class CodeExecutor(ABC): ... -class PyCodeExecutor(CodeExecutor, Action): +class ExecutePyCode(ExecuteCode, Action): """execute code, return result to llm, and display it.""" def __init__(self, name: str = "python_executor", context=None, llm=None): @@ -128,6 +128,7 @@ class PyCodeExecutor(CodeExecutor, Action): return False def _process_code(self, code: Union[str, Dict, Message], language: str = None) -> Tuple: + language = language or 'python' if isinstance(code, str) and Path(code).suffix in (".py", ".txt"): code = Path(code).read_text(encoding="utf-8") return code, language @@ -137,11 +138,15 @@ class PyCodeExecutor(CodeExecutor, Action): if isinstance(code, dict): assert "code" in code - assert "language" in code + if "language" not in code: + code['language'] = 'python' code, language = code["code"], code["language"] elif isinstance(code, Message): - assert "language" in code.content - code, language = code.content["code"], code.content["language"] + if isinstance(code.content, dict) and "language" not in code.content: + code.content["language"] = 'python' + code, language = code.content["code"], code.content["language"] + elif isinstance(code.content, str): + code, language = code.content, language else: raise ValueError(f"Not support code type {type(code).__name__}.") diff --git a/metagpt/actions/plan.py b/metagpt/actions/write_plan.py similarity index 97% rename from metagpt/actions/plan.py rename to metagpt/actions/write_plan.py index 8bc575992..96d15cb84 100644 --- a/metagpt/actions/plan.py +++ b/metagpt/actions/write_plan.py @@ -12,7 +12,7 @@ from metagpt.schema import Message from metagpt.utils.common import CodeParser -class Plan(Action): +class WritePlan(Action): def __init__(self, llm=None): super().__init__("", None, llm) diff --git a/tests/metagpt/actions/test_code_executor.py b/tests/metagpt/actions/test_execute_code.py similarity index 87% rename from tests/metagpt/actions/test_code_executor.py rename to tests/metagpt/actions/test_execute_code.py index d1833b48c..88c5adf18 100644 --- a/tests/metagpt/actions/test_code_executor.py +++ b/tests/metagpt/actions/test_execute_code.py @@ -1,25 +1,24 @@ import pytest -from metagpt.actions import PyCodeExecutor +from metagpt.actions import ExecutePyCode from metagpt.schema import Message @pytest.mark.asyncio async def test_code_running(): - pi = PyCodeExecutor() + pi = ExecutePyCode() output = await pi.run("print('hello world!')") assert output.state == "done" output = await pi.run({"code": "print('hello world!')", "language": "python"}) assert output.state == "done" code_msg = Message("print('hello world!')") - setattr(code_msg, "language", "python") output = await pi.run(code_msg) assert output.state == "done" @pytest.mark.asyncio async def test_split_code_running(): - pi = PyCodeExecutor() + pi = ExecutePyCode() output = await pi.run("x=1\ny=2") output = await pi.run("z=x+y") output = await pi.run("assert z==3") @@ -28,14 +27,14 @@ async def test_split_code_running(): @pytest.mark.asyncio async def test_execute_error(): - pi = PyCodeExecutor() + pi = ExecutePyCode() output = await pi.run("z=1/0") assert output.state == "error" @pytest.mark.asyncio async def test_plotting_code(): - pi = PyCodeExecutor() + pi = ExecutePyCode() code = """ import numpy as np import matplotlib.pyplot as plt diff --git a/tests/metagpt/actions/test_write_code_function.py b/tests/metagpt/actions/test_write_code_function.py index cac459380..4ff1a63c4 100644 --- a/tests/metagpt/actions/test_write_code_function.py +++ b/tests/metagpt/actions/test_write_code_function.py @@ -1,13 +1,13 @@ import pytest from metagpt.actions.write_code_function import WriteCodeFunction -from metagpt.actions.code_executor import PyCodeExecutor +from metagpt.actions.execute_code import ExecutePyCode @pytest.mark.asyncio async def test_write_code(): - coder = WriteCodeFunction() - code = await coder.run("Write a hello world code.") + write_code = WriteCodeFunction() + code = await write_code.run("Write a hello world code.") assert "language" in code.content assert "code" in code.content print(code) @@ -15,9 +15,9 @@ async def test_write_code(): @pytest.mark.asyncio async def test_write_code_by_list_prompt(): - coder = WriteCodeFunction() + write_code = WriteCodeFunction() msg = ["a=[1,2,5,10,-10]", "写出求a中最大值的代码python"] - code = await coder.run(msg) + code = await write_code.run(msg) assert "language" in code.content assert "code" in code.content print(code) @@ -25,17 +25,17 @@ async def test_write_code_by_list_prompt(): @pytest.mark.asyncio async def test_write_code_by_list_plan(): - coder = WriteCodeFunction() - executor = PyCodeExecutor() + write_code = WriteCodeFunction() + execute_code = ExecutePyCode() messages = [] plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] for task in plan: print(f"\n任务: {task}\n\n") messages.append(task) - code = await coder.run(messages) + code = await write_code.run(messages) messages.append(code) assert "language" in code.content assert "code" in code.content - output = await executor.run(code) + output = await execute_code.run(code) print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") messages.append(output) diff --git a/tests/metagpt/actions/test_plan.py b/tests/metagpt/actions/test_write_plan.py similarity index 88% rename from tests/metagpt/actions/test_plan.py rename to tests/metagpt/actions/test_write_plan.py index 1b1b90513..2bf200ab3 100644 --- a/tests/metagpt/actions/test_plan.py +++ b/tests/metagpt/actions/test_write_plan.py @@ -1,13 +1,13 @@ import pytest -from metagpt.actions.plan import Plan +from metagpt.actions.write_plan import WritePlan @pytest.mark.asyncio async def test_plan(): - p = Plan() + p = WritePlan() task_desc = """Here’s some background information on Cyclistic, a bike-sharing company designing a marketing strategy aimed at converting casual riders into annual members: So far, Cyclistic’s marketing strategy has relied on building general awareness and engaging a wide range of consumers. group. One way to help achieve these goals is the flexibility of its pricing plans: one-way passes, full-day passes, and annual memberships. Customers who purchase a one-way or full-day pass are known as recreational riders. Customers purchasing an annual membership are Cyclistic members. I will provide you with a data sheet that records user behavior: '/Users/vicis/Downloads/202103-divvy-tripdata.csv""" rsp = await p.run(task_desc, role="data analyst") assert len(rsp.content) > 0 - assert rsp.sent_from == "Plan" + assert rsp.sent_from == "WritePlan" print(rsp) From 3d18dfe2b582f16cf08f6b4e23eea56e85ee1c59 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 23 Nov 2023 21:59:25 +0800 Subject: [PATCH 012/637] pipeline first version --- metagpt/actions/execute_code.py | 9 +- metagpt/actions/write_code_function.py | 22 +++-- metagpt/actions/write_plan.py | 46 ++++++++--- metagpt/prompts/plan.py | 8 -- metagpt/roles/ml_engineer.py | 110 +++++++++++++++++++++++++ metagpt/schema.py | 109 ++++++++++++++++++++++++ requirements.txt | 6 +- tests/metagpt/test_schema.py | 85 +++++++++++++++++++ 8 files changed, 362 insertions(+), 33 deletions(-) delete mode 100644 metagpt/prompts/plan.py create mode 100644 metagpt/roles/ml_engineer.py diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index e80886c3e..7b16d559a 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -7,6 +7,7 @@ from abc import ABC, abstractmethod from pathlib import Path from typing import Dict, List, Tuple, Union +import traceback import nbformat from nbclient import NotebookClient @@ -152,7 +153,7 @@ class ExecutePyCode(ExecuteCode, Action): return code, language - async def run(self, code: Union[str, Dict, Message], language: str = "python") -> Message: + async def run(self, code: Union[str, Dict, Message], language: str = "python") -> Tuple[str, bool]: code, language = self._process_code(code, language) self._display(code, language) @@ -167,13 +168,11 @@ class ExecutePyCode(ExecuteCode, Action): # TODO: add max_tries for run code. cell_index = len(self.nb.cells) - 1 await self.nb_client.async_execute_cell(self.nb.cells[-1], cell_index) - return Message( - self.parse_outputs(self.nb.cells[-1].outputs), state="done", sent_from=self.__class__.__name__ - ) + return self.parse_outputs(self.nb.cells[-1].outputs), True except Exception as e: # FIXME: CellExecutionError is hard to read. for example `1\0` raise ZeroDivisionError: # CellExecutionError('An error occurred while executing the following cell:\n------------------\nz=1/0\n------------------\n\n\n\x1b[0;31m---------------------------------------------------------------------------\x1b[0m\n\x1b[0;31mZeroDivisionError\x1b[0m Traceback (most recent call last)\nCell \x1b[0;32mIn[1], line 1\x1b[0m\n\x1b[0;32m----> 1\x1b[0m z\x1b[38;5;241m=\x1b[39m\x1b[38;5;241;43m1\x1b[39;49m\x1b[38;5;241;43m/\x1b[39;49m\x1b[38;5;241;43m0\x1b[39;49m\n\n\x1b[0;31mZeroDivisionError\x1b[0m: division by zero\n') - return Message(e, state="error", sent_from=self.__class__.__name__) + return traceback.format_exc(), False else: # TODO: markdown raise NotImplementedError(f"Not support this code type : {language}, Only support code!") diff --git a/metagpt/actions/write_code_function.py b/metagpt/actions/write_code_function.py index 6fb7f535e..4ec565eb1 100644 --- a/metagpt/actions/write_code_function.py +++ b/metagpt/actions/write_code_function.py @@ -7,10 +7,20 @@ from typing import Dict, List, Union from metagpt.actions import Action -from metagpt.schema import Message +from metagpt.schema import Message, Plan +class BaseWriteAnalysisCode(Action): -class WriteCodeFunction(Action): + async def run(self, context: List[Message], plan: Plan = None, task_guidance: str = ""): + """Run of a code writing action, used in data analysis or modeling + + Args: + context (List[Message]): Action output history, source action denoted by Message.cause_by + plan (Plan, optional): Overall plan. Defaults to None. + task_guidance (str, optional): suggested step breakdown for the current task. Defaults to "". + """ + +class WriteCodeFunction(BaseWriteAnalysisCode): """Use openai function to generate code.""" def __init__(self, name: str = "", context=None, llm=None) -> str: @@ -44,8 +54,8 @@ class WriteCodeFunction(Action): return prompt async def run( - self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None, **kwargs - ) -> Message: - prompt = self.process_msg(prompt, system_msg) + self, context: [List[Message]], plan: Plan = None, task_guidance: str = "", system_msg: str = None, **kwargs + ) -> str: + prompt = self.process_msg(context, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) - return Message(content=code_content, role="assistant") + return code_content diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index 96d15cb84..48cb1aad5 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -4,21 +4,41 @@ @Author : orange-crow @File : plan.py """ -from typing import Union +from typing import List +import json from metagpt.actions import Action -from metagpt.prompts.plan import TASK_PLAN_SYSTEM_MSG -from metagpt.schema import Message -from metagpt.utils.common import CodeParser - +from metagpt.schema import Message, Task class WritePlan(Action): - def __init__(self, llm=None): - super().__init__("", None, llm) + PROMPT_TEMPLATE = """ + # Context: + __context__ + # Current Plan: + __current_plan__ + # Task: + Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to __max_tasks__ tasks. + If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. + Output a list of jsons following the format: + [ + { + "task_id": str = "unique identifier for a task in plan, can be a ordinal", + "dependent_task_ids": list[str] = "ids of tasks prerequisite to this task", + "instruction": "what you should do in this task, one short phrase or sentence", + }, + ... + ] + """ + async def run(self, context: List[Message], current_plan: str = "", max_tasks: int = 5) -> str: + prompt = ( + self.PROMPT_TEMPLATE.replace("__context__", "\n".join([str(ct) for ct in context])) + .replace("__current_plan__", current_plan).replace("__max_tasks__", str(max_tasks)) + ) + rsp = await self._aask(prompt) + return rsp - async def run(self, prompt: Union[str, Message], role: str = None, system_msg: str = None) -> str: - if role: - system_msg = TASK_PLAN_SYSTEM_MSG.format(role=role) - rsp = self._aask(system_msg + prompt.content) if isinstance(prompt, Message) else await self._aask(system_msg + prompt) - plan = CodeParser.parse_code(None, rsp).split('\n\n') - return Message(plan, role="assistant", sent_from=self.__class__.__name__) + @staticmethod + def rsp_to_tasks(rsp: str) -> List[Task]: + rsp = json.loads(rsp) + tasks = [Task(**task_config) for task_config in rsp] + return tasks diff --git a/metagpt/prompts/plan.py b/metagpt/prompts/plan.py deleted file mode 100644 index 4d3add211..000000000 --- a/metagpt/prompts/plan.py +++ /dev/null @@ -1,8 +0,0 @@ -TASK_PLAN_SYSTEM_MSG = """You are a {role}. Write a plan with single digits steps. make sure others can understand what you are doing. -Example, must start with ```, and end with ```: -``` -1. ...\n\n -2. ...\n\n -... -``` -""" diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py new file mode 100644 index 000000000..c795bda11 --- /dev/null +++ b/metagpt/roles/ml_engineer.py @@ -0,0 +1,110 @@ +from typing import Dict, List, Union +import json +import subprocess + +import fire + +from metagpt.roles import Role +from metagpt.actions import Action +from metagpt.schema import Message, Task, Plan +from metagpt.logs import logger +from metagpt.actions.write_plan import WritePlan +from metagpt.actions.write_code_function import WriteCodeFunction +from metagpt.actions.execute_code import ExecutePyCode + +class AskReview(Action): + + async def run(self, context: List[Message], plan: Plan = None): + prompt = "\n".join( + [f"{msg.cause_by() if msg.cause_by else 'Main Requirement'}: {msg.content}" for msg in context] + ) + + latest_action = context[-1].cause_by() + + prompt += f"\nPlease review output from {latest_action}, " \ + "provide feedback or type YES to continue with the process:\n" + rsp = input(prompt) + confirmed = "yes" in rsp.lower() + return rsp, confirmed + + +class MLEngineer(Role): + def __init__(self, name="ABC", profile="MLEngineer"): + super().__init__(name=name, profile=profile) + self._set_react_mode(react_mode="plan_and_act") + self.plan = Plan() + + async def _plan_and_act(self): + + # create initial plan and update until confirmation + await self._update_plan() + + while self.plan.current_task: + task = self.plan.current_task + logger.info(f"ready to take on task {task}") + + # take on current task + code, result, success = await self._write_and_exec_code() + + # ask for acceptance, users can other refuse and change tasks in the plan + task_result_confirmed = await self._ask_review() + + if success and task_result_confirmed: + # tick off this task and record progress + task.code = code + task.result = result + self.plan.finish_current_task() + + else: + # update plan according to user's feedback and to take on changed tasks + await self._update_plan() + + async def _write_and_exec_code(self, max_retry: int = 3): + counter = 0 + success = False + while not success and counter < max_retry: + context = self.get_memories() + + code = "print('abc')" + # code = await WriteCodeFunction().run(context=context) + # code = await WriteCodeWithOps.run(context, task, result) + self._rc.memory.add(Message(content=code, role="assistant", cause_by=WriteCodeFunction)) + + result, success = await ExecutePyCode().run(code) + self._rc.memory.add(Message(content=result, role="assistant", cause_by=ExecutePyCode)) + + # if not success: + # await self._ask_review() + + counter += 1 + + return code, result, success + + async def _ask_review(self): + context = self.get_memories() + review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan) + self._rc.memory.add(Message(content=review, role="assistant", cause_by=AskReview)) + return confirmed + + async def _update_plan(self, max_tasks: int = 3): + current_plan = str([task.json() for task in self.plan.tasks]) + plan_confirmed = False + while not plan_confirmed: + context = self.get_memories() + rsp = await WritePlan().run(context, current_plan=current_plan, max_tasks=max_tasks) + self._rc.memory.add(Message(content=rsp, role="assistant", cause_by=WritePlan)) + plan_confirmed = await self._ask_review() + + tasks = WritePlan.rsp_to_tasks(rsp) + self.plan.add_tasks(tasks) + + +if __name__ == "__main__": + # requirement = "create a normal distribution and visualize it" + requirement = "run some analysis on iris dataset" + + async def main(requirement: str = requirement): + role = MLEngineer() + await role.run(requirement) + + fire.Fire(main) diff --git a/metagpt/schema.py b/metagpt/schema.py index 4bada005a..3cd7d9730 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -73,6 +73,115 @@ class AIMessage(Message): super().__init__(content, 'assistant') +class Task(BaseModel): + task_id: str = "" + dependent_task_ids: list[str] = [] # Tasks prerequisite to this Task + instruction: str = "" + task_type: str = "" + code: str = "" + result: str = "" + is_finished: bool = False + + +class Plan(BaseModel): + tasks: list[Task] = [] + task_map: dict[str, Task] = {} + current_task_id = "" + + def _topological_sort(self, tasks: list[Task]): + task_map = {task.task_id: task for task in tasks} + dependencies = {task.task_id: set(task.dependent_task_ids) for task in tasks} + sorted_tasks = [] + visited = set() + + def visit(task_id): + if task_id in visited: + return + visited.add(task_id) + for dependent_id in dependencies.get(task_id, []): + visit(dependent_id) + sorted_tasks.append(task_map[task_id]) + + for task in tasks: + visit(task.task_id) + + return sorted_tasks + + def add_tasks(self, tasks: list[Task]): + """ + Integrates new tasks into the existing plan, ensuring dependency order is maintained. + + This method performs two primary functions based on the current state of the task list: + 1. If there are no existing tasks, it topologically sorts the provided tasks to ensure + correct execution order based on dependencies, and sets these as the current tasks. + 2. If there are existing tasks, it merges the new tasks with the existing ones. It maintains + any common prefix of tasks (based on task_id and instruction) and appends the remainder + of the new tasks. The current task is updated to the first unfinished task in this merged list. + + Args: + tasks (list[Task]): A list of tasks (may be unordered) to add to the plan. + + Returns: + None: The method updates the internal state of the plan but does not return anything. + """ + if not tasks: + return + + # Topologically sort the new tasks to ensure correct dependency order + new_tasks = self._topological_sort(tasks) + + if not self.tasks: + # If there are no existing tasks, set the new tasks as the current tasks + self.tasks = new_tasks + + else: + # Find the length of the common prefix between existing and new tasks + prefix_length = 0 + for old_task, new_task in zip(self.tasks, new_tasks): + if old_task.task_id != new_task.task_id or old_task.instruction != new_task.instruction: + break + prefix_length += 1 + + # Combine the common prefix with the remainder of the new tasks + final_tasks = self.tasks[:prefix_length] + new_tasks[prefix_length:] + self.tasks = final_tasks + + # Update current_task_id to the first unfinished task in the merged list + for task in self.tasks: + if not task.is_finished: + self.current_task_id = task.task_id + break + + # Update the task map for quick access to tasks by ID + self.task_map = {task.task_id: task for task in self.tasks} + + @property + def current_task(self) -> Task: + """Find current task to execute + + Returns: + Task: the current task to be executed + """ + return self.task_map.get(self.current_task_id, None) + + def finish_current_task(self): + """Finish current task, set Task.is_finished=True, set current task to next task + """ + if self.current_task_id: + current_task = self.current_task + current_task.is_finished = True + next_task_index = self.tasks.index(current_task) + 1 + self.current_task_id = self.tasks[next_task_index].task_id if next_task_index < len(self.tasks) else None + + def get_finished_tasks(self) -> list[Task]: + """return all finished tasks in correct linearized order + + Returns: + list[Task]: list of finished tasks + """ + return [task for task in self.tasks if task.is_finished] + + if __name__ == '__main__': test_content = 'test_message' msgs = [ diff --git a/requirements.txt b/requirements.txt index 53176bd0a..c0f466457 100644 --- a/requirements.txt +++ b/requirements.txt @@ -45,4 +45,8 @@ semantic-kernel==0.3.13.dev0 wrapt==1.15.0 websocket-client==0.58.0 zhipuai==1.0.7 -rich==13.6.0 \ No newline at end of file +rich==13.6.0 +nbclient==0.9.0 +nbformat==5.9.2 +ipython==8.17.2 +ipykernel==6.27.0 \ No newline at end of file diff --git a/tests/metagpt/test_schema.py b/tests/metagpt/test_schema.py index 12666e0d3..6aae82006 100644 --- a/tests/metagpt/test_schema.py +++ b/tests/metagpt/test_schema.py @@ -6,6 +6,7 @@ @File : test_schema.py """ from metagpt.schema import AIMessage, Message, SystemMessage, UserMessage +from metagpt.schema import Task, Plan def test_messages(): @@ -19,3 +20,87 @@ def test_messages(): text = str(msgs) roles = ['user', 'system', 'assistant', 'QA'] assert all([i in text for i in roles]) + + +class TestPlan: + def test_add_tasks_ordering(self): + plan = Plan() + + tasks = [ + Task(task_id="1", dependent_task_ids=["2", "3"], instruction="Third"), + Task(task_id="2", instruction="First"), + Task(task_id="3", dependent_task_ids=["2"], instruction="Second") + ] # 2 -> 3 -> 1 + plan.add_tasks(tasks) + + assert [task.task_id for task in plan.tasks] == ["2", "3", "1"] + + def test_add_tasks_to_existing_no_common_prefix(self): + plan = Plan() + + tasks = [ + Task(task_id="1", dependent_task_ids=["2", "3"], instruction="Third"), + Task(task_id="2", instruction="First"), + Task(task_id="3", dependent_task_ids=["2"], instruction="Second", is_finished=True) + ] # 2 -> 3 -> 1 + plan.add_tasks(tasks) + + new_tasks = [Task(task_id="3", instruction="")] + plan.add_tasks(new_tasks) + + assert [task.task_id for task in plan.tasks] == ["3"] + assert not plan.tasks[0].is_finished # must be the new unfinished task + + def test_add_tasks_to_existing_with_common_prefix(self): + plan = Plan() + + tasks = [ + Task(task_id="1", dependent_task_ids=["2", "3"], instruction="Third"), + Task(task_id="2", instruction="First"), + Task(task_id="3", dependent_task_ids=["2"], instruction="Second") + ] # 2 -> 3 -> 1 + plan.add_tasks(tasks) + plan.finish_current_task() # finish 2 + plan.finish_current_task() # finish 3 + + new_tasks = [ + Task(task_id="4", dependent_task_ids=["3"], instruction="Third"), + Task(task_id="2", instruction="First"), + Task(task_id="3", dependent_task_ids=["2"], instruction="Second") + ] # 2 -> 3 -> 4, so the common prefix is 2 -> 3, and these two should be obtained from the existing tasks + plan.add_tasks(new_tasks) + + assert [task.task_id for task in plan.tasks] == ["2", "3", "4"] + assert plan.tasks[0].is_finished and plan.tasks[1].is_finished # "2" and "3" should be the original finished one + assert plan.current_task_id == "4" + + def test_current_task(self): + plan = Plan() + tasks = [ + Task(task_id="1", dependent_task_ids=["2"], instruction="Second"), + Task(task_id="2", instruction="First") + ] + plan.add_tasks(tasks) + assert plan.current_task.task_id == "2" + + def test_finish_task(self): + plan = Plan() + tasks = [ + Task(task_id="1", instruction="First"), + Task(task_id="2", dependent_task_ids=["1"], instruction="Second") + ] + plan.add_tasks(tasks) + plan.finish_current_task() + assert plan.current_task.task_id == "2" + + def test_finished_tasks(self): + plan = Plan() + tasks = [ + Task(task_id="1", instruction="First"), + Task(task_id="2", dependent_task_ids=["1"], instruction="Second") + ] + plan.add_tasks(tasks) + plan.finish_current_task() + finished_tasks = plan.get_finished_tasks() + assert len(finished_tasks) == 1 + assert finished_tasks[0].task_id == "1" From 824cc247b66a385a91ff32367ec0d67936543630 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Fri, 24 Nov 2023 11:05:51 +0800 Subject: [PATCH 013/637] chore --- metagpt/actions/write_code_function.py | 2 +- metagpt/actions/write_plan.py | 3 +- metagpt/roles/ml_engineer.py | 10 ++--- .../actions/test_write_code_function.py | 38 +++++++++---------- 4 files changed, 26 insertions(+), 27 deletions(-) diff --git a/metagpt/actions/write_code_function.py b/metagpt/actions/write_code_function.py index 4ec565eb1..406d215a2 100644 --- a/metagpt/actions/write_code_function.py +++ b/metagpt/actions/write_code_function.py @@ -58,4 +58,4 @@ class WriteCodeFunction(BaseWriteAnalysisCode): ) -> str: prompt = self.process_msg(context, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) - return code_content + return code_content['code'] diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index 48cb1aad5..8db988c01 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -9,6 +9,7 @@ import json from metagpt.actions import Action from metagpt.schema import Message, Task +from metagpt.utils.common import CodeParser class WritePlan(Action): PROMPT_TEMPLATE = """ @@ -35,7 +36,7 @@ class WritePlan(Action): .replace("__current_plan__", current_plan).replace("__max_tasks__", str(max_tasks)) ) rsp = await self._aask(prompt) - return rsp + return CodeParser.parse_code(None, rsp) if rsp.startswith("```") else rsp @staticmethod def rsp_to_tasks(rsp: str) -> List[Task]: diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index c795bda11..3bb0c1660 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -64,14 +64,14 @@ class MLEngineer(Role): success = False while not success and counter < max_retry: context = self.get_memories() - - code = "print('abc')" - # code = await WriteCodeFunction().run(context=context) + print(f"{'*'*20}\n {context}") + # code = "print('abc')" + code = await WriteCodeFunction().run(context=context) # code = await WriteCodeWithOps.run(context, task, result) self._rc.memory.add(Message(content=code, role="assistant", cause_by=WriteCodeFunction)) result, success = await ExecutePyCode().run(code) - self._rc.memory.add(Message(content=result, role="assistant", cause_by=ExecutePyCode)) + self._rc.memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) # if not success: # await self._ask_review() @@ -83,7 +83,7 @@ class MLEngineer(Role): async def _ask_review(self): context = self.get_memories() review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan) - self._rc.memory.add(Message(content=review, role="assistant", cause_by=AskReview)) + self._rc.memory.add(Message(content=review, role="user", cause_by=AskReview)) return confirmed async def _update_plan(self, max_tasks: int = 3): diff --git a/tests/metagpt/actions/test_write_code_function.py b/tests/metagpt/actions/test_write_code_function.py index 4ff1a63c4..1940c9667 100644 --- a/tests/metagpt/actions/test_write_code_function.py +++ b/tests/metagpt/actions/test_write_code_function.py @@ -2,25 +2,24 @@ import pytest from metagpt.actions.write_code_function import WriteCodeFunction from metagpt.actions.execute_code import ExecutePyCode +from metagpt.schema import Message -@pytest.mark.asyncio -async def test_write_code(): - write_code = WriteCodeFunction() - code = await write_code.run("Write a hello world code.") - assert "language" in code.content - assert "code" in code.content - print(code) +# @pytest.mark.asyncio +# async def test_write_code(): +# write_code = WriteCodeFunction() +# code = await write_code.run("Write a hello world code.") +# assert len(code) > 0 +# print(code) -@pytest.mark.asyncio -async def test_write_code_by_list_prompt(): - write_code = WriteCodeFunction() - msg = ["a=[1,2,5,10,-10]", "写出求a中最大值的代码python"] - code = await write_code.run(msg) - assert "language" in code.content - assert "code" in code.content - print(code) +# @pytest.mark.asyncio +# async def test_write_code_by_list_prompt(): +# write_code = WriteCodeFunction() +# msg = ["a=[1,2,5,10,-10]", "写出求a中最大值的代码python"] +# code = await write_code.run(msg) +# assert len(code) > 0 +# print(code) @pytest.mark.asyncio @@ -31,11 +30,10 @@ async def test_write_code_by_list_plan(): plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] for task in plan: print(f"\n任务: {task}\n\n") - messages.append(task) + messages.append(Message(task, role='assistant')) code = await write_code.run(messages) - messages.append(code) - assert "language" in code.content - assert "code" in code.content + messages.append(Message(code, role='assistant')) + assert len(code) > 0 output = await execute_code.run(code) print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") - messages.append(output) + messages.append(output[0]) From bba3db5ffe062f6b5cb849b701bdd9c11665bf85 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Fri, 24 Nov 2023 12:27:17 +0800 Subject: [PATCH 014/637] fix: --- metagpt/actions/write_code_function.py | 47 +++++++++++++------------- 1 file changed, 23 insertions(+), 24 deletions(-) diff --git a/metagpt/actions/write_code_function.py b/metagpt/actions/write_code_function.py index 406d215a2..1f273a707 100644 --- a/metagpt/actions/write_code_function.py +++ b/metagpt/actions/write_code_function.py @@ -27,31 +27,30 @@ class WriteCodeFunction(BaseWriteAnalysisCode): super().__init__(name, context, llm) def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): - if isinstance(prompt, str): - return system_msg + prompt if system_msg else prompt + default_system_msg = """You are Open Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.""" + # 全部转成list + if not isinstance(prompt, list): + prompt = [prompt] + assert isinstance(prompt, list) + # 转成list[dict] + messages = [] + for p in prompt: + if isinstance(p, str): + messages.append({'role': 'user', 'content': p}) + elif isinstance(p, dict): + messages.append(p) + elif isinstance(p, Message): + if isinstance(p.content, str): + messages.append(p.to_dict()) + elif isinstance(p.content, dict) and 'code' in p.content: + messages.append(p.content['code']) - if isinstance(prompt, Message): - if isinstance(prompt.content, dict): - prompt.content = system_msg + str([(k, v) for k, v in prompt.content.items()])\ - if system_msg else prompt.content - else: - prompt.content = system_msg + prompt.content if system_msg else prompt.content - return prompt - - if isinstance(prompt, list): - _prompt = [] - for msg in prompt: - if isinstance(msg, Message) and isinstance(msg.content, dict): - msg.content = str([(k, v) for k, v in msg.content.items()]) - if isinstance(msg, Message): - msg = msg.to_dict() - _prompt.append(msg) - prompt = _prompt - - if isinstance(prompt, list) and system_msg: - if system_msg not in prompt[0]['content']: - prompt[0]['content'] = system_msg + prompt[0]['content'] - return prompt + # 添加默认的提示词 + if default_system_msg not in messages[0]['content'] and messages[0]['role'] != 'system': + messages.insert(0, {'role': 'system', 'content': default_system_msg}) + elif default_system_msg not in messages[0]['content'] and messages[0]['role'] == 'system': + messages[0] = {'role': 'system', 'content': messages[0]['content']+default_system_msg} + return messages async def run( self, context: [List[Message]], plan: Plan = None, task_guidance: str = "", system_msg: str = None, **kwargs From 8b171b51337d5c416da9c52e78c0e0cbae138d82 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Fri, 24 Nov 2023 12:37:20 +0800 Subject: [PATCH 015/637] fix: write_code_function bug. --- metagpt/actions/write_code_function.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_code_function.py b/metagpt/actions/write_code_function.py index 1f273a707..c2fb6189a 100644 --- a/metagpt/actions/write_code_function.py +++ b/metagpt/actions/write_code_function.py @@ -27,7 +27,7 @@ class WriteCodeFunction(BaseWriteAnalysisCode): super().__init__(name, context, llm) def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): - default_system_msg = """You are Open Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.""" + default_system_msg = """You are Open Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step.**""" # 全部转成list if not isinstance(prompt, list): prompt = [prompt] From fdef9c8547d743d41116d8bcf16fb3dd38b13e2d Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 24 Nov 2023 14:05:11 +0800 Subject: [PATCH 016/637] add more components in pipeline --- metagpt/actions/__init__.py | 4 +- ...ode_function.py => write_analysis_code.py} | 22 ++++-- metagpt/actions/write_plan.py | 6 +- metagpt/roles/ml_engineer.py | 70 +++++++++++++------ metagpt/schema.py | 1 + ...unction.py => test_write_analysis_code.py} | 8 +-- tests/metagpt/test_schema.py | 12 ++-- 7 files changed, 82 insertions(+), 41 deletions(-) rename metagpt/actions/{write_code_function.py => write_analysis_code.py} (76%) rename tests/metagpt/actions/{test_write_code_function.py => test_write_analysis_code.py} (86%) diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index ba2170cbd..5055ce276 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -24,7 +24,7 @@ from metagpt.actions.write_prd import WritePRD from metagpt.actions.write_prd_review import WritePRDReview from metagpt.actions.write_test import WriteTest from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_code_function import WriteCodeFunction +from metagpt.actions.write_analysis_code import WriteCodeByGenerate from metagpt.actions.write_plan import WritePlan @@ -49,7 +49,7 @@ class ActionType(Enum): WEB_BROWSE_AND_SUMMARIZE = WebBrowseAndSummarize CONDUCT_RESEARCH = ConductResearch EXECUTE_PYCODE = ExecutePyCode - WRITE_CODE_FUNCTION = WriteCodeFunction + WRITE_CODE_BY_GENERATE = WriteCodeByGenerate WRITE_PLAN = WritePlan diff --git a/metagpt/actions/write_code_function.py b/metagpt/actions/write_analysis_code.py similarity index 76% rename from metagpt/actions/write_code_function.py rename to metagpt/actions/write_analysis_code.py index 4ec565eb1..84922ada4 100644 --- a/metagpt/actions/write_code_function.py +++ b/metagpt/actions/write_analysis_code.py @@ -11,17 +11,20 @@ from metagpt.schema import Message, Plan class BaseWriteAnalysisCode(Action): - async def run(self, context: List[Message], plan: Plan = None, task_guidance: str = ""): + async def run(self, context: List[Message], plan: Plan = None, task_guide: str = "") -> str: """Run of a code writing action, used in data analysis or modeling Args: context (List[Message]): Action output history, source action denoted by Message.cause_by plan (Plan, optional): Overall plan. Defaults to None. - task_guidance (str, optional): suggested step breakdown for the current task. Defaults to "". + task_guide (str, optional): suggested step breakdown for the current task. Defaults to "". + + Returns: + str: The code string. """ -class WriteCodeFunction(BaseWriteAnalysisCode): - """Use openai function to generate code.""" +class WriteCodeByGenerate(BaseWriteAnalysisCode): + """Write code fully by generation""" def __init__(self, name: str = "", context=None, llm=None) -> str: super().__init__(name, context, llm) @@ -54,8 +57,15 @@ class WriteCodeFunction(BaseWriteAnalysisCode): return prompt async def run( - self, context: [List[Message]], plan: Plan = None, task_guidance: str = "", system_msg: str = None, **kwargs + self, context: [List[Message]], plan: Plan = None, task_guide: str = "", system_msg: str = None, **kwargs ) -> str: prompt = self.process_msg(context, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) - return code_content + return code_content["code"] + + +class WriteCodeWithTools(BaseWriteAnalysisCode): + """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" + + async def run(self, context: List[Message], plan: Plan = None, task_guide: str = "") -> str: + return "print('abc')" diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index 48cb1aad5..e35ba7a92 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -9,6 +9,7 @@ import json from metagpt.actions import Action from metagpt.schema import Message, Task +from metagpt.utils.common import CodeParser class WritePlan(Action): PROMPT_TEMPLATE = """ @@ -20,14 +21,16 @@ class WritePlan(Action): Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to __max_tasks__ tasks. If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Output a list of jsons following the format: + ```json [ { - "task_id": str = "unique identifier for a task in plan, can be a ordinal", + "task_id": str = "unique identifier for a task in plan, can be an ordinal", "dependent_task_ids": list[str] = "ids of tasks prerequisite to this task", "instruction": "what you should do in this task, one short phrase or sentence", }, ... ] + ``` """ async def run(self, context: List[Message], current_plan: str = "", max_tasks: int = 5) -> str: prompt = ( @@ -35,6 +38,7 @@ class WritePlan(Action): .replace("__current_plan__", current_plan).replace("__max_tasks__", str(max_tasks)) ) rsp = await self._aask(prompt) + rsp = CodeParser.parse_code(block=None, text=rsp) return rsp @staticmethod diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index c795bda11..480f6cecf 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -9,30 +9,41 @@ from metagpt.actions import Action from metagpt.schema import Message, Task, Plan from metagpt.logs import logger from metagpt.actions.write_plan import WritePlan -from metagpt.actions.write_code_function import WriteCodeFunction +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.execute_code import ExecutePyCode class AskReview(Action): async def run(self, context: List[Message], plan: Plan = None): - prompt = "\n".join( - [f"{msg.cause_by() if msg.cause_by else 'Main Requirement'}: {msg.content}" for msg in context] - ) + logger.info("Current overall plan:") + logger.info("\n".join([f"{task.task_id}: {task.instruction}" for task in plan.tasks])) - latest_action = context[-1].cause_by() - - prompt += f"\nPlease review output from {latest_action}, " \ - "provide feedback or type YES to continue with the process:\n" + logger.info("most recent context:") + # prompt = "\n".join( + # [f"{msg.cause_by.__name__ if msg.cause_by else 'Main Requirement'}: {msg.content}" for msg in context] + # ) + prompt = "" + latest_action = context[-1].cause_by.__name__ + prompt += f"\nPlease review output from {latest_action}:\n" \ + "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" \ + "If you confirm the output and wish to continue with the current process, type CONFIRM:\n" rsp = input(prompt) - confirmed = "yes" in rsp.lower() + confirmed = "confirm" in rsp.lower() + return rsp, confirmed +class WriteTaskGuide(Action): + + async def run(self, task_instruction: str, data_desc: str = "") -> str: + return "" class MLEngineer(Role): - def __init__(self, name="ABC", profile="MLEngineer"): - super().__init__(name=name, profile=profile) + def __init__(self, name="ABC", profile="MLEngineer", goal=""): + super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") - self.plan = Plan() + self.plan = Plan(goal=goal) + self.use_tools = False + self.use_task_guide = False async def _plan_and_act(self): @@ -60,18 +71,28 @@ class MLEngineer(Role): await self._update_plan() async def _write_and_exec_code(self, max_retry: int = 3): + + task_guide = await WriteTaskGuide().run(self.plan.current_task.instruction) if self.use_task_guide else "" + counter = 0 success = False while not success and counter < max_retry: - context = self.get_memories() + context = self.get_useful_memories() - code = "print('abc')" - # code = await WriteCodeFunction().run(context=context) - # code = await WriteCodeWithOps.run(context, task, result) - self._rc.memory.add(Message(content=code, role="assistant", cause_by=WriteCodeFunction)) + if not self.use_tools: + # code = "print('abc')" + code = await WriteCodeByGenerate().run(context=context, plan=self.plan, task_guide=task_guide) + cause_by = WriteCodeByGenerate + + else: + code = await WriteCodeWithTools().run(context=context, plan=self.plan, task_guide=task_guide) + cause_by = WriteCodeWithTools + + self._rc.memory.add(Message(content=code, role="assistant", cause_by=cause_by)) result, success = await ExecutePyCode().run(code) - self._rc.memory.add(Message(content=result, role="assistant", cause_by=ExecutePyCode)) + print(result) + self._rc.memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) # if not success: # await self._ask_review() @@ -81,16 +102,16 @@ class MLEngineer(Role): return code, result, success async def _ask_review(self): - context = self.get_memories() + context = self.get_useful_memories() review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan) - self._rc.memory.add(Message(content=review, role="assistant", cause_by=AskReview)) + self._rc.memory.add(Message(content=review, role="user", cause_by=AskReview)) return confirmed async def _update_plan(self, max_tasks: int = 3): current_plan = str([task.json() for task in self.plan.tasks]) plan_confirmed = False while not plan_confirmed: - context = self.get_memories() + context = self.get_useful_memories() rsp = await WritePlan().run(context, current_plan=current_plan, max_tasks=max_tasks) self._rc.memory.add(Message(content=rsp, role="assistant", cause_by=WritePlan)) plan_confirmed = await self._ask_review() @@ -98,13 +119,18 @@ class MLEngineer(Role): tasks = WritePlan.rsp_to_tasks(rsp) self.plan.add_tasks(tasks) + def get_useful_memories(self, current_task_memories: List[str] = []) -> List[Message]: + """find useful memories only to reduce context length and improve performance""" + memories = super().get_memories() + return memories + if __name__ == "__main__": # requirement = "create a normal distribution and visualize it" requirement = "run some analysis on iris dataset" async def main(requirement: str = requirement): - role = MLEngineer() + role = MLEngineer(goal=requirement) await role.run(requirement) fire.Fire(main) diff --git a/metagpt/schema.py b/metagpt/schema.py index 3cd7d9730..e39f54a0c 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -84,6 +84,7 @@ class Task(BaseModel): class Plan(BaseModel): + goal: str tasks: list[Task] = [] task_map: dict[str, Task] = {} current_task_id = "" diff --git a/tests/metagpt/actions/test_write_code_function.py b/tests/metagpt/actions/test_write_analysis_code.py similarity index 86% rename from tests/metagpt/actions/test_write_code_function.py rename to tests/metagpt/actions/test_write_analysis_code.py index 4ff1a63c4..41c0479a9 100644 --- a/tests/metagpt/actions/test_write_code_function.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -1,12 +1,12 @@ import pytest -from metagpt.actions.write_code_function import WriteCodeFunction +from metagpt.actions.write_analysis_code import WriteCodeByGenerate from metagpt.actions.execute_code import ExecutePyCode @pytest.mark.asyncio async def test_write_code(): - write_code = WriteCodeFunction() + write_code = WriteCodeByGenerate() code = await write_code.run("Write a hello world code.") assert "language" in code.content assert "code" in code.content @@ -15,7 +15,7 @@ async def test_write_code(): @pytest.mark.asyncio async def test_write_code_by_list_prompt(): - write_code = WriteCodeFunction() + write_code = WriteCodeByGenerate() msg = ["a=[1,2,5,10,-10]", "写出求a中最大值的代码python"] code = await write_code.run(msg) assert "language" in code.content @@ -25,7 +25,7 @@ async def test_write_code_by_list_prompt(): @pytest.mark.asyncio async def test_write_code_by_list_plan(): - write_code = WriteCodeFunction() + write_code = WriteCodeByGenerate() execute_code = ExecutePyCode() messages = [] plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] diff --git a/tests/metagpt/test_schema.py b/tests/metagpt/test_schema.py index 6aae82006..8f65d3785 100644 --- a/tests/metagpt/test_schema.py +++ b/tests/metagpt/test_schema.py @@ -24,7 +24,7 @@ def test_messages(): class TestPlan: def test_add_tasks_ordering(self): - plan = Plan() + plan = Plan(goal="") tasks = [ Task(task_id="1", dependent_task_ids=["2", "3"], instruction="Third"), @@ -36,7 +36,7 @@ class TestPlan: assert [task.task_id for task in plan.tasks] == ["2", "3", "1"] def test_add_tasks_to_existing_no_common_prefix(self): - plan = Plan() + plan = Plan(goal="") tasks = [ Task(task_id="1", dependent_task_ids=["2", "3"], instruction="Third"), @@ -52,7 +52,7 @@ class TestPlan: assert not plan.tasks[0].is_finished # must be the new unfinished task def test_add_tasks_to_existing_with_common_prefix(self): - plan = Plan() + plan = Plan(goal="") tasks = [ Task(task_id="1", dependent_task_ids=["2", "3"], instruction="Third"), @@ -75,7 +75,7 @@ class TestPlan: assert plan.current_task_id == "4" def test_current_task(self): - plan = Plan() + plan = Plan(goal="") tasks = [ Task(task_id="1", dependent_task_ids=["2"], instruction="Second"), Task(task_id="2", instruction="First") @@ -84,7 +84,7 @@ class TestPlan: assert plan.current_task.task_id == "2" def test_finish_task(self): - plan = Plan() + plan = Plan(goal="") tasks = [ Task(task_id="1", instruction="First"), Task(task_id="2", dependent_task_ids=["1"], instruction="Second") @@ -94,7 +94,7 @@ class TestPlan: assert plan.current_task.task_id == "2" def test_finished_tasks(self): - plan = Plan() + plan = Plan(goal="") tasks = [ Task(task_id="1", instruction="First"), Task(task_id="2", dependent_task_ids=["1"], instruction="Second") From bb8c39a312c558d53d803832052a39854fe6aa60 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 24 Nov 2023 15:01:52 +0800 Subject: [PATCH 017/637] init function tools and define tool schema --- metagpt/tools/functions/__init__.py | 8 ++ metagpt/tools/functions/libs/__init__.py | 6 ++ metagpt/tools/functions/schemas/__init__.py | 6 ++ metagpt/tools/functions/schemas/base.py | 100 ++++++++++++++++++++ 4 files changed, 120 insertions(+) create mode 100644 metagpt/tools/functions/__init__.py create mode 100644 metagpt/tools/functions/libs/__init__.py create mode 100644 metagpt/tools/functions/schemas/__init__.py create mode 100644 metagpt/tools/functions/schemas/base.py diff --git a/metagpt/tools/functions/__init__.py b/metagpt/tools/functions/__init__.py new file mode 100644 index 000000000..069e4297b --- /dev/null +++ b/metagpt/tools/functions/__init__.py @@ -0,0 +1,8 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/16 16:32 +# @Author : lidanyang +# @File : __init__.py +# @Desc : +from metagpt.tools.functions.register.register import registry +import metagpt.tools.functions.libs.machine_learning diff --git a/metagpt/tools/functions/libs/__init__.py b/metagpt/tools/functions/libs/__init__.py new file mode 100644 index 000000000..a0a43f507 --- /dev/null +++ b/metagpt/tools/functions/libs/__init__.py @@ -0,0 +1,6 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/16 16:32 +# @Author : lidanyang +# @File : __init__.py +# @Desc : diff --git a/metagpt/tools/functions/schemas/__init__.py b/metagpt/tools/functions/schemas/__init__.py new file mode 100644 index 000000000..e50f67d6f --- /dev/null +++ b/metagpt/tools/functions/schemas/__init__.py @@ -0,0 +1,6 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/16 16:33 +# @Author : lidanyang +# @File : __init__.py +# @Desc : diff --git a/metagpt/tools/functions/schemas/base.py b/metagpt/tools/functions/schemas/base.py new file mode 100644 index 000000000..35b9f77b7 --- /dev/null +++ b/metagpt/tools/functions/schemas/base.py @@ -0,0 +1,100 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/16 16:34 +# @Author : lidanyang +# @File : base.py +# @Desc : Build base class to generate schema for tool +from typing import Any, List, Optional, get_type_hints + + +class NoDefault: + """ + A class to represent a missing default value. + + This is used to distinguish between a default value of None and a missing default value. + """ + pass + + +def field( + description: str, default: Any = NoDefault(), enum: Optional[List[Any]] = None, **kwargs +): + """ + Create a field for a tool parameter. + + Args: + description (str): A description of the field. + default (Any, optional): The default value for the field. Defaults to None. + enum (Optional[List[Any]], optional): A list of possible values for the field. Defaults to None. + **kwargs: Additional keyword arguments. + + Returns: + dict: A dictionary representing the field with provided attributes. + """ + field_info = { + "description": description, + "default": default, + "enum": enum, + } + field_info.update(kwargs) + return field_info + + +class ToolSchema: + @staticmethod + def format_type(type_hint): + """ + Format a type hint into a string representation. + + Args: + type_hint (type): The type hint to format. + + Returns: + str: A string representation of the type hint. + """ + if isinstance(type_hint, type): + # Handle built-in types separately + if type_hint.__module__ == "builtins": + return type_hint.__name__ + else: + return f"{type_hint.__module__}.{type_hint.__name__}" + elif hasattr(type_hint, "__origin__") and hasattr(type_hint, "__args__"): + # Handle generic types (like List[int]) + origin_type = ToolSchema.format_type(type_hint.__origin__) + args_type = ", ".join( + [ToolSchema.format_type(t) for t in type_hint.__args__] + ) + return f"{origin_type}[{args_type}]" + else: + return str(type_hint) + + @classmethod + def schema(cls): + """ + Generate a schema dictionary for the class. + + The schema includes the class name, description, and information about + each class parameter based on type hints and field definitions. + + Returns: + dict: A dictionary representing the schema of the class. + """ + schema = { + "name": cls.__name__, + "description": cls.__doc__, + "parameters": {"type": "object", "properties": {}, "required": []}, + } + type_hints = get_type_hints(cls) + for attr, type_hint in type_hints.items(): + value = getattr(cls, attr, None) + if isinstance(value, dict): + # Process each attribute that is defined using the field function + prop_info = {k: v for k, v in value.items() if v is not None or k == "default"} + if isinstance(prop_info["default"], NoDefault): + del prop_info["default"] + prop_info["type"] = ToolSchema.format_type(type_hint) + schema["parameters"]["properties"][attr] = prop_info + # Check for required fields + if "default" not in prop_info: + schema["parameters"]["required"].append(attr) + return schema From b0e28838e490db5577faa9092bc7055ff3d720ae Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 24 Nov 2023 15:02:40 +0800 Subject: [PATCH 018/637] add function register --- metagpt/tools/functions/register/__init__.py | 6 ++ metagpt/tools/functions/register/register.py | 65 ++++++++++++++++++++ 2 files changed, 71 insertions(+) create mode 100644 metagpt/tools/functions/register/__init__.py create mode 100644 metagpt/tools/functions/register/register.py diff --git a/metagpt/tools/functions/register/__init__.py b/metagpt/tools/functions/register/__init__.py new file mode 100644 index 000000000..c80872750 --- /dev/null +++ b/metagpt/tools/functions/register/__init__.py @@ -0,0 +1,6 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/16 16:37 +# @Author : lidanyang +# @File : __init__.py +# @Desc : diff --git a/metagpt/tools/functions/register/register.py b/metagpt/tools/functions/register/register.py new file mode 100644 index 000000000..120c7c4a2 --- /dev/null +++ b/metagpt/tools/functions/register/register.py @@ -0,0 +1,65 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/16 16:38 +# @Author : lidanyang +# @File : register.py +# @Desc : +from typing import Type, Optional, Callable, Dict, Union, List + +from metagpt.tools.functions.schemas.base import ToolSchema + + +class FunctionRegistry: + def __init__(self): + self.functions: Dict[str, Dict[str, Dict]] = {} + + def register(self, module: str, tool_schema: Type[ToolSchema]) -> Callable: + + def wrapper(func: Callable) -> Callable: + module_registry = self.functions.setdefault(module, {}) + + if func.__name__ in module_registry: + raise ValueError(f"Function {func.__name__} is already registered in {module}") + + schema = tool_schema.schema() + schema["name"] = func.__name__ + module_registry[func.__name__] = { + "func": func, + "schema": schema, + } + return func + + return wrapper + + def get(self, module: str, name: str) -> Optional[Union[Callable, Dict]]: + """Get function by module and name""" + module_registry = self.functions.get(module, {}) + return module_registry.get(name) + + def get_by_name(self, name: str) -> Optional[Dict]: + """Get function by name""" + for module_registry in self.functions.values(): + if name in module_registry: + return module_registry.get(name, {}) + + def get_all_by_module(self, module: str) -> Optional[Dict]: + """Get all functions by module""" + return self.functions.get(module, {}) + + def get_schema(self, module: str, name: str) -> Optional[Dict]: + """Get schema by module and name""" + module_registry = self.functions.get(module, {}) + return module_registry.get(name, {}).get("schema") + + def get_schemas(self, module: str, names: List[str]) -> List[Dict]: + """Get schemas by module and names""" + module_registry = self.functions.get(module, {}) + return [module_registry.get(name, {}).get("schema") for name in names] + + def get_all_schema_by_module(self, module: str) -> List[Dict]: + """Get all schemas by module""" + module_registry = self.functions.get(module, {}) + return [v.get("schema") for v in module_registry.values()] + + +registry = FunctionRegistry() From a911f5649df85df5f1e41827a5ffebf120edba94 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 24 Nov 2023 15:03:03 +0800 Subject: [PATCH 019/637] add feature engineering tools --- .../libs/machine_learning/__init__.py | 7 + .../machine_learning/feature_engineering.py | 174 ++++++++++++++++++ .../schemas/machine_learning/__init__.py | 6 + .../machine_learning/feature_engineering.py | 98 ++++++++++ 4 files changed, 285 insertions(+) create mode 100644 metagpt/tools/functions/libs/machine_learning/__init__.py create mode 100644 metagpt/tools/functions/libs/machine_learning/feature_engineering.py create mode 100644 metagpt/tools/functions/schemas/machine_learning/__init__.py create mode 100644 metagpt/tools/functions/schemas/machine_learning/feature_engineering.py diff --git a/metagpt/tools/functions/libs/machine_learning/__init__.py b/metagpt/tools/functions/libs/machine_learning/__init__.py new file mode 100644 index 000000000..5e9760c64 --- /dev/null +++ b/metagpt/tools/functions/libs/machine_learning/__init__.py @@ -0,0 +1,7 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/16 16:36 +# @Author : lidanyang +# @File : __init__.py +# @Desc : +from metagpt.tools.functions.libs.machine_learning.feature_engineering import * diff --git a/metagpt/tools/functions/libs/machine_learning/feature_engineering.py b/metagpt/tools/functions/libs/machine_learning/feature_engineering.py new file mode 100644 index 000000000..584bd125d --- /dev/null +++ b/metagpt/tools/functions/libs/machine_learning/feature_engineering.py @@ -0,0 +1,174 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/17 10:33 +# @Author : lidanyang +# @File : feature_engineering.py +# @Desc : Feature Engineering Functions +import itertools + +from dateutil.relativedelta import relativedelta +from pandas.api.types import is_numeric_dtype +from sklearn.preprocessing import PolynomialFeatures, OneHotEncoder + +from metagpt.tools.functions import registry +from metagpt.tools.functions.schemas.machine_learning.feature_engineering import * + + +@registry.register("feature_engineering", PolynomialExpansion) +def polynomial_expansion(df, cols, degree=2): + for col in cols: + if not is_numeric_dtype(df[col]): + raise ValueError(f"Column '{col}' must be numeric.") + + poly = PolynomialFeatures(degree=degree, include_bias=False) + ts_data = poly.fit_transform(df[cols].fillna(0)) + new_columns = poly.get_feature_names_out(cols) + ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) + ts_data = ts_data.drop(cols, axis=1) + df = pd.concat([df, ts_data], axis=1) + return df + + +@registry.register("feature_engineering", OneHotEncoding) +def one_hot_encoding(df, cols): + enc = OneHotEncoder(handle_unknown="ignore", sparse=False) + ts_data = enc.fit_transform(df[cols]) + new_columns = enc.get_feature_names_out(cols) + ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) + df.drop(cols, axis=1, inplace=True) + df = pd.concat([df, ts_data], axis=1) + return df + + +@registry.register("feature_engineering", FrequencyEncoding) +def frequency_encoding(df, cols): + for col in cols: + encoder_dict = df[col].value_counts().to_dict() + df[f"{col}_cnt"] = df[col].map(encoder_dict) + return df + + +@registry.register("feature_engineering", CatCross) +def cat_cross(df, cols, max_cat_num=100): + for col in cols: + if df[col].nunique() > max_cat_num: + cols.remove(col) + + for col1, col2 in itertools.combinations(cols, 2): + cross_col = f"{col1}_cross_{col2}" + df[cross_col] = df[col1].astype(str) + "_" + df[col2].astype(str) + return df + + +@registry.register("feature_engineering", GroupStat) +def group_stat(df, group_col, agg_col, agg_funcs): + group_df = df.groupby(group_col)[agg_col].agg(agg_funcs).reset_index() + group_df.columns = group_col + [ + f"{agg_col}_{agg_func}_by_{group_col}" for agg_func in agg_funcs + ] + df = df.merge(group_df, on=group_col, how="left") + return df + + +@registry.register("feature_engineering", ExtractTimeComps) +def extract_time_comps(df, time_col, time_comps): + time_s = pd.to_datetime(df[time_col], errors="coerce") + time_comps_df = pd.DataFrame() + + if "year" in time_comps: + time_comps_df["year"] = time_s.dt.year + if "month" in time_comps: + time_comps_df["month"] = time_s.dt.month + if "day" in time_comps: + time_comps_df["day"] = time_s.dt.day + if "hour" in time_comps: + time_comps_df["hour"] = time_s.dt.hour + if "dayofweek" in time_comps: + time_comps_df["dayofweek"] = time_s.dt.dayofweek + 1 + if "is_weekend" in time_comps: + time_comps_df["is_weekend"] = time_s.dt.dayofweek.isin([5, 6]).astype(int) + df = pd.concat([df, time_comps_df], axis=1) + return df + + +@registry.register("feature_engineering", FeShiftByTime) +def fe_shift_by_time(df, time_col, group_col, shift_col, periods, freq): + df[time_col] = pd.to_datetime(df[time_col]) + + def shift_datetime(date, offset, unit): + if unit in ["year", "y", "Y"]: + return date + relativedelta(years=offset) + elif unit in ["month", "m", "M"]: + return date + relativedelta(months=offset) + elif unit in ["day", "d", "D"]: + return date + relativedelta(days=offset) + elif unit in ["week", "w", "W"]: + return date + relativedelta(weeks=offset) + elif unit in ["hour", "h", "H"]: + return date + relativedelta(hours=offset) + else: + return date + + def shift_by_time_on_key( + inner_df, time_col, group_col, shift_col, offset, unit, col_name + ): + inner_df = inner_df.drop_duplicates() + inner_df[time_col] = inner_df[time_col].map( + lambda x: shift_datetime(x, offset, unit) + ) + inner_df = inner_df.groupby([time_col, group_col], as_index=False)[ + shift_col + ].mean() + inner_df.rename(columns={shift_col: col_name}, inplace=True) + return inner_df + + shift_df = df[[time_col, group_col, shift_col]].copy() + for period in periods: + new_col_name = f"{group_col}_{shift_col}_lag_{period}_{freq}" + tmp = shift_by_time_on_key( + shift_df, time_col, group_col, shift_col, period, freq, new_col_name + ) + df = df.merge(tmp, on=[time_col, group_col], how="left") + + return df + + +@registry.register("feature_engineering", FeRollingByTime) +def fe_rolling_by_time(df, time_col, group_col, rolling_col, periods, freq, agg_funcs): + df[time_col] = pd.to_datetime(df[time_col]) + + def rolling_by_time_on_key(inner_df, offset, unit, agg_func, col_name): + time_freq = { + "Y": [365 * offset, "D"], + "M": [30 * offset, "D"], + "D": [offset, "D"], + "W": [7 * offset, "D"], + "H": [offset, "h"], + } + + if agg_func not in ["mean", "std", "max", "min", "median", "sum", "count"]: + raise ValueError(f"Invalid agg function: {agg_func}") + + rolling_feat = inner_df.rolling( + f"{time_freq[unit][0]}{time_freq[unit][1]}", closed="left" + ) + rolling_feat = getattr(rolling_feat, agg_func)() + depth = df.columns.nlevels + rolling_feat = rolling_feat.stack(list(range(depth))) + rolling_feat.name = col_name + return rolling_feat + + rolling_df = df[[time_col, group_col, rolling_col]].copy() + for period in periods: + for func in agg_funcs: + new_col_name = f"{group_col}_{rolling_col}_rolling_{period}_{freq}_{func}" + tmp = pd.pivot_table( + rolling_df, + index=time_col, + values=rolling_col, + columns=group_col, + ) + tmp = rolling_by_time_on_key(tmp, period, freq, func, new_col_name) + df = df.merge(tmp, on=[time_col, group_col], how="left") + + return df diff --git a/metagpt/tools/functions/schemas/machine_learning/__init__.py b/metagpt/tools/functions/schemas/machine_learning/__init__.py new file mode 100644 index 000000000..c80872750 --- /dev/null +++ b/metagpt/tools/functions/schemas/machine_learning/__init__.py @@ -0,0 +1,6 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/16 16:37 +# @Author : lidanyang +# @File : __init__.py +# @Desc : diff --git a/metagpt/tools/functions/schemas/machine_learning/feature_engineering.py b/metagpt/tools/functions/schemas/machine_learning/feature_engineering.py new file mode 100644 index 000000000..8237c83f4 --- /dev/null +++ b/metagpt/tools/functions/schemas/machine_learning/feature_engineering.py @@ -0,0 +1,98 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/17 10:34 +# @Author : lidanyang +# @File : feature_engineering.py +# @Desc : Schema for feature engineering functions +from typing import List + +import pandas as pd + +from metagpt.tools.functions.schemas.base import field, ToolSchema + + +class PolynomialExpansion(ToolSchema): + """Generate polynomial and interaction features from selected columns, excluding the bias column.""" + + df: pd.DataFrame = field(description="DataFrame to process.") + cols: list = field(description="Columns for polynomial expansion.") + degree: int = field(description="Degree of polynomial features.", default=2) + + +class OneHotEncoding(ToolSchema): + """Apply one-hot encoding to specified categorical columns in a DataFrame.""" + + df: pd.DataFrame = field(description="DataFrame to process.") + cols: list = field(description="Categorical columns to be one-hot encoded.") + + +class FrequencyEncoding(ToolSchema): + """Convert categorical columns to frequency encoding.""" + + df: pd.DataFrame = field(description="DataFrame to process.") + cols: list = field(description="Categorical columns to be frequency encoded.") + + +class CatCross(ToolSchema): + """Create pairwise crossed features from categorical columns, joining values with '_'.""" + + df: pd.DataFrame = field(description="DataFrame to process.") + cols: list = field(description="Columns to be pairwise crossed.") + max_cat_num: int = field( + description="Maximum unique categories per crossed feature.", default=100 + ) + + +class GroupStat(ToolSchema): + """Perform aggregation operations on a specified column grouped by certain categories.""" + + df: pd.DataFrame = field(description="DataFrame to process.") + group_col: str = field(description="Column used for grouping.") + agg_col: str = field(description="Column on which aggregation is performed.") + agg_funcs: list = field( + description="""List of aggregation functions to apply, such as ['mean', 'std']. + Each function must be supported by pandas.""" + ) + + +class ExtractTimeComps(ToolSchema): + """Extract specific time components from a designated time column in a DataFrame.""" + + df: pd.DataFrame = field(description="DataFrame to process.") + time_col: str = field(description="The name of the column containing time data.") + time_comps: List[str] = field( + description="""List of time components to extract. + Each component must be in ['year', 'month', 'day', 'hour', 'dayofweek', 'is_weekend'].""" + ) + + +class FeShiftByTime(ToolSchema): + """Shift column values in a DataFrame based on specified time intervals.""" + + df: pd.DataFrame = field(description="DataFrame to process.") + time_col: str = field(description="Column for time-based shifting.") + group_col: str = field(description="Column for grouping before shifting.") + shift_col: str = field(description="Column to shift.") + periods: list = field(description="Time intervals for shifting.") + freq: str = field( + description="Frequency unit for time intervals (e.g., 'D', 'M').", + enum=["D", "M", "Y", "W", "H"], + ) + + +class FeRollingByTime(ToolSchema): + """Calculate rolling statistics for a DataFrame column over time intervals.""" + + df: pd.DataFrame = field(description="DataFrame to process.") + time_col: str = field(description="Column for time-based rolling.") + group_col: str = field(description="Column for grouping before rolling.") + rolling_col: str = field(description="Column for rolling calculations.") + periods: list = field(description="Window sizes for rolling.") + freq: str = field( + description="Frequency unit for time windows (e.g., 'D', 'M').", + enum=["D", "M", "Y", "W", "H"], + ) + agg_funcs: list = field( + description="""List of aggregation functions for rolling, like ['mean', 'std']. + Each function must be in ['mean', 'std', 'min', 'max', 'median', 'sum', 'count'].""" + ) From 142b04fa760490062f8366b836784ee02206e491 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 24 Nov 2023 15:04:14 +0800 Subject: [PATCH 020/637] test tool register --- tests/metagpt/tools/functions/__init__.py | 6 ++ .../tools/functions/register/__init__.py | 6 ++ .../tools/functions/register/test_register.py | 55 +++++++++++++++++++ 3 files changed, 67 insertions(+) create mode 100644 tests/metagpt/tools/functions/__init__.py create mode 100644 tests/metagpt/tools/functions/register/__init__.py create mode 100644 tests/metagpt/tools/functions/register/test_register.py diff --git a/tests/metagpt/tools/functions/__init__.py b/tests/metagpt/tools/functions/__init__.py new file mode 100644 index 000000000..7d36f3404 --- /dev/null +++ b/tests/metagpt/tools/functions/__init__.py @@ -0,0 +1,6 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/17 10:24 +# @Author : lidanyang +# @File : __init__.py +# @Desc : diff --git a/tests/metagpt/tools/functions/register/__init__.py b/tests/metagpt/tools/functions/register/__init__.py new file mode 100644 index 000000000..7d36f3404 --- /dev/null +++ b/tests/metagpt/tools/functions/register/__init__.py @@ -0,0 +1,6 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/17 10:24 +# @Author : lidanyang +# @File : __init__.py +# @Desc : diff --git a/tests/metagpt/tools/functions/register/test_register.py b/tests/metagpt/tools/functions/register/test_register.py new file mode 100644 index 000000000..a71f7d01c --- /dev/null +++ b/tests/metagpt/tools/functions/register/test_register.py @@ -0,0 +1,55 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/17 10:24 +# @Author : lidanyang +# @File : test_register.py +# @Desc : +import pytest + +from metagpt.tools.functions.register.register import FunctionRegistry +from metagpt.tools.functions.schemas.base import ToolSchema, field + + +@pytest.fixture +def registry(): + return FunctionRegistry() + + +class AddNumbers(ToolSchema): + """Add two numbers""" + + num1: int = field(description="First number") + num2: int = field(description="Second number") + + +def test_register(registry): + @registry.register("module1", AddNumbers) + def add_numbers(num1, num2): + return num1 + num2 + + assert len(registry.functions["module1"]) == 1 + assert "add_numbers" in registry.functions["module1"] + + with pytest.raises(ValueError): + + @registry.register("module1", AddNumbers) + def add_numbers(num1, num2): + return num1 + num2 + + func = registry.get("module1", "add_numbers") + assert func["func"](1, 2) == 3 + assert func["schema"] == { + "name": "add_numbers", + "description": "Add two numbers", + "parameters": { + "type": "object", + "properties": { + "num1": {"description": "First number", "type": "int"}, + "num2": {"description": "Second number", "type": "int"}, + }, + "required": ["num1", "num2"], + }, + } + + module1_funcs = registry.get_all_by_module("module1") + assert len(module1_funcs) == 1 From fdc49775e613036f6da3169a1298a28792aae018 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 24 Nov 2023 17:23:39 +0800 Subject: [PATCH 021/637] reduce hierarchy of machine learning --- metagpt/tools/functions/__init__.py | 2 +- .../libs/{machine_learning => }/feature_engineering.py | 2 +- metagpt/tools/functions/libs/machine_learning/__init__.py | 7 ------- .../schemas/{machine_learning => }/feature_engineering.py | 0 .../tools/functions/schemas/machine_learning/__init__.py | 6 ------ 5 files changed, 2 insertions(+), 15 deletions(-) rename metagpt/tools/functions/libs/{machine_learning => }/feature_engineering.py (98%) delete mode 100644 metagpt/tools/functions/libs/machine_learning/__init__.py rename metagpt/tools/functions/schemas/{machine_learning => }/feature_engineering.py (100%) delete mode 100644 metagpt/tools/functions/schemas/machine_learning/__init__.py diff --git a/metagpt/tools/functions/__init__.py b/metagpt/tools/functions/__init__.py index 069e4297b..b81e85833 100644 --- a/metagpt/tools/functions/__init__.py +++ b/metagpt/tools/functions/__init__.py @@ -5,4 +5,4 @@ # @File : __init__.py # @Desc : from metagpt.tools.functions.register.register import registry -import metagpt.tools.functions.libs.machine_learning +import metagpt.tools.functions.libs.feature_engineering diff --git a/metagpt/tools/functions/libs/machine_learning/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py similarity index 98% rename from metagpt/tools/functions/libs/machine_learning/feature_engineering.py rename to metagpt/tools/functions/libs/feature_engineering.py index 584bd125d..0573f362d 100644 --- a/metagpt/tools/functions/libs/machine_learning/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -11,7 +11,7 @@ from pandas.api.types import is_numeric_dtype from sklearn.preprocessing import PolynomialFeatures, OneHotEncoder from metagpt.tools.functions import registry -from metagpt.tools.functions.schemas.machine_learning.feature_engineering import * +from metagpt.tools.functions.schemas.feature_engineering import * @registry.register("feature_engineering", PolynomialExpansion) diff --git a/metagpt/tools/functions/libs/machine_learning/__init__.py b/metagpt/tools/functions/libs/machine_learning/__init__.py deleted file mode 100644 index 5e9760c64..000000000 --- a/metagpt/tools/functions/libs/machine_learning/__init__.py +++ /dev/null @@ -1,7 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/16 16:36 -# @Author : lidanyang -# @File : __init__.py -# @Desc : -from metagpt.tools.functions.libs.machine_learning.feature_engineering import * diff --git a/metagpt/tools/functions/schemas/machine_learning/feature_engineering.py b/metagpt/tools/functions/schemas/feature_engineering.py similarity index 100% rename from metagpt/tools/functions/schemas/machine_learning/feature_engineering.py rename to metagpt/tools/functions/schemas/feature_engineering.py diff --git a/metagpt/tools/functions/schemas/machine_learning/__init__.py b/metagpt/tools/functions/schemas/machine_learning/__init__.py deleted file mode 100644 index c80872750..000000000 --- a/metagpt/tools/functions/schemas/machine_learning/__init__.py +++ /dev/null @@ -1,6 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/16 16:37 -# @Author : lidanyang -# @File : __init__.py -# @Desc : From f19003b413ab216128f55f18d1679802308049cb Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 24 Nov 2023 17:46:43 +0800 Subject: [PATCH 022/637] rename field to tool_field --- metagpt/tools/functions/__init__.py | 1 + metagpt/tools/functions/schemas/base.py | 2 +- .../functions/schemas/feature_engineering.py | 64 ++++++++++--------- .../tools/functions/register/test_register.py | 6 +- 4 files changed, 38 insertions(+), 35 deletions(-) diff --git a/metagpt/tools/functions/__init__.py b/metagpt/tools/functions/__init__.py index b81e85833..7ab850667 100644 --- a/metagpt/tools/functions/__init__.py +++ b/metagpt/tools/functions/__init__.py @@ -6,3 +6,4 @@ # @Desc : from metagpt.tools.functions.register.register import registry import metagpt.tools.functions.libs.feature_engineering +print(registry.functions) \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/base.py b/metagpt/tools/functions/schemas/base.py index 35b9f77b7..aef604c8d 100644 --- a/metagpt/tools/functions/schemas/base.py +++ b/metagpt/tools/functions/schemas/base.py @@ -16,7 +16,7 @@ class NoDefault: pass -def field( +def tool_field( description: str, default: Any = NoDefault(), enum: Optional[List[Any]] = None, **kwargs ): """ diff --git a/metagpt/tools/functions/schemas/feature_engineering.py b/metagpt/tools/functions/schemas/feature_engineering.py index 8237c83f4..c14bb933e 100644 --- a/metagpt/tools/functions/schemas/feature_engineering.py +++ b/metagpt/tools/functions/schemas/feature_engineering.py @@ -8,37 +8,37 @@ from typing import List import pandas as pd -from metagpt.tools.functions.schemas.base import field, ToolSchema +from metagpt.tools.functions.schemas.base import ToolSchema, tool_field class PolynomialExpansion(ToolSchema): """Generate polynomial and interaction features from selected columns, excluding the bias column.""" - df: pd.DataFrame = field(description="DataFrame to process.") - cols: list = field(description="Columns for polynomial expansion.") - degree: int = field(description="Degree of polynomial features.", default=2) + df: pd.DataFrame = tool_field(description="DataFrame to process.") + cols: list = tool_field(description="Columns for polynomial expansion.") + degree: int = tool_field(description="Degree of polynomial features.", default=2) class OneHotEncoding(ToolSchema): """Apply one-hot encoding to specified categorical columns in a DataFrame.""" - df: pd.DataFrame = field(description="DataFrame to process.") - cols: list = field(description="Categorical columns to be one-hot encoded.") + df: pd.DataFrame = tool_field(description="DataFrame to process.") + cols: list = tool_field(description="Categorical columns to be one-hot encoded.") class FrequencyEncoding(ToolSchema): """Convert categorical columns to frequency encoding.""" - df: pd.DataFrame = field(description="DataFrame to process.") - cols: list = field(description="Categorical columns to be frequency encoded.") + df: pd.DataFrame = tool_field(description="DataFrame to process.") + cols: list = tool_field(description="Categorical columns to be frequency encoded.") class CatCross(ToolSchema): """Create pairwise crossed features from categorical columns, joining values with '_'.""" - df: pd.DataFrame = field(description="DataFrame to process.") - cols: list = field(description="Columns to be pairwise crossed.") - max_cat_num: int = field( + df: pd.DataFrame = tool_field(description="DataFrame to process.") + cols: list = tool_field(description="Columns to be pairwise crossed.") + max_cat_num: int = tool_field( description="Maximum unique categories per crossed feature.", default=100 ) @@ -46,10 +46,10 @@ class CatCross(ToolSchema): class GroupStat(ToolSchema): """Perform aggregation operations on a specified column grouped by certain categories.""" - df: pd.DataFrame = field(description="DataFrame to process.") - group_col: str = field(description="Column used for grouping.") - agg_col: str = field(description="Column on which aggregation is performed.") - agg_funcs: list = field( + df: pd.DataFrame = tool_field(description="DataFrame to process.") + group_col: str = tool_field(description="Column used for grouping.") + agg_col: str = tool_field(description="Column on which aggregation is performed.") + agg_funcs: list = tool_field( description="""List of aggregation functions to apply, such as ['mean', 'std']. Each function must be supported by pandas.""" ) @@ -58,9 +58,11 @@ class GroupStat(ToolSchema): class ExtractTimeComps(ToolSchema): """Extract specific time components from a designated time column in a DataFrame.""" - df: pd.DataFrame = field(description="DataFrame to process.") - time_col: str = field(description="The name of the column containing time data.") - time_comps: List[str] = field( + df: pd.DataFrame = tool_field(description="DataFrame to process.") + time_col: str = tool_field( + description="The name of the column containing time data." + ) + time_comps: List[str] = tool_field( description="""List of time components to extract. Each component must be in ['year', 'month', 'day', 'hour', 'dayofweek', 'is_weekend'].""" ) @@ -69,12 +71,12 @@ class ExtractTimeComps(ToolSchema): class FeShiftByTime(ToolSchema): """Shift column values in a DataFrame based on specified time intervals.""" - df: pd.DataFrame = field(description="DataFrame to process.") - time_col: str = field(description="Column for time-based shifting.") - group_col: str = field(description="Column for grouping before shifting.") - shift_col: str = field(description="Column to shift.") - periods: list = field(description="Time intervals for shifting.") - freq: str = field( + df: pd.DataFrame = tool_field(description="DataFrame to process.") + time_col: str = tool_field(description="Column for time-based shifting.") + group_col: str = tool_field(description="Column for grouping before shifting.") + shift_col: str = tool_field(description="Column to shift.") + periods: list = tool_field(description="Time intervals for shifting.") + freq: str = tool_field( description="Frequency unit for time intervals (e.g., 'D', 'M').", enum=["D", "M", "Y", "W", "H"], ) @@ -83,16 +85,16 @@ class FeShiftByTime(ToolSchema): class FeRollingByTime(ToolSchema): """Calculate rolling statistics for a DataFrame column over time intervals.""" - df: pd.DataFrame = field(description="DataFrame to process.") - time_col: str = field(description="Column for time-based rolling.") - group_col: str = field(description="Column for grouping before rolling.") - rolling_col: str = field(description="Column for rolling calculations.") - periods: list = field(description="Window sizes for rolling.") - freq: str = field( + df: pd.DataFrame = tool_field(description="DataFrame to process.") + time_col: str = tool_field(description="Column for time-based rolling.") + group_col: str = tool_field(description="Column for grouping before rolling.") + rolling_col: str = tool_field(description="Column for rolling calculations.") + periods: list = tool_field(description="Window sizes for rolling.") + freq: str = tool_field( description="Frequency unit for time windows (e.g., 'D', 'M').", enum=["D", "M", "Y", "W", "H"], ) - agg_funcs: list = field( + agg_funcs: list = tool_field( description="""List of aggregation functions for rolling, like ['mean', 'std']. Each function must be in ['mean', 'std', 'min', 'max', 'median', 'sum', 'count'].""" ) diff --git a/tests/metagpt/tools/functions/register/test_register.py b/tests/metagpt/tools/functions/register/test_register.py index a71f7d01c..8c9821268 100644 --- a/tests/metagpt/tools/functions/register/test_register.py +++ b/tests/metagpt/tools/functions/register/test_register.py @@ -7,7 +7,7 @@ import pytest from metagpt.tools.functions.register.register import FunctionRegistry -from metagpt.tools.functions.schemas.base import ToolSchema, field +from metagpt.tools.functions.schemas.base import ToolSchema, tool_field @pytest.fixture @@ -18,8 +18,8 @@ def registry(): class AddNumbers(ToolSchema): """Add two numbers""" - num1: int = field(description="First number") - num2: int = field(description="Second number") + num1: int = tool_field(description="First number") + num2: int = tool_field(description="Second number") def test_register(registry): From 8ff833f1e355d5adb5341dbbd8e3fd4265c110f3 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 24 Nov 2023 18:22:06 +0800 Subject: [PATCH 023/637] update increment development, add bug fix function --- metagpt/actions/refine_design_api.py | 25 ++- metagpt/actions/refine_prd.py | 17 +- metagpt/actions/refine_project_management.py | 34 ++-- metagpt/actions/write_code_refine.py | 80 ++++++++++ metagpt/roles/engineer.py | 154 +++++++++++++++---- metagpt/roles/qa_engineer.py | 36 ++++- startup.py | 68 +++++--- 7 files changed, 334 insertions(+), 80 deletions(-) create mode 100644 metagpt/actions/write_code_refine.py diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py index 591db175a..f1c231525 100644 --- a/metagpt/actions/refine_design_api.py +++ b/metagpt/actions/refine_design_api.py @@ -1,5 +1,6 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- +import re import shutil from pathlib import Path from typing import List @@ -25,7 +26,7 @@ templates = { ## Format example {format_example} ----- -Role: You are an architect; the goal is to design a SOTA PEP8-compliant python system; make the best use of good open source tools +Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and the provided difference descriptions. Make the best use of good open source tools. Requirement: Fill in the following missing information based on the context, each section name is a key in json Max Output: 8192 chars or 2048 tokens. Try to use them up. @@ -83,7 +84,7 @@ and only output the json inside this tag, nothing else ## Format example {format_example} ----- -Role: You are an architect; the goal is to design a SOTA PEP8-compliant python system; make the best use of good open source tools +Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and the provided difference descriptions. Make the best use of good open source tools. Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately Max Output: 8192 chars or 2048 tokens. Try to use them up. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. @@ -179,6 +180,25 @@ class RefineDesign(Action): pass # Folder does not exist, but we don't care workspace.mkdir(parents=True, exist_ok=True) + def create_or_increment_workspace(self, workspace: Path): + # 如果工作空间已存在,添加数字以区分 + original_workspace = workspace + index = 1 + while workspace.exists(): + ws_name_match = re.match(r'^(.*)_([\d]+)$', original_workspace.name) + if ws_name_match: + base_name, existing_index = ws_name_match.groups() + index = int(existing_index) + index += 1 + workspace = original_workspace.parent / f"{base_name}_{index}" + else: + workspace = original_workspace.parent / f"{original_workspace.name}_{index}" + index += 1 + + # 创建工作空间,包括所有必要的父文件夹 + workspace.mkdir(parents=True, exist_ok=True) + return workspace + async def _save_prd(self, docs_path, resources_path, context): prd_file = docs_path / "prd.md" if context[-1].instruct_content and context[-1].instruct_content.dict()["Competitive Quadrant Chart"]: @@ -208,6 +228,7 @@ class RefineDesign(Action): else: ws_name = CodeParser.parse_str(block="Python package name", text=system_design) workspace = WORKSPACE_ROOT / ws_name + # workspace = self.create_or_increment_workspace(workspace) self.recreate_workspace(workspace) docs_path = workspace / "docs" resources_path = workspace / "resources" diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py index 1e2709c6b..81cb02af8 100644 --- a/metagpt/actions/refine_prd.py +++ b/metagpt/actions/refine_prd.py @@ -43,13 +43,15 @@ quadrantChart ## Format example {format_example} ----- -Role: You are a professional product manager; the goal is to design a concise, usable, efficient product based on the new requirements, the difference description and Legacy PRD. +Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design ## New Requirements: Provide as Plain text and place the new requirements here ## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple +## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple + ## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple ## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less @@ -72,9 +74,10 @@ and only output the json inside this tag, nothing else [CONTENT] { "New Requirements": "", - "Difference Description": "", + "Difference Description": [], "Search Information": "", "Requirements": "", + "Incremental Development Plan": [], "Product Goals": [], "User Stories": [], "Competitive Analysis": [], @@ -138,7 +141,7 @@ quadrantChart ## Format example {format_example} ----- -Role: You are a professional product manager; the goal is to design a concise, usable, efficient product +Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format. @@ -146,6 +149,8 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD W ## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple +## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple + ## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple ## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less @@ -172,6 +177,11 @@ The boss ... "...", ] +## Incremental Development Plan +[ + "It ...", +] + ## Product Goals ```python [ @@ -225,6 +235,7 @@ INCREMENT_OUTPUT_MAPPING = { "New Requirements": (str, ...), # "Major Enhancements": (List[str], ...), "Difference Description": (List[str], ...), + "Incremental Development Plan": (List[str], ...), "Product Goals": (List[str], ...), "User Stories": (List[str], ...), "Competitive Analysis": (List[str], ...), diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py index 354996f3e..7d9e118d8 100644 --- a/metagpt/actions/refine_project_management.py +++ b/metagpt/actions/refine_project_management.py @@ -21,7 +21,7 @@ templates = { ## Format example {format_example} ----- -Role: You are a project manager; the goal is to break down tasks according to PRD/technical design, give a task list, and analyze task dependencies to start with the prerequisite modules +Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a task list, and analyze task dependencies to start with the prerequisite modules. Requirements: Based on the context, fill in the following missing information, each section name is a key in json. Here the granularity of the task is a file, if there are any missing files, you can supplement them Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. @@ -31,14 +31,14 @@ Attention: Use '##' to split sections, not '#', and '## ' SHOULD W ## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. +## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. + ## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first ## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first ## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. -## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. - ## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, @@ -58,6 +58,9 @@ and only output the json inside this tag, nothing else ... description: A JSON object ... """, + "Difference Description": """ + The ... + """, "Logic Analysis": [ ["game.py","Contains..."] ], @@ -67,9 +70,6 @@ and only output the json inside this tag, nothing else "Shared Knowledge": """ 'game.py' contains ... """, - "Difference Description": """ - The ... - """, "Anything UNCLEAR": "We need ... how to start." } ''', @@ -85,7 +85,7 @@ and only output the json inside this tag, nothing else ## Format example {format_example} ----- -Role: You are a project manager; the goal is to break down tasks according to PRD/technical design, give a task list, and analyze task dependencies to start with the prerequisite modules +Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a task list, and analyze task dependencies to start with the prerequisite modules. Requirements: Based on the context, fill in the following missing information, note that all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file, if there are any missing files, you can supplement them Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. @@ -95,14 +95,14 @@ Attention: Use '##' to split sections, not '#', and '## ' SHOULD W ## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. +## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. + ## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first ## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first ## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. -## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. - ## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. """, @@ -132,6 +132,13 @@ description: A JSON object ... """ ``` +## Difference Description +```python +""" +The ... +""" +``` + ## Logic Analysis ```python [ @@ -153,13 +160,6 @@ description: A JSON object ... """ ``` -## Difference Description -```python -""" -The ... -""" -``` - ## Anything UNCLEAR We need ... how to start. --- @@ -170,10 +170,10 @@ OUTPUT_MAPPING = { "Required Python third-party packages": (List[str], ...), "Required Other language third-party packages": (List[str], ...), "Full API spec": (str, ...), + "Difference Description": (str, ...), "Logic Analysis": (List[List[str]], ...), "Task list": (List[str], ...), "Shared Knowledge": (str, ...), - "Difference Description": (str, ...), "Anything UNCLEAR": (str, ...), } diff --git a/metagpt/actions/write_code_refine.py b/metagpt/actions/write_code_refine.py new file mode 100644 index 000000000..527952aed --- /dev/null +++ b/metagpt/actions/write_code_refine.py @@ -0,0 +1,80 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +from metagpt.actions.action import Action +from metagpt.logs import logger +from metagpt.schema import Message +from metagpt.utils.common import CodeParser +from tenacity import retry, stop_after_attempt, wait_fixed + + +PROMPT_TEMPLATE = """ +NOTICE +Role: You are a professional software engineer, and your main task is to conduct incremental development, which includes reviewing existing code, providing modification suggestions, rewriting code, and optimizing the codebase. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". + +## Code Review: Based on the following context and legacy code, and following the checklist, provide key, clear, concise, and specific code modification suggestions, up to 5. +``` +1. Check 0: Is the code implemented as per the requirements? +2. Check 1: Are there any issues with the code logic? +3. Check 2: Does the existing code follow the "Data structures and interface definitions"? +4. Check 3: Is there a function in the code that is omitted or not fully implemented that needs to be implemented? +5. Check 4: Does the code have unnecessary or lack dependencies? +``` + +## Incremental Development: {filename} Based on the findings from the "Code Review," context, and the legacy code, conduct incremental development by rewriting, optimizing, and adding new code using triple quotes. +----- +# Context +{context} + +## Legacy Code +You are tasked with conducting incremental development in the existing code and creating a new code file, {filename}, based on the provided legacy code and above information. +``` +{code} +``` +----- + +## Format example +----- +{format_example} +----- + +""" + +FORMAT_EXAMPLE = """ + +## Code Review +1. The code ... +2. ... +3. ... +4. ... +5. ... + +## Incremental Development: {filename} +```python +## {filename} - Incremental Development +... +``` +""" + + + +class WriteCodeRefine(Action): + def __init__(self, name="WriteCodeRefine", context: list[Message] = None, llm=None): + super().__init__(name, context, llm) + + @retry(stop=stop_after_attempt(2), wait=wait_fixed(1)) + async def write_code(self, prompt): + code_rsp = await self._aask(prompt) + code = CodeParser.parse_code(block="", text=code_rsp) + return code + + async def run(self, context, code, filename): + format_example = FORMAT_EXAMPLE.format(filename=filename) + prompt = PROMPT_TEMPLATE.format(context=context, code=code, filename=filename, format_example=format_example) + logger.info(f'Code refine {filename}..') + code = await self.write_code(prompt) + # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) + # self._save(context, filename, code) + return code + \ No newline at end of file diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 0a0107636..13034acaa 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -6,13 +6,16 @@ @File : engineer.py """ import asyncio +import re import shutil from collections import OrderedDict from pathlib import Path +from typing import List -from metagpt.actions import WriteCode, WriteCodeReview, WriteDesign, WriteTasks +from metagpt.actions import WriteCode, WriteCodeReview, WriteDesign, WriteTasks, BossRequirement from metagpt.actions.refine_design_api import RefineDesign from metagpt.actions.refine_project_management import RefineTasks +from metagpt.actions.write_code_refine import WriteCodeRefine from metagpt.const import WORKSPACE_ROOT from metagpt.logs import logger from metagpt.roles import Role @@ -72,6 +75,8 @@ class Engineer(Role): use_code_review: bool = False, legacy: str = "", increment: bool = False, + bug_msgs: List = None, + bug_fix: bool = False, ) -> None: """Initializes the Engineer role with given attributes.""" super().__init__(name, profile, goal, constraints) @@ -79,13 +84,20 @@ class Engineer(Role): self.use_code_review = use_code_review self.legacy = legacy self.increment = increment + self.bug_msgs = bug_msgs + self.bug_fix = bug_fix if self.use_code_review or self.increment: - self._init_actions([WriteCode, WriteCodeReview]) + self._init_actions([WriteCode, WriteCodeReview, WriteCodeRefine]) + if self.increment: self._watch([RefineTasks]) + self.todos = [] + elif self.bug_fix: + self._watch([BossRequirement]) + self.todos = [] else: self._watch([WriteTasks]) - self.todos = [] + self.todos = [] self.n_borg = n_borg @classmethod @@ -94,6 +106,18 @@ class Engineer(Role): return task_msg.instruct_content.dict().get("Task list") return CodeParser.parse_file_list(block="Task list", text=task_msg.content) + @classmethod + def parse_todos(self, bug_context: List) -> list[str]: + for msg in bug_context: + if msg.sent_from == "ProjectManager": + content = msg.content + tasks_section = re.search(r"## Task list\n\n(.*?)(\n\n|$)", content, re.DOTALL) + if tasks_section: + tasks = tasks_section.group(1).split("\n") + tasks = [task.strip("-").strip() for task in tasks] + return tasks + return [] + @classmethod def parse_code(self, code_text: str) -> str: return CodeParser.parse_code(block="", text=code_text) @@ -107,13 +131,17 @@ class Engineer(Role): def get_workspace(self) -> Path: if self.increment: msg = self._rc.memory.get_by_action(RefineDesign)[-1] + elif self.bug_fix: + msg = self._rc.memory.get_by_action(BossRequirement)[-1] else: msg = self._rc.memory.get_by_action(WriteDesign)[-1] if not msg: return WORKSPACE_ROOT / "src" workspace = self.parse_workspace(msg) + workspace = workspace if workspace else "src" # Codes are written in workspace/{package_name}/{package_name} return WORKSPACE_ROOT / workspace / workspace + # return self.create_or_increment_workspace(WORKSPACE_ROOT, workspace) def recreate_workspace(self): workspace = self.get_workspace() @@ -123,8 +151,7 @@ class Engineer(Role): pass # The folder does not exist, but we don't care workspace.mkdir(parents=True, exist_ok=True) - def write_file(self, filename: str, code: str): - workspace = self.get_workspace() + def write_file(self, workspace: Path, filename: str, code: str) -> Path: filename = filename.replace('"', "").replace("\n", "") file = workspace / filename file.parent.mkdir(parents=True, exist_ok=True) @@ -134,7 +161,10 @@ class Engineer(Role): def recv(self, message: Message) -> None: self._rc.memory.add(message) if message in self._rc.important_memory: - self.todos = self.parse_tasks(message) + if not self.bug_fix: + self.todos = self.parse_tasks(message) + else: + self.todos = self.parse_todos(self.bug_msgs) async def _act_mp(self) -> Message: # self.recreate_workspace() @@ -161,12 +191,13 @@ class Engineer(Role): async def _act_sp(self) -> Message: code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later + workspace = self.get_workspace() for todo in self.todos: code = await WriteCode().run(context=self._rc.history, filename=todo) # logger.info(todo) # logger.info(code_rsp) # code = self.parse_code(code_rsp) - file_path = self.write_file(todo, code) + file_path = self.write_file(workspace, todo, code) msg = Message(content=code, role=self.profile, cause_by=type(self._rc.todo)) self._rc.memory.add(msg) @@ -181,41 +212,76 @@ class Engineer(Role): async def _act_increment(self, legacy) -> Message: code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later + workspace = self.get_workspace() flag = True + # legacy_codes = legacy.split('---') for todo in self.todos: - """ - # Select essential information from the historical data to reduce the length of the prompt (summarized from human experience): - 1. All from Architect - 2. All from ProjectManager - 3. Do we need other codes (currently needed)? - TODO: The goal is not to need it. After clear task decomposition, based on the design idea, you should be able to write a single file without needing other codes. If you can't, it means you need a clearer definition. This is the key to writing longer code. - """ context = [] - if self.increment: - msg = self._rc.memory.get_by_actions([RefineDesign, RefineTasks, WriteCode]) - else: - msg = self._rc.memory.get_by_actions([WriteDesign, WriteTasks, WriteCode]) - + msg = self._rc.memory.get_by_actions([RefineDesign, RefineTasks, WriteCodeRefine]) for m in msg: context.append(m.content) context_str = "\n".join(context) + code = legacy + # Refine code or Write code - if flag and self.increment: - code = legacy - flag = False - else: - code = await WriteCode().run(context=context_str, filename=todo) + # if self.increment and len(legacy_codes) > 0: + # code = legacy_codes.pop(0) # Code review - if self.use_code_review: - try: - rewrite_code = await WriteCode().run(context=context_str, code=code, filename=todo) - code = rewrite_code - except Exception as e: - logger.error("code review failed!", e) - pass - file_path = self.write_file(todo, code) + try: + rewrite_code = await WriteCodeRefine().run(context=context_str, code=code, filename=todo) + code = rewrite_code + except Exception as e: + logger.error("code review failed!", e) + pass + + # code = await WriteCode().run(context=context_str, filename=todo) + + file_path = self.write_file(workspace, todo, code) + msg = Message(content=code, role=self.profile, cause_by=WriteCode) + self._rc.memory.add(msg) + + code_msg = todo + FILENAME_CODE_SEP + str(file_path) + code_msg_all.append(code_msg) + + logger.info(f"Done {self.get_workspace()} generating.") + msg = Message( + content=MSG_SEP.join(code_msg_all), role=self.profile, cause_by=type(self._rc.todo), send_to="QaEngineer" + ) + return msg + + async def _act_bug_fix(self, bug_msgs) -> Message: + code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later + workspace = self.get_workspace() + flag = True + # legacy_codes = legacy.split('---') + for todo in self.todos: + context = [] + + for m in bug_msgs: + if m.sent_from != "Engineer": + context.append(m.content) + context.append(m.content) + context_str = "\n".join(context) + code = [m.content for m in bug_msgs if m.sent_from == "Engineer"] + code = "\n".join(code) + + # Refine code or Write code + # if self.increment and len(legacy_codes) > 0: + # code = legacy_codes.pop(0) + + # Code review + try: + rewrite_code = await WriteCodeRefine().run(context=context_str, code=code, filename=todo) + code = rewrite_code + except Exception as e: + logger.error("code review failed!", e) + pass + + # code = await WriteCode().run(context=context_str, filename=todo) + + file_path = self.write_file(workspace, todo, code) msg = Message(content=code, role=self.profile, cause_by=WriteCode) self._rc.memory.add(msg) @@ -230,6 +296,7 @@ class Engineer(Role): async def _act_sp_precision(self) -> Message: code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later + workspace = self.get_workspace() for todo in self.todos: """ # Select essential information from the historical data to reduce the length of the prompt (summarized from human experience): @@ -253,7 +320,7 @@ class Engineer(Role): except Exception as e: logger.error("code review failed!", e) pass - file_path = self.write_file(todo, code) + file_path = self.write_file(workspace, todo, code) msg = Message(content=code, role=self.profile, cause_by=WriteCode) self._rc.memory.add(msg) @@ -266,14 +333,35 @@ class Engineer(Role): ) return msg + async def _observe(self) -> int: + if self.bug_fix: + msg = Message( + content=self.bug_msgs[0].content + "\n---\n" + self.legacy, + role=self.profile, + cause_by=BossRequirement, + sent_from=self.profile, + send_to=self.profile, + ) + self._publish_message(msg) + await super()._observe() + self._rc.news = [ + msg for msg in self._rc.news if msg.send_to == self.profile + ] # only relevant msgs count as observed news + return len(self._rc.news) + async def _act(self) -> Message: """Determines the mode of action based on whether code review is used.""" if self.increment: logger.info(f"{self._setting}: ready to RefineWriteCode") + elif self.bug_fix: + logger.info(f"{self._setting}: ready to BugFix") else: logger.info(f"{self._setting}: ready to WriteCode") + if self.use_code_review: return await self._act_sp_precision() elif self.increment: return await self._act_increment(self.legacy) + elif self.bug_fix: + return await self._act_bug_fix(self.bug_msgs) return await self._act_sp() diff --git a/metagpt/roles/qa_engineer.py b/metagpt/roles/qa_engineer.py index a763c2ce8..c67d42b5c 100644 --- a/metagpt/roles/qa_engineer.py +++ b/metagpt/roles/qa_engineer.py @@ -14,8 +14,9 @@ from metagpt.actions import ( WriteCode, WriteCodeReview, WriteDesign, - WriteTest, + WriteTest, BossRequirement, ) +from metagpt.actions.write_code_refine import WriteCodeRefine from metagpt.const import WORKSPACE_ROOT from metagpt.logs import logger from metagpt.roles import Role @@ -32,12 +33,20 @@ class QaEngineer(Role): goal="Write comprehensive and robust tests to ensure codes will work as expected without bugs", constraints="The test code you write should conform to code standard like PEP8, be modular, easy to read and maintain", test_round_allowed=5, + legacy="", + bug_context="", ): super().__init__(name, profile, goal, constraints) - self._init_actions( - [WriteTest] - ) # FIXME: a bit hack here, only init one action to circumvent _think() logic, will overwrite _think() in future updates - self._watch([WriteCode, WriteCodeReview, WriteTest, RunCode, DebugError]) + self.legacy = legacy + self.bug_context = bug_context + if self.bug_context: + self._init_actions([WriteTest]) + self._watch([WriteCode, WriteCodeRefine, WriteTest, RunCode, DebugError]) + else: + self._init_actions( + [WriteTest] + ) # FIXME: a bit hack here, only init one action to circumvent _think() logic, will overwrite _think() in future updates + self._watch([WriteCode, WriteCodeReview, WriteTest, RunCode, DebugError]) self.test_round = 0 self.test_round_allowed = test_round_allowed @@ -48,10 +57,14 @@ class QaEngineer(Role): return CodeParser.parse_str(block="Python package name", text=system_design_msg.content) def get_workspace(self, return_proj_dir=True) -> Path: - msg = self._rc.memory.get_by_action(WriteDesign)[-1] + if self.bug_context: + msg = self._rc.memory.get_by_action(WriteCodeRefine)[-1] + else: + msg = self._rc.memory.get_by_action(WriteDesign)[-1] if not msg: return WORKSPACE_ROOT / "src" workspace = self.parse_workspace(msg) + workspace = workspace if workspace else "src" # project directory: workspace/{package_name}, which contains package source code folder, tests folder, resources folder, etc. if return_proj_dir: return WORKSPACE_ROOT / workspace @@ -146,6 +159,15 @@ class QaEngineer(Role): self._publish_message(msg) async def _observe(self) -> int: + if self.bug_context: + msg = Message( + content=self.bug_context + "\n---\n" + self.legacy, + role=self.profile, + cause_by=BossRequirement, + sent_from=self.profile, + send_to=self.profile, + ) + self._publish_message(msg) await super()._observe() self._rc.news = [ msg for msg in self._rc.news if msg.send_to == self.profile @@ -166,7 +188,7 @@ class QaEngineer(Role): for msg in self._rc.news: # Decide what to do based on observed msg type, currently defined by human, # might potentially be moved to _think, that is, let the agent decides for itself - if msg.cause_by in [WriteCode, WriteCodeReview]: + if msg.cause_by in [WriteCode, WriteCodeReview, WriteCodeRefine]: # engineer wrote a code, time to write a test for it await self._write_test(msg) elif msg.cause_by in [WriteTest, DebugError]: diff --git a/startup.py b/startup.py index 7e8e0a692..8bb8517c9 100644 --- a/startup.py +++ b/startup.py @@ -2,6 +2,7 @@ # -*- coding: utf-8 -*- import asyncio import os +import platform import fire @@ -12,29 +13,32 @@ from metagpt.roles import ( ProjectManager, QaEngineer, ) +from metagpt.schema import Message from metagpt.team import Team +from metagpt.utils.special_tokens import MSG_SEP async def startup( idea: str, difference_description: str = "", - path: str = "", + project_path: str = "", investment: float = 3.0, n_round: int = 5, code_review: bool = False, run_tests: bool = False, implement: bool = True, increment: bool = False, + bug_fix: bool = False, ): """Run a startup. Be a boss.""" company = Team() - if increment: + if increment or bug_fix: # 读取文件 - prd_path = os.path.join(path, 'docs/prd.md') - design_path = os.path.join(path, 'docs/system_design.md') - api_spec_path = os.path.join(path, 'docs/api_spec_and_tasks.md') - code_path = os.path.join(path, os.path.basename(path)) + prd_path = os.path.join(project_path, 'docs/prd.md') + design_path = os.path.join(project_path, 'docs/system_design.md') + api_spec_and_tasks_path = os.path.join(project_path, 'docs/api_spec_and_tasks.md') + code_path = os.path.join(project_path, os.path.basename(project_path)) with open(prd_path, 'r', encoding='utf-8') as f: legacy_prd = f.read() @@ -42,34 +46,59 @@ async def startup( with open(design_path, 'r', encoding='utf-8') as f: legacy_design = f.read() - with open(api_spec_path, 'r', encoding='utf-8') as f: - legacy_api_spec = f.read() + with open(api_spec_and_tasks_path, 'r', encoding='utf-8') as f: + legacy_api_spec_and_tasks = f.read() # 遍历文件夹,获取所有代码文件 legacy_code = '' for root, dirs, files in os.walk(code_path): filenames = [filename for filename in files if filename.endswith('.py')] - legacy_code += f'There are {len(files)} scripts in the current folder: {", ".join(filenames)}\n\n' + legacy_code += f'There are {len(files)} scripts in the current folder: {", ".join(filenames)}\n---\n' for file in files: if file.endswith('.py'): with open(os.path.join(root, file), 'r', encoding='utf-8') as f: - legacy_code += f.read() + '\n\n' + legacy_code += f.read() + '\n---\n' - company.hire( - [ProductManager(difference_description=difference_description, legacy=legacy_prd, increment=increment), - Architect(legacy=legacy_design, increment=increment), - ProjectManager(legacy=legacy_api_spec, increment=increment)]) + if bug_fix: + boss_msg = Message( + content=f"Boss's requirement\n:{idea}\n---\nBoss's difference description:{difference_description}\n---\n", + sent_from="Boss", + ) + product_manager_msg = Message( + content=f"Product Manager's prd legacy:\n{legacy_prd}\n---\n", + sent_from="ProductManager" + ) + architect_msg = Message( + content=f"Architect's design legacy:\n{legacy_design}\n---\n", + sent_from="Architect" + ) + project_manager_msg = Message( + content=f"Project Manager's api spec and tasks legacy:\n{legacy_api_spec_and_tasks}\n---\n", + sent_from="ProjectManager" + ) + engineer_msg = Message( + content=f"Engineer's code legacy:\n{legacy_code}\n---\n", + sent_from="Engineer" + ) + bug_msgs = [boss_msg, product_manager_msg, architect_msg, project_manager_msg, engineer_msg] + else: + company.hire( + [ProductManager(difference_description=difference_description, legacy=legacy_prd, increment=increment), + Architect(legacy=legacy_design, increment=increment), + ProjectManager(legacy=legacy_api_spec_and_tasks, increment=increment)]) else: company.hire([ProductManager(), Architect(), ProjectManager()]) # if implement or code_review - if (implement or code_review) and not increment: + if bug_fix: + company.hire([Engineer(n_borg=5, bug_msgs=bug_msgs, bug_fix=bug_fix)]) + elif implement or code_review: # developing features: implement the idea company.hire([Engineer(n_borg=5, use_code_review=code_review)]) elif (implement or code_review) and increment: company.hire([Engineer(n_borg=5, use_code_review=code_review, legacy=legacy_code, increment=increment)]) - if run_tests: + if run_tests or bug_fix: # developing features: run tests on the spot and identify bugs # (bug fixing capability comes soon!) company.hire([QaEngineer()]) @@ -82,13 +111,14 @@ async def startup( def main( idea: str, difference_description: str = "", - path: str = "", + project_path: str = "", investment: float = 3.0, n_round: int = 5, code_review: bool = True, run_tests: bool = False, implement: bool = True, increment: bool = False, + bug_fix: bool = False, ): """ We are a software startup comprised of AI. By investing in us, @@ -100,8 +130,10 @@ def main( :param code_review: Whether to use code review. :return: """ + + asyncio.set_event_loop_policy(asyncio.WindowsProactorEventLoopPolicy()) asyncio.run( - startup(idea, difference_description, path, investment, n_round, code_review, run_tests, implement, increment)) + startup(idea, difference_description, project_path, investment, n_round, code_review, run_tests, implement, increment, bug_fix)) if __name__ == "__main__": From c159260717acb5f98c7ed3add259b5fe3db9c3d5 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 24 Nov 2023 18:56:15 +0800 Subject: [PATCH 024/637] check_param_consistency --- metagpt/tools/functions/register/register.py | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/metagpt/tools/functions/register/register.py b/metagpt/tools/functions/register/register.py index 120c7c4a2..0731e31c0 100644 --- a/metagpt/tools/functions/register/register.py +++ b/metagpt/tools/functions/register/register.py @@ -4,6 +4,7 @@ # @Author : lidanyang # @File : register.py # @Desc : +import inspect from typing import Type, Optional, Callable, Dict, Union, List from metagpt.tools.functions.schemas.base import ToolSchema @@ -13,16 +14,28 @@ class FunctionRegistry: def __init__(self): self.functions: Dict[str, Dict[str, Dict]] = {} - def register(self, module: str, tool_schema: Type[ToolSchema]) -> Callable: + @staticmethod + def _check_param_consistency(func_params, schema): + param_names = set(func_params.keys()) + schema_names = set(schema["parameters"]["properties"].keys()) + if param_names != schema_names: + raise ValueError("Function parameters do not match schema properties") + + def register(self, module: str, tool_schema: Type[ToolSchema]) -> Callable: def wrapper(func: Callable) -> Callable: module_registry = self.functions.setdefault(module, {}) if func.__name__ in module_registry: raise ValueError(f"Function {func.__name__} is already registered in {module}") + func_params = inspect.signature(func).parameters + schema = tool_schema.schema() schema["name"] = func.__name__ + + self._check_param_consistency(func_params, schema) + module_registry[func.__name__] = { "func": func, "schema": schema, From b19e4908b20d1c3217ed97edcfd75c4dc18569f2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 28 Nov 2023 11:24:52 +0800 Subject: [PATCH 025/637] fix: install missing package. --- metagpt/actions/write_analysis_code.py | 2 +- metagpt/roles/ml_engineer.py | 23 +++++++++++++++++++---- 2 files changed, 20 insertions(+), 5 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 8d8f80f4a..038f3db7f 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -30,7 +30,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): super().__init__(name, context, llm) def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): - default_system_msg = """You are Open Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step.**""" + default_system_msg = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Reuse existing code directly. Use !pip install to install missing packages.**""" # 全部转成list if not isinstance(prompt, list): prompt = [prompt] diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 480f6cecf..2e4bbfc82 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -44,6 +44,7 @@ class MLEngineer(Role): self.plan = Plan(goal=goal) self.use_tools = False self.use_task_guide = False + self.execute_code = ExecutePyCode() async def _plan_and_act(self): @@ -90,9 +91,10 @@ class MLEngineer(Role): self._rc.memory.add(Message(content=code, role="assistant", cause_by=cause_by)) - result, success = await ExecutePyCode().run(code) - print(result) - self._rc.memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) + result, success = await self.execute_code.run(code) + # truncated the result + print(self.truncate(result)) + self._rc.memory.add(Message(content=self.truncate(result), role="user", cause_by=ExecutePyCode)) # if not success: # await self._ask_review() @@ -104,7 +106,8 @@ class MLEngineer(Role): async def _ask_review(self): context = self.get_useful_memories() review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan) - self._rc.memory.add(Message(content=review, role="user", cause_by=AskReview)) + if review.lower() not in ("confirm", "y", "yes"): + self._rc.memory.add(Message(content=review, role="user", cause_by=AskReview)) return confirmed async def _update_plan(self, max_tasks: int = 3): @@ -124,6 +127,18 @@ class MLEngineer(Role): memories = super().get_memories() return memories + def truncate(self, result: str, keep_len: int = 1000) -> str: + desc = """I truncated the result to only keep the last 1000 characters\n""" + if result.startswith(desc): + result = result[-len(desc):] + + if len(result) > keep_len: + result = result[-keep_len:] + + if not result.startswith(desc): + return desc + result + return desc + if __name__ == "__main__": # requirement = "create a normal distribution and visualize it" From 311cb5e8b4f5b564b6709fe3a65c3d6b2deef75b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 28 Nov 2023 12:13:13 +0800 Subject: [PATCH 026/637] add comment for system message. --- metagpt/actions/write_analysis_code.py | 1 + 1 file changed, 1 insertion(+) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 038f3db7f..409de5a8f 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -30,6 +30,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): super().__init__(name, context, llm) def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): + # Reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt default_system_msg = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Reuse existing code directly. Use !pip install to install missing packages.**""" # 全部转成list if not isinstance(prompt, list): From 460e373dae6078c6353b755e4b3db0b5e449a300 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 28 Nov 2023 13:40:28 +0800 Subject: [PATCH 027/637] feat: add auto_run. --- metagpt/roles/ml_engineer.py | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 2e4bbfc82..af1f3b5b5 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -38,13 +38,14 @@ class WriteTaskGuide(Action): return "" class MLEngineer(Role): - def __init__(self, name="ABC", profile="MLEngineer", goal=""): + def __init__(self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") self.plan = Plan(goal=goal) self.use_tools = False self.use_task_guide = False self.execute_code = ExecutePyCode() + self.auto_run = auto_run async def _plan_and_act(self): @@ -104,11 +105,13 @@ class MLEngineer(Role): return code, result, success async def _ask_review(self): - context = self.get_useful_memories() - review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan) - if review.lower() not in ("confirm", "y", "yes"): - self._rc.memory.add(Message(content=review, role="user", cause_by=AskReview)) - return confirmed + if not self.auto_run: + context = self.get_useful_memories() + review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan) + if review.lower() not in ("confirm", "y", "yes"): + self._rc.memory.add(Message(content=review, role="user", cause_by=AskReview)) + return confirmed + return True async def _update_plan(self, max_tasks: int = 3): current_plan = str([task.json() for task in self.plan.tasks]) From 608126e1f9906bc69c6ea489674c25c0198ec9ae Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 28 Nov 2023 13:50:15 +0800 Subject: [PATCH 028/637] update context --- metagpt/actions/write_plan.py | 7 +++-- metagpt/roles/ml_engineer.py | 48 +++++++++++++++++++++++++++-------- 2 files changed, 40 insertions(+), 15 deletions(-) diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index e35ba7a92..dcfa25d55 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -15,8 +15,6 @@ class WritePlan(Action): PROMPT_TEMPLATE = """ # Context: __context__ - # Current Plan: - __current_plan__ # Task: Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to __max_tasks__ tasks. If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. @@ -32,10 +30,11 @@ class WritePlan(Action): ] ``` """ - async def run(self, context: List[Message], current_plan: str = "", max_tasks: int = 5) -> str: + async def run(self, context: List[Message], max_tasks: int = 5) -> str: prompt = ( self.PROMPT_TEMPLATE.replace("__context__", "\n".join([str(ct) for ct in context])) - .replace("__current_plan__", current_plan).replace("__max_tasks__", str(max_tasks)) + # .replace("__current_plan__", current_plan) + .replace("__max_tasks__", str(max_tasks)) ) rsp = await self._aask(prompt) rsp = CodeParser.parse_code(block=None, text=rsp) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 480f6cecf..910b94432 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -12,18 +12,27 @@ from metagpt.actions.write_plan import WritePlan from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.execute_code import ExecutePyCode +STRUCTURAL_CONTEXT = """ +## User Requirement +{user_requirement} +## Current Plan +{tasks} +## Current Task +{current_task} +""" + class AskReview(Action): async def run(self, context: List[Message], plan: Plan = None): logger.info("Current overall plan:") - logger.info("\n".join([f"{task.task_id}: {task.instruction}" for task in plan.tasks])) + logger.info("\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks])) logger.info("most recent context:") # prompt = "\n".join( # [f"{msg.cause_by.__name__ if msg.cause_by else 'Main Requirement'}: {msg.content}" for msg in context] # ) prompt = "" - latest_action = context[-1].cause_by.__name__ + latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" prompt += f"\nPlease review output from {latest_action}:\n" \ "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" \ "If you confirm the output and wish to continue with the current process, type CONFIRM:\n" @@ -44,6 +53,7 @@ class MLEngineer(Role): self.plan = Plan(goal=goal) self.use_tools = False self.use_task_guide = False + self.execute_code_action = ExecutePyCode() async def _plan_and_act(self): @@ -65,6 +75,7 @@ class MLEngineer(Role): task.code = code task.result = result self.plan.finish_current_task() + self.working_memory.clear() else: # update plan according to user's feedback and to take on changed tasks @@ -79,6 +90,11 @@ class MLEngineer(Role): while not success and counter < max_retry: context = self.get_useful_memories() + # print("*" * 10) + # print(context) + # print("*" * 10) + # breakpoint() + if not self.use_tools: # code = "print('abc')" code = await WriteCodeByGenerate().run(context=context, plan=self.plan, task_guide=task_guide) @@ -88,11 +104,11 @@ class MLEngineer(Role): code = await WriteCodeWithTools().run(context=context, plan=self.plan, task_guide=task_guide) cause_by = WriteCodeWithTools - self._rc.memory.add(Message(content=code, role="assistant", cause_by=cause_by)) + self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) - result, success = await ExecutePyCode().run(code) + result, success = await self.execute_code_action.run(code) print(result) - self._rc.memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) + self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) # if not success: # await self._ask_review() @@ -108,21 +124,31 @@ class MLEngineer(Role): return confirmed async def _update_plan(self, max_tasks: int = 3): - current_plan = str([task.json() for task in self.plan.tasks]) plan_confirmed = False while not plan_confirmed: context = self.get_useful_memories() - rsp = await WritePlan().run(context, current_plan=current_plan, max_tasks=max_tasks) - self._rc.memory.add(Message(content=rsp, role="assistant", cause_by=WritePlan)) + rsp = await WritePlan().run(context, max_tasks=max_tasks) + self.working_memory.add(Message(content=rsp, role="assistant", cause_by=WritePlan)) plan_confirmed = await self._ask_review() tasks = WritePlan.rsp_to_tasks(rsp) self.plan.add_tasks(tasks) + self.working_memory.clear() - def get_useful_memories(self, current_task_memories: List[str] = []) -> List[Message]: + def get_useful_memories(self) -> List[Message]: """find useful memories only to reduce context length and improve performance""" - memories = super().get_memories() - return memories + + user_requirement = self.plan.goal + tasks = json.dumps([task.dict() for task in self.plan.tasks], indent=4, ensure_ascii=False) + current_task = self.plan.current_task.json() if self.plan.current_task else {} + context = STRUCTURAL_CONTEXT.format(user_requirement=user_requirement, tasks=tasks, current_task=current_task) + context_msg = [Message(content=context, role="user")] + + return context_msg + self.working_memory.get() + + @property + def working_memory(self): + return self._rc.memory if __name__ == "__main__": From 0843a82a1bf40fe1111482dc16bc20ee955122c7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 28 Nov 2023 14:21:32 +0800 Subject: [PATCH 029/637] fix: module not found error. --- metagpt/roles/ml_engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 53d693c8e..56fbd2525 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -84,7 +84,7 @@ class MLEngineer(Role): # ask for acceptance, users can other refuse and change tasks in the plan task_result_confirmed = await self._ask_review() - if success and task_result_confirmed: + if success and task_result_confirmed and not code.startswith("!pip"): # tick off this task and record progress task.code = code task.result = result From 4760dfd13b84db970a32647f0d2e774999ba3c12 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 28 Nov 2023 14:50:59 +0800 Subject: [PATCH 030/637] fix: module not found. --- metagpt/roles/ml_engineer.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 56fbd2525..3f46b9451 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -84,7 +84,7 @@ class MLEngineer(Role): # ask for acceptance, users can other refuse and change tasks in the plan task_result_confirmed = await self._ask_review() - if success and task_result_confirmed and not code.startswith("!pip"): + if success and task_result_confirmed: # tick off this task and record progress task.code = code task.result = result @@ -126,6 +126,8 @@ class MLEngineer(Role): # print(result) self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) + if code.startswith("!pip"): + success = False # if not success: # await self._ask_review() From f5baa34b0fcfea16c2ebb9e7c4469a2022e01d93 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Tue, 28 Nov 2023 17:07:02 +0800 Subject: [PATCH 031/637] define prompt for ml_engineer --- metagpt/prompts/ml_engineer.py | 162 +++++++++++++++++++++++++++++++++ 1 file changed, 162 insertions(+) create mode 100644 metagpt/prompts/ml_engineer.py diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py new file mode 100644 index 000000000..7f798a098 --- /dev/null +++ b/metagpt/prompts/ml_engineer.py @@ -0,0 +1,162 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/11/24 15:43 +# @Author : lidanyang +# @File : ml_engineer +# @Desc : +TOOL_RECOMMENDATION_PROMPT = """ +## Comprehensive Task Description: +{task} + +This task is divided into several steps, and you need to select the most suitable tools for each step. A tool means a function that can be used to help you solve the task. + +## Detailed Code Steps for the Task: +{code_steps} + +## List of Available Tools: +{available_tools} + +## Tool Selection and Instructions: +- For each code step listed above, choose up to five tools that are most likely to be useful in solving the task. +- If you believe that no tools are suitable for a step, indicate with an empty list. +- Only list the names of the tools, not the full schema of each tool. +- The result should only contain tool names that are in the list of available tools. +- The result list should be in the same order as the code steps. +""" + +SELECT_FUNCTION_TOOLS = { + "name": "select_function_tools", + "description": "Given code steps to generate full code for a task, select suitable tools for each step by order.", + "parameters": { + "type": "object", + "properties": { + "recommend_tools": { + "type": "array", + "description": "List of tool names for each code step. Empty list if no tool is suitable.", + "items": { + "type": "array", + "items": { + "type": "string", + }, + }, + }, + }, + "required": ["recommend_tools"], + }, +} + + +CODE_GENERATOR_WITH_TOOLS = { + "name": "add_subtask_code", + "description": "Add new code of current subtask to the end of an active Jupyter notebook.", + "parameters": { + "type": "object", + "properties": { + "code": { + "type": "string", + "description": "The code to be added.", + }, + }, + "required": ["code"], + }, +} + +TOO_ORGANIZATION_PROMPT = """ +As a senior data scientist, your role involves developing code for a specific sub-task within a larger project. This project is divided into several sub-tasks, which may either be new challenges or extensions of previous work. + +## Sub-tasks Overview +Here's a list of all the sub-tasks, indicating their current status (DONE or TODO). Your responsibility is the first TODO task on this list. +{all_tasks} + +## Historical Code (Previously Done Sub-tasks): +This code, already executed in the Jupyter notebook, is critical for understanding the background and foundation for your current task. +```python +{completed_code} +``` + +## Dataset Description: +Details about the dataset for the project: +{data_desc} + +## Current Task Notion: +{special_prompt} + +## Code Steps for Your Sub-task: +Follow these steps to complete your current TODO task. You may use external Python functions or write custom code as needed. Ensure your code is self-contained. +{code_steps} + +When you call a function, you should import the function from `{module_name}` first, e.g.: +```python +from metagpt.tools.functions.libs.feature_engineering import fill_missing_value +``` + +## Available Functions for Each Step: +Each function is described in JSON format, including the function name and parameters. {output_desc} +{available_tools} + +## Your Output Format: +Generate the complete code for every step, listing any used function tools at the beginning of the step: +```python +# Step 1 +# Tools used: [function names or 'none'] + + +# Step 2 +# Tools used: [function names or 'none'] + + +# Continue with additional steps, following the same format... +```end + +*** Important Rules *** +- Use only the tools designated for each code step. +- Your output should only include code for the current sub-task. Don't repeat historical code. +- Only mention functions in comments if used in the code. +- Ensure the output new code is executable in the current Jupyter notebook environment, with all historical code executed. +""" + + +DATA_PREPROCESS_PROMPT = """ +In data preprocessing, closely monitor each column's data type. Apply suitable methods for various types (numerical, categorical, datetime, textual, etc.) to ensure the pandas.DataFrame is correctly formatted. +Additionally, ensure that the columns being processed must be the ones that actually exist in the dataset. +""" + +FEATURE_ENGINEERING_PROMPT = """ +""" + +CLASSIFICATION_MODEL_PROMPT = """ +""" + +REGRESSION_MODEL_PROMPT = """ +""" + + +DATA_PREPROCESS_OUTPUT_DESC = "Please note that all functions uniformly output a processed pandas.DataFrame, facilitating seamless integration into the broader workflow." + +FEATURE_ENGINEERING_OUTPUT_DESC = "" + +CLASSIFICATION_MODEL_OUTPUT_DESC = "" + +REGRESSION_MODEL_OUTPUT_DESC = "" + + +ML_SPECIFIC_PROMPT = { + "data_preprocess": DATA_PREPROCESS_PROMPT, + "feature_engineering": FEATURE_ENGINEERING_PROMPT, + "classification_model": CLASSIFICATION_MODEL_PROMPT, + "regression_model": REGRESSION_MODEL_PROMPT, +} + +TOOL_OUTPUT_DESC = { + "data_preprocess": DATA_PREPROCESS_OUTPUT_DESC, + "feature_engineering": FEATURE_ENGINEERING_OUTPUT_DESC, + "classification_model": CLASSIFICATION_MODEL_OUTPUT_DESC, + "regression_model": REGRESSION_MODEL_OUTPUT_DESC, +} + +ML_MODULE_MAP = { + "data_preprocess": "metagpt.tools.functions.libs.machine_learning.data_preprocess", + "feature_engineering": "metagpt.tools.functions.libs.machine_learning.feature_engineering", + "classification_model": "metagpt.tools.functions.libs.machine_learning.ml_model", + "regression_model": "metagpt.tools.functions.libs.machine_learning.ml_model", +} From a969d54c9ad1ebe1d59d5c49ce71a3ad8e176a65 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Tue, 28 Nov 2023 17:09:15 +0800 Subject: [PATCH 032/637] create func config --- metagpt/utils/common.py | 30 ++++++++++++++++++++++++++---- 1 file changed, 26 insertions(+), 4 deletions(-) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index f09666beb..fac6a478d 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -24,7 +24,11 @@ def check_cmd_exists(command) -> int: if platform.system().lower() == "windows": check_command = "where " + command else: - check_command = "command -v " + command + ' >/dev/null 2>&1 || { echo >&2 "no mermaid"; exit 1; }' + check_command = ( + "command -v " + + command + + ' >/dev/null 2>&1 || { echo >&2 "no mermaid"; exit 1; }' + ) result = os.system(check_command) return result @@ -134,7 +138,11 @@ class OutputParser: typing = typing_define[0] else: typing = typing_define - if typing == List[str] or typing == List[Tuple[str, str]] or typing == List[List[str]]: + if ( + typing == List[str] + or typing == List[Tuple[str, str]] + or typing == List[List[str]] + ): # 尝试解析list try: content = cls.parse_file_list(text=content) @@ -151,7 +159,9 @@ class OutputParser: return parsed_data @classmethod - def extract_struct(cls, text: str, data_type: Union[type(list), type(dict)]) -> Union[list, dict]: + def extract_struct( + cls, text: str, data_type: Union[type(list), type(dict)] + ) -> Union[list, dict]: """Extracts and parses a specified type of structure (dictionary or list) from the given text. The text only contains a list or dictionary, which may have nested structures. @@ -193,7 +203,9 @@ class OutputParser: raise ValueError(f"The extracted structure is not a {data_type}.") except (ValueError, SyntaxError) as e: - raise Exception(f"Error while extracting and parsing the {data_type}: {e}") + raise Exception( + f"Error while extracting and parsing the {data_type}: {e}" + ) else: logger.error(f"No {data_type} found in the text.") return [] if data_type is list else {} @@ -305,3 +317,13 @@ def parse_recipient(text): pattern = r"## Send To:\s*([A-Za-z]+)\s*?" # hard code for now recipient = re.search(pattern, text) return recipient.group(1) if recipient else "" + + +def create_func_config(func_schema: dict) -> dict: + """Create new function call config""" + tools = [{"type": "function", "function": func_schema}] + tool_choice = {"type": "function", "function": {"name": func_schema["name"]}} + return { + "tools": tools, + "tool_choice": tool_choice, + } From d711df0ef256dd170a7eb78931d68c981f2a3166 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 29 Nov 2023 14:24:37 +0800 Subject: [PATCH 033/637] add write code with tools --- metagpt/actions/write_analysis_code.py | 153 ++++++++++++++++++++++--- 1 file changed, 139 insertions(+), 14 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 8d8f80f4a..4694a62b9 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -4,32 +4,51 @@ @Author : orange-crow @File : write_code_v2.py """ +import json from typing import Dict, List, Union from metagpt.actions import Action +from metagpt.prompts.ml_engineer import ( + TOOL_RECOMMENDATION_PROMPT, + SELECT_FUNCTION_TOOLS, + CODE_GENERATOR_WITH_TOOLS, + TOO_ORGANIZATION_PROMPT, + ML_SPECIFIC_PROMPT, + ML_MODULE_MAP, + TOOL_OUTPUT_DESC, +) from metagpt.schema import Message, Plan +from metagpt.tools.functions import registry +from metagpt.utils.common import create_func_config + class BaseWriteAnalysisCode(Action): - - async def run(self, context: List[Message], plan: Plan = None, task_guide: str = "") -> str: + async def run( + self, context: List[Message], plan: Plan = None, task_guide: str = "" + ) -> str: """Run of a code writing action, used in data analysis or modeling Args: context (List[Message]): Action output history, source action denoted by Message.cause_by plan (Plan, optional): Overall plan. Defaults to None. task_guide (str, optional): suggested step breakdown for the current task. Defaults to "". - + Returns: str: The code string. """ + class WriteCodeByGenerate(BaseWriteAnalysisCode): """Write code fully by generation""" def __init__(self, name: str = "", context=None, llm=None) -> str: super().__init__(name, context, llm) - def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): + def process_msg( + self, + prompt: Union[str, List[Dict], Message, List[Message]], + system_msg: str = None, + ): default_system_msg = """You are Open Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step.**""" # 全部转成list if not isinstance(prompt, list): @@ -39,24 +58,38 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): messages = [] for p in prompt: if isinstance(p, str): - messages.append({'role': 'user', 'content': p}) + messages.append({"role": "user", "content": p}) elif isinstance(p, dict): messages.append(p) elif isinstance(p, Message): if isinstance(p.content, str): messages.append(p.to_dict()) - elif isinstance(p.content, dict) and 'code' in p.content: - messages.append(p.content['code']) + elif isinstance(p.content, dict) and "code" in p.content: + messages.append(p.content["code"]) # 添加默认的提示词 - if default_system_msg not in messages[0]['content'] and messages[0]['role'] != 'system': - messages.insert(0, {'role': 'system', 'content': default_system_msg}) - elif default_system_msg not in messages[0]['content'] and messages[0]['role'] == 'system': - messages[0] = {'role': 'system', 'content': messages[0]['content']+default_system_msg} + if ( + default_system_msg not in messages[0]["content"] + and messages[0]["role"] != "system" + ): + messages.insert(0, {"role": "system", "content": default_system_msg}) + elif ( + default_system_msg not in messages[0]["content"] + and messages[0]["role"] == "system" + ): + messages[0] = { + "role": "system", + "content": messages[0]["content"] + default_system_msg, + } return messages async def run( - self, context: [List[Message]], plan: Plan = None, task_guide: str = "", system_msg: str = None, **kwargs + self, + context: [List[Message]], + plan: Plan = None, + task_guide: str = "", + system_msg: str = None, + **kwargs, ) -> str: prompt = self.process_msg(context, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) @@ -66,5 +99,97 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - async def run(self, context: List[Message], plan: Plan = None, task_guide: str = "") -> str: - return "print('abc')" + @staticmethod + def _parse_recommend_tools(module: str, recommend_tools: list) -> str: + """ + Converts recommended tools to a JSON string and checks tool availability in the registry. + + Args: + module (str): The module name for querying tools in the registry. + recommend_tools (list): A list of lists of recommended tools for each step. + + Returns: + str: A JSON string with available tools and their schemas for each step. + """ + valid_tools = {} + available_tools = registry.get_all_by_module(module).keys() + for index, tools in enumerate(recommend_tools): + key = f"Step {index + 1}" + tools = [tool for tool in tools if tool in available_tools] + valid_tools[key] = registry.get_schemas(module, tools) + return json.dumps(valid_tools) + + async def _tool_recommendation( + self, task: str, code_steps: str, available_tools: list + ) -> list: + """ + Recommend tools for each step of the specified task + + Args: + task (str): the task description + code_steps (str): the code steps to generate the full code for the task + available_tools (list): the available tools for the task + + Returns: + list: recommended tools for each step of the specified task + """ + prompt = TOOL_RECOMMENDATION_PROMPT.format( + task=task, + code_steps=code_steps, + available_tools=available_tools, + ) + tool_config = create_func_config(SELECT_FUNCTION_TOOLS) + rsp = await self.llm.aask_code(prompt, **tool_config) + recommend_tools = rsp["recommend_tools"] + return recommend_tools + + async def run( + self, + context: List[Message], + plan: Plan = None, + task_guide: str = "", + data_desc: str = "", + ) -> str: + task_type = plan.current_task.task_type + task = plan.current_task.instruction + available_tools = registry.get_all_schema_by_module(task_type) + available_tools = [ + {k: tool[k] for k in ["name", "description"] if k in tool} + for tool in available_tools + ] + task_guide = "\n".join( + [f"Step {step.strip()}" for step in task_guide.split("\n")] + ) + + recommend_tools = await self._tool_recommendation( + task, task_guide, available_tools + ) + recommend_tools = self._parse_recommend_tools(task_type, recommend_tools) + + specific_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") + module_name = ML_MODULE_MAP[task_type] + output_desc = TOOL_OUTPUT_DESC.get(task_type, "") + all_tasks = "" + completed_code = "" + + for i, task in enumerate(plan.tasks): + stats = "DONE" if task.is_finished else "TODO" + all_tasks += f"Subtask {task.task_id}: {task.instruction}({stats})\n" + + for task in plan.tasks: + if task.code: + completed_code += task.code + "\n" + + prompt = TOO_ORGANIZATION_PROMPT.format( + all_tasks=all_tasks, + completed_code=completed_code, + data_desc=data_desc, + special_prompt=specific_prompt, + code_steps=task_guide, + module_name=module_name, + output_desc=output_desc, + available_tools=recommend_tools, + ) + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + rsp = await self.llm.aask_code(prompt, **tool_config) + return rsp["code"] From 25c01abaf45f53b51d66fd44029ba1acb15deb15 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 29 Nov 2023 14:55:54 +0800 Subject: [PATCH 034/637] add data_desc to WriteCodeWithTools --- metagpt/roles/ml_engineer.py | 74 ++++++++++++++++++++++++++---------- 1 file changed, 53 insertions(+), 21 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 3f46b9451..b8a258b46 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -21,10 +21,11 @@ STRUCTURAL_CONTEXT = """ {current_task} """ + def truncate(result: str, keep_len: int = 1000) -> str: desc = """I truncated the result to only keep the last 1000 characters\n""" if result.startswith(desc): - result = result[-len(desc):] + result = result[-len(desc) :] if len(result) > keep_len: result = result[-keep_len:] @@ -35,10 +36,16 @@ def truncate(result: str, keep_len: int = 1000) -> str: class AskReview(Action): - async def run(self, context: List[Message], plan: Plan = None): logger.info("Current overall plan:") - logger.info("\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks])) + logger.info( + "\n".join( + [ + f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" + for task in plan.tasks + ] + ) + ) logger.info("most recent context:") # prompt = "\n".join( @@ -46,21 +53,26 @@ class AskReview(Action): # ) prompt = "" latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" - prompt += f"\nPlease review output from {latest_action}:\n" \ - "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" \ + prompt += ( + f"\nPlease review output from {latest_action}:\n" + "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" "If you confirm the output and wish to continue with the current process, type CONFIRM:\n" + ) rsp = input(prompt) confirmed = "confirm" in rsp.lower() return rsp, confirmed -class WriteTaskGuide(Action): +class WriteTaskGuide(Action): async def run(self, task_instruction: str, data_desc: str = "") -> str: return "" + class MLEngineer(Role): - def __init__(self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False): + def __init__( + self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False + ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") self.plan = Plan(goal=goal) @@ -70,7 +82,6 @@ class MLEngineer(Role): self.auto_run = auto_run async def _plan_and_act(self): - # create initial plan and update until confirmation await self._update_plan() @@ -96,8 +107,11 @@ class MLEngineer(Role): await self._update_plan() async def _write_and_exec_code(self, max_retry: int = 3): - - task_guide = await WriteTaskGuide().run(self.plan.current_task.instruction) if self.use_task_guide else "" + task_guide = ( + await WriteTaskGuide().run(self.plan.current_task.instruction) + if self.use_task_guide + else "" + ) counter = 0 success = False @@ -109,22 +123,29 @@ class MLEngineer(Role): # print("*" * 10) # breakpoint() - if not self.use_tools: + if not self.use_tools or self.plan.current_task.task_type == "unknown": # code = "print('abc')" - code = await WriteCodeByGenerate().run(context=context, plan=self.plan, task_guide=task_guide) + code = await WriteCodeByGenerate().run( + context=context, plan=self.plan, task_guide=task_guide + ) cause_by = WriteCodeByGenerate - else: - code = await WriteCodeWithTools().run(context=context, plan=self.plan, task_guide=task_guide) + code = await WriteCodeWithTools().run( + context=context, plan=self.plan, task_guide=task_guide, data_desc="" + ) cause_by = WriteCodeWithTools - self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) + self.working_memory.add( + Message(content=code, role="assistant", cause_by=cause_by) + ) result, success = await self.execute_code.run(code) # truncated the result print(truncate(result)) # print(result) - self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) + self.working_memory.add( + Message(content=result, role="user", cause_by=ExecutePyCode) + ) if code.startswith("!pip"): success = False @@ -138,9 +159,13 @@ class MLEngineer(Role): async def _ask_review(self): if not self.auto_run: context = self.get_useful_memories() - review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan) + review, confirmed = await AskReview().run( + context=context[-5:], plan=self.plan + ) if review.lower() not in ("confirm", "y", "yes"): - self._rc.memory.add(Message(content=review, role="user", cause_by=AskReview)) + self._rc.memory.add( + Message(content=review, role="user", cause_by=AskReview) + ) return confirmed return True @@ -149,7 +174,9 @@ class MLEngineer(Role): while not plan_confirmed: context = self.get_useful_memories() rsp = await WritePlan().run(context, max_tasks=max_tasks) - self.working_memory.add(Message(content=rsp, role="assistant", cause_by=WritePlan)) + self.working_memory.add( + Message(content=rsp, role="assistant", cause_by=WritePlan) + ) plan_confirmed = await self._ask_review() tasks = WritePlan.rsp_to_tasks(rsp) @@ -160,9 +187,13 @@ class MLEngineer(Role): """find useful memories only to reduce context length and improve performance""" user_requirement = self.plan.goal - tasks = json.dumps([task.dict() for task in self.plan.tasks], indent=4, ensure_ascii=False) + tasks = json.dumps( + [task.dict() for task in self.plan.tasks], indent=4, ensure_ascii=False + ) current_task = self.plan.current_task.json() if self.plan.current_task else {} - context = STRUCTURAL_CONTEXT.format(user_requirement=user_requirement, tasks=tasks, current_task=current_task) + context = STRUCTURAL_CONTEXT.format( + user_requirement=user_requirement, tasks=tasks, current_task=current_task + ) context_msg = [Message(content=context, role="user")] return context_msg + self.working_memory.get() @@ -171,6 +202,7 @@ class MLEngineer(Role): def working_memory(self): return self._rc.memory + if __name__ == "__main__": # requirement = "create a normal distribution and visualize it" requirement = "run some analysis on iris dataset" From 047bb10e72c7f9eb290ce08845f11fa6d308b016 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 29 Nov 2023 15:00:25 +0800 Subject: [PATCH 035/637] add logic for unknown task_type --- metagpt/roles/ml_engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index b8a258b46..08c5649d4 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -123,7 +123,7 @@ class MLEngineer(Role): # print("*" * 10) # breakpoint() - if not self.use_tools or self.plan.current_task.task_type == "unknown": + if not self.use_tools or self.plan.current_task.task_type == "": # code = "print('abc')" code = await WriteCodeByGenerate().run( context=context, plan=self.plan, task_guide=task_guide From a50a1f738fba5992e3ef3cfbe917a81a6675576d Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 29 Nov 2023 17:31:15 +0800 Subject: [PATCH 036/637] update increment development, add write_code_guide.py file --- metagpt/actions/refine_design_api.py | 73 +++---- metagpt/actions/refine_prd.py | 192 ++----------------- metagpt/actions/refine_project_management.py | 118 +++++------- metagpt/actions/write_code_guide.py | 67 +++++++ metagpt/actions/write_code_refine.py | 61 +++--- metagpt/environment.py | 13 ++ metagpt/roles/architect.py | 5 +- metagpt/roles/engineer.py | 122 ++++++------ metagpt/roles/product_manager.py | 7 +- metagpt/roles/project_manager.py | 9 +- metagpt/team.py | 6 + startup.py | 66 +++---- 12 files changed, 290 insertions(+), 449 deletions(-) create mode 100644 metagpt/actions/write_code_guide.py diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py index f1c231525..909fa6db9 100644 --- a/metagpt/actions/refine_design_api.py +++ b/metagpt/actions/refine_design_api.py @@ -3,7 +3,7 @@ import re import shutil from pathlib import Path -from typing import List +from typing import List, Union from metagpt.actions import Action, ActionOutput from metagpt.config import CONFIG @@ -26,34 +26,29 @@ templates = { ## Format example {format_example} ----- -Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and the provided difference descriptions. Make the best use of good open source tools. -Requirement: Fill in the following missing information based on the context, each section name is a key in json +Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and legacy design. Make the best use of good open source tools. +Requirement: Fill in the following missing information based on the context, each section name is a key in json. Output exactly as shown in the example, including single and double quotes. Max Output: 8192 chars or 2048 tokens. Try to use them up. -## Difference Description: Provided as Python list[str], the list of differences between the new design and the legacy design +## Difference Description: Provide as list, the foremost differences description for system design here based on the previous. -## Implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. +## Incremental implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. ## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores -## File list: Provided as Python list[str], the list of ONLY REQUIRED files needed to write the program(LESS IS MORE!). Only need relative paths, comply with PEP8 standards. ALWAYS write a main.py or app.py here +## Data structures and interface definitions: Use single quotes to wrap content. Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. -## Data structures and interface definitions: Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. +## Program call flow: Use single quotes to wrap content. Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. -## Program call flow: Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. - -## Anything UNCLEAR: Provide as Plain text. Make clear here. - -output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, -and only output the json inside this tag, nothing else +output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example. +Output exactly as shown in the example, including single and double quotes, and only output the json inside this tag, nothing else """, "FORMAT_EXAMPLE": """ [CONTENT] { "Difference Description": ["The ..."], - "Implementation approach": "We will ...", - "Python package name": "snake_game", - "File list": ["main.py"], + "Incremental implementation approach": "We will ...", + "Python package name": "new_name", "Data structures and interface definitions": ' classDiagram class Game{ @@ -67,8 +62,7 @@ and only output the json inside this tag, nothing else participant M as Main ... G->>M: end game - ', - "Anything UNCLEAR": "The requirement is clear to me." + ' } [/CONTENT] """, @@ -84,24 +78,20 @@ and only output the json inside this tag, nothing else ## Format example {format_example} ----- -Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and the provided difference descriptions. Make the best use of good open source tools. -Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately +Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and legacy design. Make the best use of good open source tools. +Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately. Output exactly as shown in the example, including single and double quotes. Max Output: 8192 chars or 2048 tokens. Try to use them up. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Difference Description: Provided as Python list[str], the list of differences between the new design and the legacy design +## Difference Description: Provide as list, the foremost differences description for system design here based on the previous. -## Implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. +## Incremental implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. ## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores -## File list: Provided as Python list[str], the list of ONLY REQUIRED files needed to write the program(LESS IS MORE!). Only need relative paths, comply with PEP8 standards. ALWAYS write a main.py or app.py here +## Data structures and interface definitions: Use single quotes to wrap content. Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. -## Data structures and interface definitions: Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. - -## Program call flow: Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. - -## Anything UNCLEAR: Provide as Plain text. Make clear here. +## Program call flow: Use single quotes to wrap content. Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. """, "FORMAT_EXAMPLE": """ @@ -113,19 +103,12 @@ Attention: Use '##' to split sections, not '#', and '## ' SHOULD W ] ``` -## Implementation approach +## Incremental implementation approach We will ... ## Python package name ```python -"snake_game" -``` - -## File list -```python -[ - "main.py", -] +"new_name" ``` ## Data structures and interface definitions @@ -145,22 +128,19 @@ sequenceDiagram ... G->>M: end game ``` - -## Anything UNCLEAR -The requirement is clear to me. --- """, }, } OUTPUT_MAPPING = { - "Difference Description": (List[str], ...), - "Implementation approach": (str, ...), + # "Incremental Requirements": (str, ...), + "Difference Description": (Union[List[str], str], ...), + "Incremental implementation approach": (str, ...), "Python package name": (str, ...), - "File list": (List[str], ...), + # "File list": (List[str], ...), "Data structures and interface definitions": (str, ...), - "Program call flow": (str, ...), - "Anything UNCLEAR": (str, ...), + "Program call flow": (str, ...) } @@ -201,9 +181,6 @@ class RefineDesign(Action): async def _save_prd(self, docs_path, resources_path, context): prd_file = docs_path / "prd.md" - if context[-1].instruct_content and context[-1].instruct_content.dict()["Competitive Quadrant Chart"]: - quadrant_chart = context[-1].instruct_content.dict()["Competitive Quadrant Chart"] - await mermaid_to_file(quadrant_chart, resources_path / "competitive_analysis") if context[-1].instruct_content: logger.info(f"Saving PRD to {prd_file}") diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py index 81cb02af8..0f01b9904 100644 --- a/metagpt/actions/refine_prd.py +++ b/metagpt/actions/refine_prd.py @@ -1,4 +1,4 @@ -from typing import List +from typing import List, Union from metagpt.actions import Refine, ActionOutput, SearchAndSummarize from metagpt.config import CONFIG @@ -9,96 +9,36 @@ increment_template = { "json": { "PROMPT_TEMPLATE": """ # Context -## User's New Requirements +## User's Incremental Requirements {new_requirements} -## Difference Description -{difference_description} - ## Legacy PRD {legacy} ## Search Information {search_information} -## mermaid quadrantChart code syntax example. DONT USE QUOTO IN CODE DUE TO INVALID SYNTAX. Replace the with REAL COMPETITOR NAME -```mermaid -quadrantChart - title Reach and engagement of campaigns - x-axis Low Reach --> High Reach - y-axis Low Engagement --> High Engagement - quadrant-1 We should expand - quadrant-2 Need to promote - quadrant-3 Re-evaluate - quadrant-4 May be improved - "Campaign: A": [0.3, 0.6] - "Campaign B": [0.45, 0.23] - "Campaign C": [0.57, 0.69] - "Campaign D": [0.78, 0.34] - "Campaign E": [0.40, 0.34] - "Campaign F": [0.35, 0.78] - "Our Target Product": [0.5, 0.6] -``` - ## Format example {format_example} ----- Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. -Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design +Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive designOnly output one json, nothing else. -## New Requirements: Provide as Plain text and place the new requirements here +## Incremental Requirements: Provide as str, the foremost incremental requirements for PRD here based on the previous. -## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple +## Difference Description: Provide as str, the foremost differences description for PRD here based on the previous. ## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple -## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple - -## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less - -## Competitive Analysis: Provided as Python list[str], up to 7 competitive product analyses, consider as similar competitors as possible - -## Competitive Quadrant Chart: Use mermaid quadrantChart code syntax. up to 14 competitive products. Translation: Distribute these competitor scores evenly between 0 and 1, trying to conform to a normal distribution centered around 0.5 as much as possible. - -## Requirement Analysis: Provide as Plain text. Be simple. LESS IS MORE. Make your requirements less dumb. Delete the parts unnessasery. - -## Requirement Pool: Provided as Python list[list[str], the parameters are requirement description, priority(P0/P1/P2), respectively, comply with PEP standards; no more than 5 requirements and consider to make its difficulty lower - -## UI Design draft: Provide as Plain text. Be simple. Describe the elements and functions, also provide a simple style description and layout description. -## Anything UNCLEAR: Provide as Plain text. Make clear here. - output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, and only output the json inside this tag, nothing else """, "FORMAT_EXAMPLE": """ [CONTENT] { - "New Requirements": "", + "Incremental Requirements": "", "Difference Description": [], - "Search Information": "", - "Requirements": "", "Incremental Development Plan": [], - "Product Goals": [], - "User Stories": [], - "Competitive Analysis": [], - "Competitive Quadrant Chart": "quadrantChart - title Reach and engagement of campaigns - x-axis Low Reach --> High Reach - y-axis Low Engagement --> High Engagement - quadrant-1 We should expand - quadrant-2 Need to promote - quadrant-3 Re-evaluate - quadrant-4 May be improved - Campaign A: [0.3, 0.6] - Campaign B: [0.45, 0.23] - Campaign C: [0.57, 0.69] - Campaign D: [0.78, 0.34] - Campaign E: [0.40, 0.34] - Campaign F: [0.35, 0.78]", - "Requirement Analysis": "", - "Requirement Pool": [["P0","P0 requirement"],["P1","P1 requirement"]], - "UI Design draft": "", - "Anything UNCLEAR": "", } [/CONTENT] """, @@ -106,154 +46,58 @@ and only output the json inside this tag, nothing else "markdown": { "PROMPT_TEMPLATE": """ # Context -You need to refine the requirements based on the new requirements and the existing requirements' output. -## User's New Requirements +You need to refine the requirements based on the Incremental Requirements and the existing requirements' output. +## User's Incremental Requirements {new_requirements} -## Difference Description -{difference_description} - ## Legacy PRD {legacy} ## Search Information {search_information} -## mermaid quadrantChart code syntax example. DONT USE QUOTO IN CODE DUE TO INVALID SYNTAX. Replace the with REAL COMPETITOR NAME -```mermaid -quadrantChart - title Reach and engagement of campaigns - x-axis Low Reach --> High Reach - y-axis Low Engagement --> High Engagement - quadrant-1 We should expand - quadrant-2 Need to promote - quadrant-3 Re-evaluate - quadrant-4 May be improved - "Campaign: A": [0.3, 0.6] - "Campaign B": [0.45, 0.23] - "Campaign C": [0.57, 0.69] - "Campaign D": [0.78, 0.34] - "Campaign E": [0.40, 0.34] - "Campaign F": [0.35, 0.78] - "Our Target Product": [0.5, 0.6] -``` - ## Format example {format_example} ----- Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format. +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format.Only output one json, nothing else. -## New Requirements: Provide as Plain text and place the new requirements here +## Incremental Requirements: Provide as str, the foremost incremental requirements for PRD here based on the previous. -## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple +## Difference Description: Provide as str, the foremost differences description for PRD here based on the previous. ## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple - -## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple - -## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less - -## Competitive Analysis: Provided as Python list[str], up to 7 competitive product analyses, consider as similar competitors as possible - -## Competitive Quadrant Chart: Use mermaid quadrantChart code syntax. up to 14 competitive products. Translation: Distribute these competitor scores evenly between 0 and 1, trying to conform to a normal distribution centered around 0.5 as much as possible. - -## Requirement Analysis: Provide as Plain text. Be simple. LESS IS MORE. Make your requirements less dumb. Delete the parts unnessasery. - -## Requirement Pool: Provided as Python list[list[str], the parameters are requirement description, priority(P0/P1/P2), respectively, comply with PEP standards; no more than 5 requirements and consider to make its difficulty lower - -## UI Design draft: Provide as Plain text. Be simple. Describe the elements and functions, also provide a simple style description and layout description. -## Anything UNCLEAR: Provide as Plain text. Make clear here. """, "FORMAT_EXAMPLE": """ --- -## New Requirements +## Incremental Requirements The boss ... ## Difference Description -```python -[ - "...", -] +... ## Incremental Development Plan [ - "It ...", + "...", ] - -## Product Goals -```python -[ - "Create a ...", -] -``` - -## User Stories -```python -[ - "As a user, ...", -] -``` - -## Competitive Analysis -```python -[ - "Python Snake Game: ...", -] -``` - -## Competitive Quadrant Chart -```mermaid -quadrantChart - title Reach and engagement of campaigns - ... - "Our Target Product": [0.6, 0.7] -``` - -## Requirement Analysis -The product should be a ... - -## Requirement Pool -```python -[ - ["End game ...", "P0"] -] -``` - -## UI Design draft -Give a basic function description, and a draft - -## Anything UNCLEAR -There are no unclear points. ---- """, }, } INCREMENT_OUTPUT_MAPPING = { - "New Requirements": (str, ...), - # "Major Enhancements": (List[str], ...), - "Difference Description": (List[str], ...), + "Incremental Requirements": (str, ...), + "Difference Description": (Union[List[str], str], ...), "Incremental Development Plan": (List[str], ...), - "Product Goals": (List[str], ...), - "User Stories": (List[str], ...), - "Competitive Analysis": (List[str], ...), - "Competitive Quadrant Chart": (str, ...), - "Requirement Analysis": (str, ...), - "Requirement Pool": (List[List[str]], ...), - "UI Design draft": (str, ...), - "Anything UNCLEAR": (str, ...), } -# 对于产品经理,增量开发的动作是:RefinePDR,输出是结合新需求和已有需求输出的新的PDR class RefinePRD(Refine): def __init__(self, name="RefinePRD", context=None, llm=None): super().__init__(name, context, llm) - async def run(self, new_requirements, difference_description, legacy, format=CONFIG.prompt_format, *args, **kwargs): + async def run(self, new_requirements, legacy, format=CONFIG.prompt_format, *args, **kwargs): sas = SearchAndSummarize() rsp = "" info = f"### Search Results\n{sas.result}\n\n### Search Summary\n{rsp}" @@ -263,7 +107,7 @@ class RefinePRD(Refine): prompt_template, format_example = get_template(increment_template, format) prompt = prompt_template.format( - new_requirements=new_requirements, difference_description=difference_description, legacy=legacy, search_information=info, + new_requirements=new_requirements, legacy=legacy, search_information=info, format_example=format_example ) logger.debug(prompt) diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py index 7d9e118d8..dfeeb2db0 100644 --- a/metagpt/actions/refine_project_management.py +++ b/metagpt/actions/refine_project_management.py @@ -1,6 +1,6 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- -from typing import List +from typing import List, Union from metagpt.actions.action import Action from metagpt.config import CONFIG @@ -15,62 +15,50 @@ templates = { # Context {context} -## Legacy Design +## Legacy {legacy} ## Format example {format_example} ----- -Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a task list, and analyze task dependencies to start with the prerequisite modules. -Requirements: Based on the context, fill in the following missing information, each section name is a key in json. Here the granularity of the task is a file, if there are any missing files, you can supplement them +Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a Task list, and analyze task dependencies to start with the prerequisite modules. +Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Required Python third-party packages: Provided in requirements.txt format +## Difference Description: Provide as a python list, the foremost differences description for project management here based on the previous. -## Required Other language third-party packages: Provided in requirements.txt format +## Incremental Required Python third-party packages: Provided as a python list, the requirements.txt format -## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. +## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. -## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. +## Logic Analysis: Only files need to modified, Provided as a Python list[list[str]. If the file has no changes, the file will not be output. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first based on the previous. -## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first - -## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first - -## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. - -## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. +## Task list: Only files need to modified, provided as Python list[str]. If the file has no changes, the file will not be output. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, and only output the json inside this tag, nothing else """, "FORMAT_EXAMPLE": ''' { - "Required Python third-party packages": [ + "Incremental Requirements": "...", + "Difference Description": [ + "...", + ] + "Incremental Required Python third-party packages": [ "flask==1.1.2", "bcrypt==3.2.0" ], - "Required Other language third-party packages": [ - "No third-party ..." - ], "Full API spec": """ openapi: 3.0.0 ... description: A JSON object ... """, - "Difference Description": """ - The ... - """, "Logic Analysis": [ ["game.py","Contains..."] ], "Task list": [ "game.py" - ], - "Shared Knowledge": """ - 'game.py' contains ... - """, - "Anything UNCLEAR": "We need ... how to start." + ] } ''', }, @@ -79,48 +67,44 @@ and only output the json inside this tag, nothing else # Context {context} -## Legacy Design +## Legacy {legacy} ## Format example {format_example} ----- -Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a task list, and analyze task dependencies to start with the prerequisite modules. -Requirements: Based on the context, fill in the following missing information, note that all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file, if there are any missing files, you can supplement them +Role: You are a project manager; the goal is to perform incremental development based on the context and the legacy. Break down tasks according to PRD/technical design, provide a Task list need to modified files, and analyze task dependencies to start with the prerequisite modules. +Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Required Python third-party packages: Provided in requirements.txt format +## Difference Description: Provided as a python list, the foremost differences description for project management here based on the previous. -## Required Other language third-party packages: Provided in requirements.txt format +## Incremental Required Python third-party packages: Provided as a python list, the requirements.txt format -## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. +## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. -## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. - -## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first - -## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first - -## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. - -## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. +## Logic Analysis: Only files need to modified, Provided as a Python list[list[str]. If the file has no changes, the file will not be output. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first based on the previous. +## Task list: Only files need to modified, provided as Python list[str]. If the file has no changes, the file will not be output. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first """, "FORMAT_EXAMPLE": ''' --- -## Required Python third-party packages +## Incremental Requirements +... + +## Difference Description ```python -""" -flask==1.1.2 -bcrypt==3.2.0 -""" +[ + "The ...", +] ``` -## Required Other language third-party packages +## Incremental Required Python third-party packages ```python -""" -No third-party ... -""" +[ + "flask==1.1.2", + "bcrypt==3.2.0" +] ``` ## Full API spec @@ -132,13 +116,6 @@ description: A JSON object ... """ ``` -## Difference Description -```python -""" -The ... -""" -``` - ## Logic Analysis ```python [ @@ -152,29 +129,18 @@ The ... "game.py", ] ``` - -## Shared Knowledge -```python -""" -'game.py' contains ... -""" -``` - -## Anything UNCLEAR -We need ... how to start. --- ''', }, } OUTPUT_MAPPING = { - "Required Python third-party packages": (List[str], ...), - "Required Other language third-party packages": (List[str], ...), + # "Incremental Requirements": (str, ...), + # ## Incremental Requirements: Provided as a str, the foremost incremental requirements for project management here based on the previous. + "Difference Description": (Union[List[str], str], ...), + "Incremental Required Python third-party packages": (Union[List[str], str], ...), "Full API spec": (str, ...), - "Difference Description": (str, ...), "Logic Analysis": (List[List[str]], ...), "Task list": (List[str], ...), - "Shared Knowledge": (str, ...), - "Anything UNCLEAR": (str, ...), } @@ -192,11 +158,13 @@ class RefineTasks(Action): # Write requirements.txt requirements_path = WORKSPACE_ROOT / ws_name / "requirements.txt" - requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Required Python third-party packages"))) + requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Incremental Required Python third-party packages"))) async def run(self, context, legacy, format=CONFIG.prompt_format): prompt_template, format_example = get_template(templates, format) - prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) + prompt = prompt_template.format(context=context, + legacy=legacy, + format_example=format_example) rsp = await self._aask_v1(prompt, "task", OUTPUT_MAPPING, format=format) self._save(context, rsp) return rsp diff --git a/metagpt/actions/write_code_guide.py b/metagpt/actions/write_code_guide.py new file mode 100644 index 000000000..391173bb3 --- /dev/null +++ b/metagpt/actions/write_code_guide.py @@ -0,0 +1,67 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +from metagpt.actions.action import Action +from metagpt.logs import logger +from metagpt.schema import Message +from metagpt.utils.common import CodeParser +from tenacity import retry, stop_after_attempt, wait_fixed + + +PROMPT_TEMPLATE = """ +NOTICE +Role: You are a professional software engineer, and your main task is to conduct incremental development, proposing incremental development plans and code guideance based on context and legacy code. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". + +## Regulations Review: To make the software directly operable without further coding, follow the regulations below during incremental development: +1) Import all referenced classes. +2) Implement all methods. +3) Add necessary explanation to all methods. +4) Ensure there are no potential bugs. +5) Confirm that the entire project conforms to the tasks proposed by the user. +6) Review the code thoroughly, checking for errors and validating the logic to ensure seamless user interaction without compromising any specified requirements. + +## Incremental Development Plan: Proposed the Minimum essential incremental development plan, based on the following context and legacy code by thinking step by step. +... + +## Code guidelines: Propose the foremost guidelines that how to implement code of modification part for incremental development based on the above context, legacy code and incremental development plan. +```python +... +''' +----- +# Context +{context} + +## Legacy Code +You are tasked with conducting incremental development in the existing code based on the provided legacy code and above information. +``` +{code} +``` +----- + +## Format example +----- +## Incremental Development Guide: +... + +## Code Guidance: +# Implementation the ... +```python +... +''' +----- +""" + + +class WriteCodeGuide(Action): + def __init__(self, name="WriteCodeGuide", context: list[Message] = None, llm=None): + super().__init__(name, context, llm) + + async def run(self, context, code): + prompt = PROMPT_TEMPLATE.format(context=context, code=code) + logger.info(f'Write Code Guide ..') + code_guide = await self._aask(prompt) + # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) + # self._save(context, filename, code) + return code_guide + \ No newline at end of file diff --git a/metagpt/actions/write_code_refine.py b/metagpt/actions/write_code_refine.py index 527952aed..466c30679 100644 --- a/metagpt/actions/write_code_refine.py +++ b/metagpt/actions/write_code_refine.py @@ -7,58 +7,44 @@ from metagpt.schema import Message from metagpt.utils.common import CodeParser from tenacity import retry, stop_after_attempt, wait_fixed - PROMPT_TEMPLATE = """ NOTICE -Role: You are a professional software engineer, and your main task is to conduct incremental development, which includes reviewing existing code, providing modification suggestions, rewriting code, and optimizing the codebase. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". +Role: You are a professional engineer; your primary goal is to write PEP8 compliant, elegant, modular, easy-to-read, and maintainable Python 3.9 code (or any other programming language of your choice). +Requirements: You should modify the corresponding code based on the guidance. Then, output the complete code, fixing all errors according to the context. Ensure that you adhere to the specified guidelines for incremental development and modification of legacy code. +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format should be carefully referenced using the "Format example". Only output the current modified code, nothing else. In the modified code, if unchanged, you should output it, the complete code. -## Code Review: Based on the following context and legacy code, and following the checklist, provide key, clear, concise, and specific code modification suggestions, up to 5. -``` -1. Check 0: Is the code implemented as per the requirements? -2. Check 1: Are there any issues with the code logic? -3. Check 2: Does the existing code follow the "Data structures and interface definitions"? -4. Check 3: Is there a function in the code that is omitted or not fully implemented that needs to be implemented? -5. Check 4: Does the code have unnecessary or lack dependencies? -``` - -## Incremental Development: {filename} Based on the findings from the "Code Review," context, and the legacy code, conduct incremental development by rewriting, optimizing, and adding new code using triple quotes. +## Code: Only Write {filename}, Write code using triple quotes, based on the following list and context. +1. Do your best to implement THIS ONLY ONE FILE. ONLY USE EXISTING API. IF NO API, IMPLEMENT IT. +2. Requirement: Implement one of the following code files based on the provided context. Return the code in the specified format. Your code will be part of the entire project, so ensure it is complete, reliable, and reusable. +3. Attention1: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. +4. Attention2: YOU MUST FOLLOW "Data structures and interface definitions". DONT CHANGE ANY DESIGN. +5. Think before writing: What should be implemented and provided in this document? +6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. +7. Do not use public member functions that do not exist in your design. ----- # Context {context} -## Legacy Code -You are tasked with conducting incremental development in the existing code and creating a new code file, {filename}, based on the provided legacy code and above information. -``` -{code} -``` ----- +## Guidelines: The foremost guidelines of modification for incremental development. +{guide} +----- +## Legacy Code: The Legacy Code that needs to be modified. +{legacy} + +----- ## Format example ----- -{format_example} ------ - -""" - -FORMAT_EXAMPLE = """ - -## Code Review -1. The code ... -2. ... -3. ... -4. ... -5. ... - -## Incremental Development: {filename} +## Modified/Added Code: {filename} ```python -## {filename} - Incremental Development +# {filename} ... ``` +----- """ - class WriteCodeRefine(Action): def __init__(self, name="WriteCodeRefine", context: list[Message] = None, llm=None): super().__init__(name, context, llm) @@ -69,9 +55,8 @@ class WriteCodeRefine(Action): code = CodeParser.parse_code(block="", text=code_rsp) return code - async def run(self, context, code, filename): - format_example = FORMAT_EXAMPLE.format(filename=filename) - prompt = PROMPT_TEMPLATE.format(context=context, code=code, filename=filename, format_example=format_example) + async def run(self, context, code, filename, guide): + prompt = PROMPT_TEMPLATE.format(context=context, legacy=code, filename=filename, guide=guide) logger.info(f'Code refine {filename}..') code = await self.write_code(prompt) # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) diff --git a/metagpt/environment.py b/metagpt/environment.py index 24e6ada2f..4e409b921 100644 --- a/metagpt/environment.py +++ b/metagpt/environment.py @@ -24,6 +24,7 @@ class Environment(BaseModel): roles: dict[str, Role] = Field(default_factory=dict) memory: Memory = Field(default_factory=Memory) history: str = Field(default='') + legacy: dict = Field(default_factory=dict) class Config: arbitrary_types_allowed = True @@ -77,3 +78,15 @@ class Environment(BaseModel): get all the environment roles """ return self.roles.get(name, None) + + def set_legacy(self, legacy_dict: dict) -> None: + """设置环境的遗产 + set the environment legacy + """ + self.legacy = legacy_dict + + def get_legacy(self) -> dict: + """获得环境的遗产 + get the environment legacy + """ + return self.legacy diff --git a/metagpt/roles/architect.py b/metagpt/roles/architect.py index dddcd2a8b..f87f8899d 100644 --- a/metagpt/roles/architect.py +++ b/metagpt/roles/architect.py @@ -32,12 +32,10 @@ class Architect(Role): profile: str = "Architect", goal: str = "Design a concise, usable, complete python system", constraints: str = "Try to specify good open source tools as much as possible", - legacy: str = "", increment: bool = False, ) -> None: """Initializes the Architect with given attributes.""" super().__init__(name, profile, goal, constraints) - self.legacy = legacy self.increment = increment # Initialize actions specific to the Architect role @@ -52,7 +50,8 @@ class Architect(Role): async def _act(self) -> Message: if self.increment: logger.info(f"{self._setting}: ready to RefineDesign") - response = await self._rc.todo.run(self._rc.history, self.legacy) + legacy = self._rc.env.get_legacy()["legacy_design"] + response = await self._rc.todo.run(self._rc.history, legacy) else: logger.info(f"{self._setting}: ready to WriteDesign") diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 13034acaa..6f96aeb78 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -14,8 +14,10 @@ from typing import List from metagpt.actions import WriteCode, WriteCodeReview, WriteDesign, WriteTasks, BossRequirement from metagpt.actions.refine_design_api import RefineDesign +from metagpt.actions.refine_prd import RefinePRD from metagpt.actions.refine_project_management import RefineTasks from metagpt.actions.write_code_refine import WriteCodeRefine +from metagpt.actions.write_code_guide import WriteCodeGuide from metagpt.const import WORKSPACE_ROOT from metagpt.logs import logger from metagpt.roles import Role @@ -73,31 +75,25 @@ class Engineer(Role): constraints: str = "The code should conform to standards like PEP8 and be modular and maintainable", n_borg: int = 1, use_code_review: bool = False, - legacy: str = "", increment: bool = False, - bug_msgs: List = None, bug_fix: bool = False, ) -> None: """Initializes the Engineer role with given attributes.""" super().__init__(name, profile, goal, constraints) self._init_actions([WriteCode]) self.use_code_review = use_code_review - self.legacy = legacy self.increment = increment - self.bug_msgs = bug_msgs self.bug_fix = bug_fix - if self.use_code_review or self.increment: - self._init_actions([WriteCode, WriteCodeReview, WriteCodeRefine]) + if self.use_code_review: + self._init_actions([WriteCode, WriteCodeReview]) if self.increment: + self._init_actions([WriteCodeGuide, WriteCodeRefine, WriteCodeReview]) self._watch([RefineTasks]) - self.todos = [] - elif self.bug_fix: - self._watch([BossRequirement]) - self.todos = [] else: self._watch([WriteTasks]) - self.todos = [] + + self.todos = [] self.n_borg = n_borg @classmethod @@ -210,36 +206,50 @@ class Engineer(Role): ) return msg - async def _act_increment(self, legacy) -> Message: + async def _act_increment(self) -> Message: code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later workspace = self.get_workspace() - flag = True - # legacy_codes = legacy.split('---') + # human_str = "\n".join([msg.content for msg in self._rc.memory.get_by_role("Human")]) + human_str = str(self._rc.memory.get_by_role("Human")[0]) + code = self._rc.env.get_legacy()["legacy_code"] + + # Refine code + context = [] + msg = self._rc.memory.get_by_actions([RefinePRD, RefineDesign, RefineTasks]) + + for m in msg: + context.append(m.content) + context_str = human_str + "\n".join(context) + try: + logger.info("Write Code Guide start!") + guide = await WriteCodeGuide().run(context=context_str, code=code) + msg = Message(content=guide, role=self.profile, cause_by=WriteCodeGuide) + self._rc.memory.add(msg) + except Exception as e: + logger.error("Write Code Guide failed!", e) + pass + + # Write code or Code review for todo in self.todos: - context = [] - - msg = self._rc.memory.get_by_actions([RefineDesign, RefineTasks, WriteCodeRefine]) - for m in msg: - context.append(m.content) - context_str = "\n".join(context) - code = legacy - - # Refine code or Write code - # if self.increment and len(legacy_codes) > 0: - # code = legacy_codes.pop(0) - - # Code review + msg = self._rc.memory.get_by_actions([RefineTasks]) + context_str = human_str + "\n".join([m.content for m in msg]) + # WriteCodeRefine try: - rewrite_code = await WriteCodeRefine().run(context=context_str, code=code, filename=todo) - code = rewrite_code + logger.info("Write Code Refine start!") + code = await WriteCodeRefine().run(context=context_str, code=code, filename=todo, guide=guide) except Exception as e: - logger.error("code review failed!", e) + logger.error("Write Code Refine failed!", e) pass - - # code = await WriteCode().run(context=context_str, filename=todo) - + # FIXME: Code review Action + # if self.use_code_review: + # try: + # rewrite_code = await WriteCodeReview().run(context=context_str, code=code, filename=todo) + # code = rewrite_code + # except Exception as e: + # logger.error("code review failed!", e) + # pass file_path = self.write_file(workspace, todo, code) - msg = Message(content=code, role=self.profile, cause_by=WriteCode) + msg = Message(content=code, role=self.profile, cause_by=WriteCodeRefine) self._rc.memory.add(msg) code_msg = todo + FILENAME_CODE_SEP + str(file_path) @@ -273,7 +283,7 @@ class Engineer(Role): # Code review try: - rewrite_code = await WriteCodeRefine().run(context=context_str, code=code, filename=todo) + rewrite_code = await WriteCodeGuide().run(context=context_str, code=code, filename=todo) code = rewrite_code except Exception as e: logger.error("code review failed!", e) @@ -333,35 +343,31 @@ class Engineer(Role): ) return msg - async def _observe(self) -> int: - if self.bug_fix: - msg = Message( - content=self.bug_msgs[0].content + "\n---\n" + self.legacy, - role=self.profile, - cause_by=BossRequirement, - sent_from=self.profile, - send_to=self.profile, - ) - self._publish_message(msg) - await super()._observe() - self._rc.news = [ - msg for msg in self._rc.news if msg.send_to == self.profile - ] # only relevant msgs count as observed news - return len(self._rc.news) + # async def _observe(self) -> int: + # if self.bug_fix: + # msg = Message( + # content=self.bug_msgs[0].content + "\n---\n" + self.legacy, + # role=self.profile, + # cause_by=BossRequirement, + # sent_from=self.profile, + # send_to=self.profile, + # ) + # self._publish_message(msg) + # await super()._observe() + # self._rc.news = [ + # msg for msg in self._rc.news if msg.send_to == self.profile + # ] # only relevant msgs count as observed news + # return len(self._rc.news) async def _act(self) -> Message: """Determines the mode of action based on whether code review is used.""" if self.increment: - logger.info(f"{self._setting}: ready to RefineWriteCode") - elif self.bug_fix: - logger.info(f"{self._setting}: ready to BugFix") + logger.info(f"{self._setting}: ready to WriteExtraCode and WriteCodeRefine") else: logger.info(f"{self._setting}: ready to WriteCode") - if self.use_code_review: + if self.increment: + return await self._act_increment() + elif self.use_code_review: return await self._act_sp_precision() - elif self.increment: - return await self._act_increment(self.legacy) - elif self.bug_fix: - return await self._act_bug_fix(self.bug_msgs) return await self._act_sp() diff --git a/metagpt/roles/product_manager.py b/metagpt/roles/product_manager.py index 0b2d83ed0..f48720430 100644 --- a/metagpt/roles/product_manager.py +++ b/metagpt/roles/product_manager.py @@ -29,8 +29,6 @@ class ProductManager(Role): profile: str = "Product Manager", goal: str = "Efficiently create a successful product", constraints: str = "", - difference_description: str = "", - legacy: str = "", increment: bool = False, ) -> None: """ @@ -43,8 +41,6 @@ class ProductManager(Role): constraints (str): Constraints or limitations for the product manager. """ super().__init__(name, profile, goal, constraints) - self.difference_description = difference_description - self.legacy = legacy self.increment = increment if self.increment: @@ -56,7 +52,8 @@ class ProductManager(Role): async def _act(self) -> Message: if self.increment: logger.info(f"{self._setting}: ready to RefinePRD") - response = await self._rc.todo.run(self._rc.history, self.difference_description, self.legacy) + legacy = self._rc.env.get_legacy()["legacy_prd"] + response = await self._rc.todo.run(self._rc.history, legacy) else: logger.info(f"{self._setting}: ready to WritePRD") diff --git a/metagpt/roles/project_manager.py b/metagpt/roles/project_manager.py index 21a196d21..5d4820f4a 100644 --- a/metagpt/roles/project_manager.py +++ b/metagpt/roles/project_manager.py @@ -32,7 +32,6 @@ class ProjectManager(Role): goal: str = "Improve team efficiency and deliver with quality and quantity", constraints: str = "", increment: bool = False, - legacy: str = "", ) -> None: """ Initializes the ProjectManager role with given attributes. @@ -45,8 +44,6 @@ class ProjectManager(Role): """ super().__init__(name, profile, goal, constraints) self.increment = increment - self.legacy = legacy - if self.increment: self._init_actions([RefineTasks]) self._watch([RefineDesign]) @@ -57,7 +54,11 @@ class ProjectManager(Role): async def _act(self) -> Message: if self.increment: logger.info(f"{self._setting}: ready to RefineTasks") - response = await self._rc.todo.run(self._rc.history, self.legacy) + human_str = "\n".join([msg.content for msg in self._rc.memory.get_by_role("Human")]) + # legacy_project_management and legacy_code + legacy_dict = self._rc.env.get_legacy() + legacy_str = "Legacy Project Management:\n" + legacy_dict["legacy_project_management"] + "\nLegacy Code:\n" + legacy_dict["legacy_code"] + response = await self._rc.todo.run(self._rc.history, legacy=legacy_str) else: logger.info(f"{self._setting}: ready to WriteTasks") diff --git a/metagpt/team.py b/metagpt/team.py index 67d3ecec8..67f1973b5 100644 --- a/metagpt/team.py +++ b/metagpt/team.py @@ -59,4 +59,10 @@ class Team(BaseModel): self._check_balance() await self.environment.run() return self.environment.history + + def set_legacy(self, legacy_dict): + self.environment.legacy = legacy_dict + + def get_legacy(self): + return self.environment.legacy \ No newline at end of file diff --git a/startup.py b/startup.py index 8bb8517c9..647fc307e 100644 --- a/startup.py +++ b/startup.py @@ -28,12 +28,11 @@ async def startup( run_tests: bool = False, implement: bool = True, increment: bool = False, - bug_fix: bool = False, ): """Run a startup. Be a boss.""" company = Team() - if increment or bug_fix: + if increment: # 读取文件 prd_path = os.path.join(project_path, 'docs/prd.md') design_path = os.path.join(project_path, 'docs/system_design.md') @@ -47,64 +46,46 @@ async def startup( legacy_design = f.read() with open(api_spec_and_tasks_path, 'r', encoding='utf-8') as f: - legacy_api_spec_and_tasks = f.read() + legacy_project_management = f.read() - # 遍历文件夹,获取所有代码文件 legacy_code = '' for root, dirs, files in os.walk(code_path): - filenames = [filename for filename in files if filename.endswith('.py')] - legacy_code += f'There are {len(files)} scripts in the current folder: {", ".join(filenames)}\n---\n' for file in files: if file.endswith('.py'): with open(os.path.join(root, file), 'r', encoding='utf-8') as f: legacy_code += f.read() + '\n---\n' - if bug_fix: - boss_msg = Message( - content=f"Boss's requirement\n:{idea}\n---\nBoss's difference description:{difference_description}\n---\n", - sent_from="Boss", - ) - product_manager_msg = Message( - content=f"Product Manager's prd legacy:\n{legacy_prd}\n---\n", - sent_from="ProductManager" - ) - architect_msg = Message( - content=f"Architect's design legacy:\n{legacy_design}\n---\n", - sent_from="Architect" - ) - project_manager_msg = Message( - content=f"Project Manager's api spec and tasks legacy:\n{legacy_api_spec_and_tasks}\n---\n", - sent_from="ProjectManager" - ) - engineer_msg = Message( - content=f"Engineer's code legacy:\n{legacy_code}\n---\n", - sent_from="Engineer" - ) - bug_msgs = [boss_msg, product_manager_msg, architect_msg, project_manager_msg, engineer_msg] - else: - company.hire( - [ProductManager(difference_description=difference_description, legacy=legacy_prd, increment=increment), - Architect(legacy=legacy_design, increment=increment), - ProjectManager(legacy=legacy_api_spec_and_tasks, increment=increment)]) + legacy_dict = { + 'legacy_prd': legacy_prd, + 'legacy_design': legacy_design, + 'legacy_project_management': legacy_project_management, + 'legacy_code': legacy_code + } + company.set_legacy(legacy_dict) + company.hire( + [ + ProductManager(increment=increment), + Architect(increment=increment), + ProjectManager(increment=increment) + ] + ) else: company.hire([ProductManager(), Architect(), ProjectManager()]) # if implement or code_review - if bug_fix: - company.hire([Engineer(n_borg=5, bug_msgs=bug_msgs, bug_fix=bug_fix)]) - elif implement or code_review: + if (implement or code_review) and increment: + company.hire([Engineer(n_borg=5, use_code_review=code_review, increment=increment)]) + elif implement: # developing features: implement the idea company.hire([Engineer(n_borg=5, use_code_review=code_review)]) - elif (implement or code_review) and increment: - company.hire([Engineer(n_borg=5, use_code_review=code_review, legacy=legacy_code, increment=increment)]) - if run_tests or bug_fix: + if run_tests: # developing features: run tests on the spot and identify bugs # (bug fixing capability comes soon!) company.hire([QaEngineer()]) company.invest(investment) - company.start_project(idea) + company.start_project(idea+"\n"+difference_description) await company.run(n_round=n_round) @@ -118,7 +99,6 @@ def main( run_tests: bool = False, implement: bool = True, increment: bool = False, - bug_fix: bool = False, ): """ We are a software startup comprised of AI. By investing in us, @@ -130,10 +110,8 @@ def main( :param code_review: Whether to use code review. :return: """ - - asyncio.set_event_loop_policy(asyncio.WindowsProactorEventLoopPolicy()) asyncio.run( - startup(idea, difference_description, project_path, investment, n_round, code_review, run_tests, implement, increment, bug_fix)) + startup(idea, difference_description, project_path, investment, n_round, code_review, run_tests, implement, increment)) if __name__ == "__main__": From 452cdb7ff6d1fa6a855820530c51e71f481eea30 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 29 Nov 2023 17:56:30 +0800 Subject: [PATCH 037/637] update increment development, add write_code_guide.py file --- metagpt/roles/engineer.py | 19 +------------------ metagpt/team.py | 6 ------ startup.py | 2 +- 3 files changed, 2 insertions(+), 25 deletions(-) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 6f96aeb78..66ce12f83 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -209,13 +209,12 @@ class Engineer(Role): async def _act_increment(self) -> Message: code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later workspace = self.get_workspace() - # human_str = "\n".join([msg.content for msg in self._rc.memory.get_by_role("Human")]) human_str = str(self._rc.memory.get_by_role("Human")[0]) code = self._rc.env.get_legacy()["legacy_code"] # Refine code context = [] - msg = self._rc.memory.get_by_actions([RefinePRD, RefineDesign, RefineTasks]) + msg = self._rc.memory.get_by_actions([RefineDesign, RefineTasks]) for m in msg: context.append(m.content) @@ -343,22 +342,6 @@ class Engineer(Role): ) return msg - # async def _observe(self) -> int: - # if self.bug_fix: - # msg = Message( - # content=self.bug_msgs[0].content + "\n---\n" + self.legacy, - # role=self.profile, - # cause_by=BossRequirement, - # sent_from=self.profile, - # send_to=self.profile, - # ) - # self._publish_message(msg) - # await super()._observe() - # self._rc.news = [ - # msg for msg in self._rc.news if msg.send_to == self.profile - # ] # only relevant msgs count as observed news - # return len(self._rc.news) - async def _act(self) -> Message: """Determines the mode of action based on whether code review is used.""" if self.increment: diff --git a/metagpt/team.py b/metagpt/team.py index 67f1973b5..ea5de4a12 100644 --- a/metagpt/team.py +++ b/metagpt/team.py @@ -60,9 +60,3 @@ class Team(BaseModel): await self.environment.run() return self.environment.history - def set_legacy(self, legacy_dict): - self.environment.legacy = legacy_dict - - def get_legacy(self): - return self.environment.legacy - \ No newline at end of file diff --git a/startup.py b/startup.py index 647fc307e..60298ed36 100644 --- a/startup.py +++ b/startup.py @@ -61,7 +61,7 @@ async def startup( 'legacy_project_management': legacy_project_management, 'legacy_code': legacy_code } - company.set_legacy(legacy_dict) + company.environment.set_legacy(legacy_dict) company.hire( [ ProductManager(increment=increment), From dc2247010e1fbad48eacf621efb523efb40c1a4f Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 29 Nov 2023 20:29:15 +0800 Subject: [PATCH 038/637] reuse code --- metagpt/actions/write_analysis_code.py | 5 +++- metagpt/roles/ml_engineer.py | 38 +++++++++++++++----------- 2 files changed, 26 insertions(+), 17 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 409de5a8f..7e282b5a2 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -25,13 +25,15 @@ class BaseWriteAnalysisCode(Action): class WriteCodeByGenerate(BaseWriteAnalysisCode): """Write code fully by generation""" + DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: Use !pip install to install missing packages.**""" + REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous steps in your current code block, include new codes only, DONT repeat codes!""" def __init__(self, name: str = "", context=None, llm=None) -> str: super().__init__(name, context, llm) def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): # Reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt - default_system_msg = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Reuse existing code directly. Use !pip install to install missing packages.**""" + default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG # 全部转成list if not isinstance(prompt, list): prompt = [prompt] @@ -59,6 +61,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): async def run( self, context: [List[Message]], plan: Plan = None, task_guide: str = "", system_msg: str = None, **kwargs ) -> str: + context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) prompt = self.process_msg(context, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) return code_content["code"] diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 3f46b9451..7ad29a532 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -22,7 +22,7 @@ STRUCTURAL_CONTEXT = """ """ def truncate(result: str, keep_len: int = 1000) -> str: - desc = """I truncated the result to only keep the last 1000 characters\n""" + desc = "Truncated to show only the last 1000 characters\n" if result.startswith(desc): result = result[-len(desc):] @@ -38,19 +38,22 @@ class AskReview(Action): async def run(self, context: List[Message], plan: Plan = None): logger.info("Current overall plan:") - logger.info("\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks])) + logger.info( + "\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks]) + ) logger.info("most recent context:") - # prompt = "\n".join( - # [f"{msg.cause_by.__name__ if msg.cause_by else 'Main Requirement'}: {msg.content}" for msg in context] - # ) - prompt = "" latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" - prompt += f"\nPlease review output from {latest_action}:\n" \ + prompt = f"\nPlease review output from {latest_action}:\n" \ "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" \ - "If you confirm the output and wish to continue with the current process, type CONFIRM:\n" + "If you confirm the output and wish to continue with the current process, type CONFIRM\n" \ + "If you want to terminate the process, type exit:\n" rsp = input(prompt) - confirmed = "confirm" in rsp.lower() + + if rsp.lower() in ("exit"): + exit() + + confirmed = rsp.lower() in ("confirm", "yes", "y") return rsp, confirmed @@ -126,7 +129,7 @@ class MLEngineer(Role): # print(result) self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) - if code.startswith("!pip"): + if "!pip" in code: success = False # if not success: # await self._ask_review() @@ -139,8 +142,8 @@ class MLEngineer(Role): if not self.auto_run: context = self.get_useful_memories() review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan) - if review.lower() not in ("confirm", "y", "yes"): - self._rc.memory.add(Message(content=review, role="user", cause_by=AskReview)) + if not confirmed: + self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) return confirmed return True @@ -172,11 +175,14 @@ class MLEngineer(Role): return self._rc.memory if __name__ == "__main__": - # requirement = "create a normal distribution and visualize it" - requirement = "run some analysis on iris dataset" + requirement = "Run data analysis on sklearn Iris dataset, include a plot" + # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" + # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" + # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" + # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" - async def main(requirement: str = requirement): - role = MLEngineer(goal=requirement) + async def main(requirement: str = requirement, auto_run: bool = False): + role = MLEngineer(goal=requirement, auto_run=auto_run) await role.run(requirement) fire.Fire(main) From 41c507aa6e00650c8ac7de98a58fb47b3a4e18bb Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 30 Nov 2023 14:06:54 +0800 Subject: [PATCH 039/637] rollback format --- metagpt/utils/common.py | 20 ++++---------------- 1 file changed, 4 insertions(+), 16 deletions(-) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index fac6a478d..8f8edbc6d 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -24,11 +24,7 @@ def check_cmd_exists(command) -> int: if platform.system().lower() == "windows": check_command = "where " + command else: - check_command = ( - "command -v " - + command - + ' >/dev/null 2>&1 || { echo >&2 "no mermaid"; exit 1; }' - ) + check_command = "command -v " + command + ' >/dev/null 2>&1 || { echo >&2 "no mermaid"; exit 1; }' result = os.system(check_command) return result @@ -138,11 +134,7 @@ class OutputParser: typing = typing_define[0] else: typing = typing_define - if ( - typing == List[str] - or typing == List[Tuple[str, str]] - or typing == List[List[str]] - ): + if typing == List[str] or typing == List[Tuple[str, str]] or typing == List[List[str]]: # 尝试解析list try: content = cls.parse_file_list(text=content) @@ -159,9 +151,7 @@ class OutputParser: return parsed_data @classmethod - def extract_struct( - cls, text: str, data_type: Union[type(list), type(dict)] - ) -> Union[list, dict]: + def extract_struct(cls, text: str, data_type: Union[type(list), type(dict)]) -> Union[list, dict]: """Extracts and parses a specified type of structure (dictionary or list) from the given text. The text only contains a list or dictionary, which may have nested structures. @@ -203,9 +193,7 @@ class OutputParser: raise ValueError(f"The extracted structure is not a {data_type}.") except (ValueError, SyntaxError) as e: - raise Exception( - f"Error while extracting and parsing the {data_type}: {e}" - ) + raise Exception(f"Error while extracting and parsing the {data_type}: {e}") else: logger.error(f"No {data_type} found in the text.") return [] if data_type is list else {} From ae7fecd201d1f677c891f133bf775564c4d6ad28 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 30 Nov 2023 14:08:59 +0800 Subject: [PATCH 040/637] add data_desc to tool recommendation --- metagpt/actions/write_analysis_code.py | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 4694a62b9..787fb8d3e 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -120,13 +120,14 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): return json.dumps(valid_tools) async def _tool_recommendation( - self, task: str, code_steps: str, available_tools: list + self, task: str, data_desc: str, code_steps: str, available_tools: list ) -> list: """ Recommend tools for each step of the specified task Args: task (str): the task description + data_desc (str): the description of the dataset for the task code_steps (str): the code steps to generate the full code for the task available_tools (list): the available tools for the task @@ -135,6 +136,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): """ prompt = TOOL_RECOMMENDATION_PROMPT.format( task=task, + data_desc=data_desc, code_steps=code_steps, available_tools=available_tools, ) @@ -166,7 +168,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): ) recommend_tools = self._parse_recommend_tools(task_type, recommend_tools) - specific_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") + special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") module_name = ML_MODULE_MAP[task_type] output_desc = TOOL_OUTPUT_DESC.get(task_type, "") all_tasks = "" @@ -184,7 +186,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): all_tasks=all_tasks, completed_code=completed_code, data_desc=data_desc, - special_prompt=specific_prompt, + special_prompt=special_prompt, code_steps=task_guide, module_name=module_name, output_desc=output_desc, From f61dd32cf74cf0b5294056d4ef01c312e4594bb6 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 30 Nov 2023 14:14:05 +0800 Subject: [PATCH 041/637] add feature engineering prompt --- metagpt/prompts/ml_engineer.py | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 7f798a098..55ac27d82 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -8,6 +8,10 @@ TOOL_RECOMMENDATION_PROMPT = """ ## Comprehensive Task Description: {task} +## Dataset Description: +Details about the dataset for the project: +{data_desc} + This task is divided into several steps, and you need to select the most suitable tools for each step. A tool means a function that can be used to help you solve the task. ## Detailed Code Steps for the Task: @@ -122,6 +126,11 @@ Additionally, ensure that the columns being processed must be the ones that actu """ FEATURE_ENGINEERING_PROMPT = """ +When performing feature engineering, please adhere to the following principles: +- For specific user requests (such as removing a feature, creating a new feature based on existing data), directly generate the corresponding code. +- In cases of unclear user requirements, write feature engineering code that you believe will most improve model performance. This may include feature transformation, combination, aggregation, etc., with a limit of five features at a time. +- Ensure that the feature you're working with is indeed present in the dataset and consider the data type (numerical, categorical, etc.) and application scenario (classification, regression tasks, etc.). +- Importantly, provide detailed comments explaining the purpose of each feature and how it might enhance model performance, especially when the features are generated based on semantic understanding without clear user directives. """ CLASSIFICATION_MODEL_PROMPT = """ @@ -130,10 +139,9 @@ CLASSIFICATION_MODEL_PROMPT = """ REGRESSION_MODEL_PROMPT = """ """ - DATA_PREPROCESS_OUTPUT_DESC = "Please note that all functions uniformly output a processed pandas.DataFrame, facilitating seamless integration into the broader workflow." -FEATURE_ENGINEERING_OUTPUT_DESC = "" +FEATURE_ENGINEERING_OUTPUT_DESC = "Please note that all functions uniformly output updated pandas.DataFrame with feature engineering applied." CLASSIFICATION_MODEL_OUTPUT_DESC = "" From 3461b1b4c02e3891935c854448857d3c3d888a3c Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 30 Nov 2023 14:40:51 +0800 Subject: [PATCH 042/637] add unit tests for reuse code --- metagpt/actions/write_analysis_code.py | 5 +- .../actions/test_write_analysis_code.py | 166 ++++++++++++++++-- 2 files changed, 150 insertions(+), 21 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 7e282b5a2..51cfa6d49 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -25,14 +25,13 @@ class BaseWriteAnalysisCode(Action): class WriteCodeByGenerate(BaseWriteAnalysisCode): """Write code fully by generation""" - DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: Use !pip install to install missing packages.**""" - REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous steps in your current code block, include new codes only, DONT repeat codes!""" + DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: Use !pip install in a standalone block to install missing packages.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt + REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def __init__(self, name: str = "", context=None, llm=None) -> str: super().__init__(name, context, llm) def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): - # Reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG # 全部转成list if not isinstance(prompt, list): diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index cde5fa7ad..80d9438af 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -1,26 +1,10 @@ +import asyncio import pytest from metagpt.actions.write_analysis_code import WriteCodeByGenerate from metagpt.actions.execute_code import ExecutePyCode from metagpt.schema import Message - - -# @pytest.mark.asyncio -# async def test_write_code(): -# write_code = WriteCodeFunction() -# code = await write_code.run("Write a hello world code.") -# assert len(code) > 0 -# print(code) - - -# @pytest.mark.asyncio -# async def test_write_code_by_list_prompt(): -# write_code = WriteCodeFunction() -# msg = ["a=[1,2,5,10,-10]", "写出求a中最大值的代码python"] -# code = await write_code.run(msg) -# assert len(code) > 0 -# print(code) - +from metagpt.logs import logger @pytest.mark.asyncio async def test_write_code_by_list_plan(): @@ -37,3 +21,149 @@ async def test_write_code_by_list_plan(): output = await execute_code.run(code) print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") messages.append(output[0]) + +@pytest.mark.asyncio +async def test_write_code_to_correct_error(): + + structural_context = """ + ## User Requirement + read a dataset test.csv and print its head + ## Current Plan + [ + { + "task_id": "1", + "dependent_task_ids": [], + "instruction": "import pandas and load the dataset from 'test.csv'.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + }, + { + "task_id": "2", + "dependent_task_ids": [ + "1" + ], + "instruction": "Print the head of the dataset to display the first few rows.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + } + ] + ## Current Task + {"task_id": "1", "dependent_task_ids": [], "instruction": "import pandas and load the dataset from 'test.csv'.", "task_type": "", "code": "", "result": "", "is_finished": false} + """ + wrong_code = """import pandas as pd\ndata = pd.read_excel('test.csv')\ndata""" # use read_excel to read a csv + error = """ + Traceback (most recent call last): + File "", line 2, in + File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 478, in read_excel + io = ExcelFile(io, storage_options=storage_options, engine=engine) + File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 1500, in __init__ + raise ValueError( + ValueError: Excel file format cannot be determined, you must specify an engine manually. + """ + context = [ + Message(content=structural_context, role="user"), + Message(content=wrong_code, role="assistant"), + Message(content=error, role="user"), + ] + new_code = await WriteCodeByGenerate().run(context=context) + print(new_code) + assert "read_csv" in new_code # should correct read_excel to read_csv + +@pytest.mark.asyncio +async def test_write_code_reuse_code_simple(): + structural_context = """ + ## User Requirement + read a dataset test.csv and print its head + ## Current Plan + [ + { + "task_id": "1", + "dependent_task_ids": [], + "instruction": "import pandas and load the dataset from 'test.csv'.", + "task_type": "", + "code": "import pandas as pd\ndata = pd.read_csv('test.csv')", + "result": "", + "is_finished": true + }, + { + "task_id": "2", + "dependent_task_ids": [ + "1" + ], + "instruction": "Print the head of the dataset to display the first few rows.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + } + ] + ## Current Task + {"task_id": "2", "dependent_task_ids": ["1"], "instruction": "Print the head of the dataset to display the first few rows.", "task_type": "", "code": "", "result": "", "is_finished": false} + """ + context = [ + Message(content=structural_context, role="user"), + ] + code = await WriteCodeByGenerate().run(context=context) + print(code) + assert "pandas" not in code and "read_csv" not in code # should reuse import and read statement from previous one + +@pytest.mark.asyncio +async def test_write_code_reuse_code_long(): + """test code reuse for long context""" + + structural_context = """ + ## User Requirement + Run data analysis on sklearn Iris dataset, include a plot + ## Current Plan + [ + { + "task_id": "1", + "dependent_task_ids": [], + "instruction": "Load the Iris dataset from sklearn.", + "task_type": "", + "code": "from sklearn.datasets import load_iris\niris_data = load_iris()\niris_data['data'][0:5], iris_data['target'][0:5]", + "result": "(array([[5.1, 3.5, 1.4, 0.2],\n [4.9, 3. , 1.4, 0.2],\n [4.7, 3.2, 1.3, 0.2],\n [4.6, 3.1, 1.5, 0.2],\n [5. , 3.6, 1.4, 0.2]]),\n array([0, 0, 0, 0, 0]))", + "is_finished": true + }, + { + "task_id": "2", + "dependent_task_ids": [ + "1" + ], + "instruction": "Perform exploratory data analysis on the Iris dataset.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + }, + { + "task_id": "3", + "dependent_task_ids": [ + "2" + ], + "instruction": "Create a plot visualizing the Iris dataset features.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + } + ] + ## Current Task + {"task_id": "2", "dependent_task_ids": ["1"], "instruction": "Perform exploratory data analysis on the Iris dataset.", "task_type": "", "code": "", "result": "", "is_finished": false} + """ + context = [ + Message(content=structural_context, role="user"), + ] + trials_num = 5 + trials = [WriteCodeByGenerate().run(context=context) for _ in range(trials_num)] + trial_results = await asyncio.gather(*trials) + print(*trial_results, sep="\n\n***\n\n") + success = ["load_iris" not in result and "iris_data" in result \ + for result in trial_results] # should reuse iris_data from previous tasks + success_rate = sum(success) / trials_num + logger.info(f"success rate: {success_rate :.2f}") + assert success_rate >= 0.8 From 68635ff4aaac7af6abcf324a95baeb28cbd38cc2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 15:43:13 +0800 Subject: [PATCH 043/637] add typing-extensions-4.8.0 for nbclient --- requirements.txt | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/requirements.txt b/requirements.txt index c72260c04..1d1bc95a1 100644 --- a/requirements.txt +++ b/requirements.txt @@ -50,4 +50,5 @@ nbclient==0.9.0 nbformat==5.9.2 ipython==8.17.2 ipykernel==6.27.0 -scikit_learn==1.3.2 \ No newline at end of file +scikit_learn==1.3.2 +typing-extensions==4.8.0 \ No newline at end of file From b28111ab3476f6ce51beffa60819ab82f1fc28b5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 16:05:56 +0800 Subject: [PATCH 044/637] fix: "image/png" not in output["data"]. --- metagpt/actions/execute_code.py | 9 +- tests/metagpt/actions/test_execute_code.py | 114 +++++++++++++-------- 2 files changed, 76 insertions(+), 47 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 7b16d559a..981aa894c 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -17,6 +17,7 @@ from rich.syntax import Syntax from metagpt.actions import Action from metagpt.schema import Message +from metagpt.logs import logger class ExecuteCode(ABC): @@ -90,11 +91,14 @@ class ExecutePyCode(ExecuteCode, Action): if not outputs: return parsed_output - for output in outputs: + for i, output in enumerate(outputs): if output["output_type"] == "stream": parsed_output += output["text"] elif output["output_type"] == "display_data": - self.show_bytes_figure(output["data"]["image/png"], self.interaction) + if "image/png" in output["data"]: + self.show_bytes_figure(output["data"]["image/png"], self.interaction) + else: + logger.info(f"{i}th output['data'] from nbclient outputs dont have image/png, continue next output ...") elif output["output_type"] == "execute_result": parsed_output += output["data"]["text/plain"] return parsed_output @@ -136,7 +140,6 @@ class ExecutePyCode(ExecuteCode, Action): if isinstance(code, str): return code, language - if isinstance(code, dict): assert "code" in code if "language" not in code: diff --git a/tests/metagpt/actions/test_execute_code.py b/tests/metagpt/actions/test_execute_code.py index 88c5adf18..8894f2cb9 100644 --- a/tests/metagpt/actions/test_execute_code.py +++ b/tests/metagpt/actions/test_execute_code.py @@ -1,57 +1,83 @@ import pytest -from metagpt.actions import ExecutePyCode +from metagpt.actions.execute_code import ExecutePyCode from metagpt.schema import Message -@pytest.mark.asyncio -async def test_code_running(): - pi = ExecutePyCode() - output = await pi.run("print('hello world!')") - assert output.state == "done" - output = await pi.run({"code": "print('hello world!')", "language": "python"}) - assert output.state == "done" - code_msg = Message("print('hello world!')") - output = await pi.run(code_msg) - assert output.state == "done" +# @pytest.mark.asyncio +# async def test_code_running(): +# pi = ExecutePyCode() +# output = await pi.run("print('hello world!')") +# assert output[1] is True +# output = await pi.run({"code": "print('hello world!')", "language": "python"}) +# assert output[1] is True +# code_msg = Message("print('hello world!')") +# output = await pi.run(code_msg) +# assert output[1] is True + + +# @pytest.mark.asyncio +# async def test_split_code_running(): +# pi = ExecutePyCode() +# output = await pi.run("x=1\ny=2") +# output = await pi.run("z=x+y") +# output = await pi.run("assert z==3") +# assert output[1] is True + + +# @pytest.mark.asyncio +# async def test_execute_error(): +# pi = ExecutePyCode() +# output = await pi.run("z=1/0") +# assert output[1] is False + + +# @pytest.mark.asyncio +# async def test_plotting_code(): +# pi = ExecutePyCode() +# code = """ +# import numpy as np +# import matplotlib.pyplot as plt + +# # 生成随机数据 +# random_data = np.random.randn(1000) # 生成1000个符合标准正态分布的随机数 + +# # 绘制直方图 +# plt.hist(random_data, bins=30, density=True, alpha=0.7, color='blue', edgecolor='black') + +# # 添加标题和标签 +# plt.title('Histogram of Random Data') +# plt.xlabel('Value') +# plt.ylabel('Frequency') + +# # 显示图形 +# plt.show() +# """ +# output = await pi.run(code) +# assert output[1] is True @pytest.mark.asyncio -async def test_split_code_running(): - pi = ExecutePyCode() - output = await pi.run("x=1\ny=2") - output = await pi.run("z=x+y") - output = await pi.run("assert z==3") - assert output.state == "done" - - -@pytest.mark.asyncio -async def test_execute_error(): - pi = ExecutePyCode() - output = await pi.run("z=1/0") - assert output.state == "error" - - -@pytest.mark.asyncio -async def test_plotting_code(): - pi = ExecutePyCode() +async def test_plotting_bug(): code = """ - import numpy as np import matplotlib.pyplot as plt - - # 生成随机数据 - random_data = np.random.randn(1000) # 生成1000个符合标准正态分布的随机数 - - # 绘制直方图 - plt.hist(random_data, bins=30, density=True, alpha=0.7, color='blue', edgecolor='black') - - # 添加标题和标签 - plt.title('Histogram of Random Data') - plt.xlabel('Value') - plt.ylabel('Frequency') - - # 显示图形 + import seaborn as sns + import pandas as pd + from sklearn.datasets import load_iris + # Load the Iris dataset + iris_data = load_iris() + # Convert the loaded Iris dataset into a DataFrame for easier manipulation + iris_df = pd.DataFrame(iris_data['data'], columns=iris_data['feature_names']) + # Add a column for the target + iris_df['species'] = pd.Categorical.from_codes(iris_data['target'], iris_data['target_names']) + # Set the style of seaborn + sns.set(style='whitegrid') + # Create a pairplot of the iris dataset + plt.figure(figsize=(10, 8)) + pairplot = sns.pairplot(iris_df, hue='species') + # Show the plot plt.show() """ + pi = ExecutePyCode() output = await pi.run(code) - assert output.state == "done" + assert output[1] is True From 8aa096a33469fb5c3730d5c9b413772c1c7f2f8c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 16:07:12 +0800 Subject: [PATCH 045/637] fix: remove escape and color codes for output of nbclient. --- metagpt/roles/ml_engineer.py | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 1e4367372..5120a9011 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -3,6 +3,7 @@ import json import subprocess import fire +import re from metagpt.roles import Role from metagpt.actions import Action @@ -35,6 +36,13 @@ def truncate(result: str, keep_len: int = 1000) -> str: return desc +def remove_escape_and_color_codes(input_str): + # 使用正则表达式去除转义字符和颜色代码 + pattern = re.compile(r'\x1b\[[0-9;]*[mK]') + result = pattern.sub('', input_str) + return result + + class AskReview(Action): async def run(self, context: List[Message], plan: Plan = None): logger.info("Current overall plan:") @@ -137,8 +145,9 @@ class MLEngineer(Role): # truncated the result print(truncate(result)) # print(result) + _result = truncate(remove_escape_and_color_codes(result)) self.working_memory.add( - Message(content=result, role="user", cause_by=ExecutePyCode) + Message(content=_result, role="user", cause_by=ExecutePyCode) ) if "!pip" in code: From b81fefffa17233ff0654395841e8d5bdd604a225 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 30 Nov 2023 16:28:02 +0800 Subject: [PATCH 046/637] avoid repetitive tool desc between steps --- metagpt/actions/write_analysis_code.py | 22 +++++++++++++++------- metagpt/prompts/ml_engineer.py | 6 +++++- 2 files changed, 20 insertions(+), 8 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 787fb8d3e..6fff1c66f 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -5,7 +5,7 @@ @File : write_code_v2.py """ import json -from typing import Dict, List, Union +from typing import Dict, List, Union, Tuple from metagpt.actions import Action from metagpt.prompts.ml_engineer import ( @@ -100,24 +100,31 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" @staticmethod - def _parse_recommend_tools(module: str, recommend_tools: list) -> str: + def _parse_recommend_tools(module: str, recommend_tools: list) -> Tuple[Dict, List[Dict]]: """ - Converts recommended tools to a JSON string and checks tool availability in the registry. + Parses and validates a list of recommended tools, and retrieves their schema from registry. Args: module (str): The module name for querying tools in the registry. recommend_tools (list): A list of lists of recommended tools for each step. Returns: - str: A JSON string with available tools and their schemas for each step. + Tuple[Dict, List[Dict]]: + - valid_tools: A dict of lists of valid tools for each step. + - tool_catalog: A list of dicts of unique tool schemas. """ valid_tools = {} available_tools = registry.get_all_by_module(module).keys() for index, tools in enumerate(recommend_tools): key = f"Step {index + 1}" tools = [tool for tool in tools if tool in available_tools] - valid_tools[key] = registry.get_schemas(module, tools) - return json.dumps(valid_tools) + valid_tools[key] = tools + + unique_tools = set() + for tools in valid_tools.values(): + unique_tools.update(tools) + tool_catalog = registry.get_schemas(module, unique_tools) + return valid_tools, tool_catalog async def _tool_recommendation( self, task: str, data_desc: str, code_steps: str, available_tools: list @@ -166,7 +173,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): recommend_tools = await self._tool_recommendation( task, task_guide, available_tools ) - recommend_tools = self._parse_recommend_tools(task_type, recommend_tools) + recommend_tools, tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") module_name = ML_MODULE_MAP[task_type] @@ -191,6 +198,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name=module_name, output_desc=output_desc, available_tools=recommend_tools, + tool_catalog=tool_catalog, ) tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 55ac27d82..70a40ef34 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -95,9 +95,13 @@ from metagpt.tools.functions.libs.feature_engineering import fill_missing_value ``` ## Available Functions for Each Step: -Each function is described in JSON format, including the function name and parameters. {output_desc} +Here's a list of all available functions for each step. You can find more details about each function in [## Function Catalog] {available_tools} +## Function Catalog: +Each function is described in JSON format, including the function name and parameters. {output_desc} +{function_catalog} + ## Your Output Format: Generate the complete code for every step, listing any used function tools at the beginning of the step: ```python From 2dd754d97740243602edec17a4611bbaa8a0c0dd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 16:36:35 +0800 Subject: [PATCH 047/637] fix: reuse variables. --- metagpt/actions/write_analysis_code.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 66e2137fe..ee4555ee1 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -40,8 +40,8 @@ class BaseWriteAnalysisCode(Action): class WriteCodeByGenerate(BaseWriteAnalysisCode): """Write code fully by generation""" - DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: Use !pip install in a standalone block to install missing packages.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt - REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" + DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Reuse variables in other code directly. Use !pip install in a standalone block to install missing packages.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt + # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def __init__(self, name: str = "", context=None, llm=None) -> str: super().__init__(name, context, llm) @@ -89,7 +89,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): system_msg: str = None, **kwargs, ) -> str: - context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) + # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) prompt = self.process_msg(context, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) return code_content["code"] From 9d49caa8cc8566aeee5a8f8a7ad0c22d1271dae6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 17:09:43 +0800 Subject: [PATCH 048/637] test: set temperature=0.0 --- tests/metagpt/actions/test_write_analysis_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index 80d9438af..d4bccb552 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -159,7 +159,7 @@ async def test_write_code_reuse_code_long(): Message(content=structural_context, role="user"), ] trials_num = 5 - trials = [WriteCodeByGenerate().run(context=context) for _ in range(trials_num)] + trials = [WriteCodeByGenerate().run(context=context, temperature=0.0) for _ in range(trials_num)] trial_results = await asyncio.gather(*trials) print(*trial_results, sep="\n\n***\n\n") success = ["load_iris" not in result and "iris_data" in result \ From 870ece45b23dbdd27fb9407b8127865a21279d8b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 17:10:36 +0800 Subject: [PATCH 049/637] fix: set temperature=0.0 for WriteCodeByGenerate. --- metagpt/roles/ml_engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 5120a9011..f5bb559e1 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -128,7 +128,7 @@ class MLEngineer(Role): if not self.use_tools or self.plan.current_task.task_type == "": # code = "print('abc')" code = await WriteCodeByGenerate().run( - context=context, plan=self.plan, task_guide=task_guide + context=context, plan=self.plan, task_guide=task_guide, temperature=0.0 ) cause_by = WriteCodeByGenerate else: From c2dba151fbe139291d8fd185aea87e15a04a093a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 17:42:55 +0800 Subject: [PATCH 050/637] add unit test : write_code_reuse_code_long_for_wine. --- .../actions/test_write_analysis_code.py | 274 +++++++++++------- 1 file changed, 173 insertions(+), 101 deletions(-) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index d4bccb552..1a727a9e4 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -6,110 +6,110 @@ from metagpt.actions.execute_code import ExecutePyCode from metagpt.schema import Message from metagpt.logs import logger -@pytest.mark.asyncio -async def test_write_code_by_list_plan(): - write_code = WriteCodeByGenerate() - execute_code = ExecutePyCode() - messages = [] - plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] - for task in plan: - print(f"\n任务: {task}\n\n") - messages.append(Message(task, role='assistant')) - code = await write_code.run(messages) - messages.append(Message(code, role='assistant')) - assert len(code) > 0 - output = await execute_code.run(code) - print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") - messages.append(output[0]) +# @pytest.mark.asyncio +# async def test_write_code_by_list_plan(): +# write_code = WriteCodeByGenerate() +# execute_code = ExecutePyCode() +# messages = [] +# plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] +# for task in plan: +# print(f"\n任务: {task}\n\n") +# messages.append(Message(task, role='assistant')) +# code = await write_code.run(messages) +# messages.append(Message(code, role='assistant')) +# assert len(code) > 0 +# output = await execute_code.run(code) +# print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") +# messages.append(output[0]) -@pytest.mark.asyncio -async def test_write_code_to_correct_error(): +# @pytest.mark.asyncio +# async def test_write_code_to_correct_error(): - structural_context = """ - ## User Requirement - read a dataset test.csv and print its head - ## Current Plan - [ - { - "task_id": "1", - "dependent_task_ids": [], - "instruction": "import pandas and load the dataset from 'test.csv'.", - "task_type": "", - "code": "", - "result": "", - "is_finished": false - }, - { - "task_id": "2", - "dependent_task_ids": [ - "1" - ], - "instruction": "Print the head of the dataset to display the first few rows.", - "task_type": "", - "code": "", - "result": "", - "is_finished": false - } - ] - ## Current Task - {"task_id": "1", "dependent_task_ids": [], "instruction": "import pandas and load the dataset from 'test.csv'.", "task_type": "", "code": "", "result": "", "is_finished": false} - """ - wrong_code = """import pandas as pd\ndata = pd.read_excel('test.csv')\ndata""" # use read_excel to read a csv - error = """ - Traceback (most recent call last): - File "", line 2, in - File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 478, in read_excel - io = ExcelFile(io, storage_options=storage_options, engine=engine) - File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 1500, in __init__ - raise ValueError( - ValueError: Excel file format cannot be determined, you must specify an engine manually. - """ - context = [ - Message(content=structural_context, role="user"), - Message(content=wrong_code, role="assistant"), - Message(content=error, role="user"), - ] - new_code = await WriteCodeByGenerate().run(context=context) - print(new_code) - assert "read_csv" in new_code # should correct read_excel to read_csv +# structural_context = """ +# ## User Requirement +# read a dataset test.csv and print its head +# ## Current Plan +# [ +# { +# "task_id": "1", +# "dependent_task_ids": [], +# "instruction": "import pandas and load the dataset from 'test.csv'.", +# "task_type": "", +# "code": "", +# "result": "", +# "is_finished": false +# }, +# { +# "task_id": "2", +# "dependent_task_ids": [ +# "1" +# ], +# "instruction": "Print the head of the dataset to display the first few rows.", +# "task_type": "", +# "code": "", +# "result": "", +# "is_finished": false +# } +# ] +# ## Current Task +# {"task_id": "1", "dependent_task_ids": [], "instruction": "import pandas and load the dataset from 'test.csv'.", "task_type": "", "code": "", "result": "", "is_finished": false} +# """ +# wrong_code = """import pandas as pd\ndata = pd.read_excel('test.csv')\ndata""" # use read_excel to read a csv +# error = """ +# Traceback (most recent call last): +# File "", line 2, in +# File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 478, in read_excel +# io = ExcelFile(io, storage_options=storage_options, engine=engine) +# File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 1500, in __init__ +# raise ValueError( +# ValueError: Excel file format cannot be determined, you must specify an engine manually. +# """ +# context = [ +# Message(content=structural_context, role="user"), +# Message(content=wrong_code, role="assistant"), +# Message(content=error, role="user"), +# ] +# new_code = await WriteCodeByGenerate().run(context=context) +# print(new_code) +# assert "read_csv" in new_code # should correct read_excel to read_csv -@pytest.mark.asyncio -async def test_write_code_reuse_code_simple(): - structural_context = """ - ## User Requirement - read a dataset test.csv and print its head - ## Current Plan - [ - { - "task_id": "1", - "dependent_task_ids": [], - "instruction": "import pandas and load the dataset from 'test.csv'.", - "task_type": "", - "code": "import pandas as pd\ndata = pd.read_csv('test.csv')", - "result": "", - "is_finished": true - }, - { - "task_id": "2", - "dependent_task_ids": [ - "1" - ], - "instruction": "Print the head of the dataset to display the first few rows.", - "task_type": "", - "code": "", - "result": "", - "is_finished": false - } - ] - ## Current Task - {"task_id": "2", "dependent_task_ids": ["1"], "instruction": "Print the head of the dataset to display the first few rows.", "task_type": "", "code": "", "result": "", "is_finished": false} - """ - context = [ - Message(content=structural_context, role="user"), - ] - code = await WriteCodeByGenerate().run(context=context) - print(code) - assert "pandas" not in code and "read_csv" not in code # should reuse import and read statement from previous one +# @pytest.mark.asyncio +# async def test_write_code_reuse_code_simple(): +# structural_context = """ +# ## User Requirement +# read a dataset test.csv and print its head +# ## Current Plan +# [ +# { +# "task_id": "1", +# "dependent_task_ids": [], +# "instruction": "import pandas and load the dataset from 'test.csv'.", +# "task_type": "", +# "code": "import pandas as pd\ndata = pd.read_csv('test.csv')", +# "result": "", +# "is_finished": true +# }, +# { +# "task_id": "2", +# "dependent_task_ids": [ +# "1" +# ], +# "instruction": "Print the head of the dataset to display the first few rows.", +# "task_type": "", +# "code": "", +# "result": "", +# "is_finished": false +# } +# ] +# ## Current Task +# {"task_id": "2", "dependent_task_ids": ["1"], "instruction": "Print the head of the dataset to display the first few rows.", "task_type": "", "code": "", "result": "", "is_finished": false} +# """ +# context = [ +# Message(content=structural_context, role="user"), +# ] +# code = await WriteCodeByGenerate().run(context=context) +# print(code) +# assert "pandas" not in code and "read_csv" not in code # should reuse import and read statement from previous one @pytest.mark.asyncio async def test_write_code_reuse_code_long(): @@ -167,3 +167,75 @@ async def test_write_code_reuse_code_long(): success_rate = sum(success) / trials_num logger.info(f"success rate: {success_rate :.2f}") assert success_rate >= 0.8 + + +@pytest.mark.asyncio +async def test_write_code_reuse_code_long_for_wine(): + """test code reuse for long context""" + + structural_context = """ + ## User Requirement + Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy + ## Current Plan + [ + { + "task_id": "1", + "dependent_task_ids": [], + "instruction": "Load the sklearn Wine recognition dataset and perform exploratory data analysis." + "task_type": "", + "code": "from sklearn.datasets import load_wine\n# Load the Wine recognition dataset\nwine_data = load_wine()\n# Perform exploratory data analysis\nwine_data.keys()", + "result": "Truncated to show only the last 1000 characters\ndict_keys(['data', 'target', 'frame', 'target_names', 'DESCR', 'feature_names'])", + "is_finished": true + }, + { + "task_id": "2", + "dependent_task_ids": ["1"], + "instruction": "Create a plot to visualize some aspect of the wine dataset." + "task_type": "", + "code": "", + "result": "", + "is_finished": false + }, + { + "task_id": "3", + "dependent_task_ids": ["1"], + "instruction": "Split the dataset into training and validation sets with a 20% validation size.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + }, + { + "task_id": "4", + "dependent_task_ids": ["3"], + "instruction": "Train a model on the training set to predict wine class.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + }, + { + "task_id": "5", + "dependent_task_ids": ["4"], + "instruction": "Evaluate the model on the validation set and report the accuracy.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + } + ] + ## Current Task + {"task_id": "2", "dependent_task_ids": ["1"], "instruction": "Create a plot to visualize some aspect of the Wine dataset.", "task_type": "", "code": "", "result": "", "is_finished": false} + """ + context = [ + Message(content=structural_context, role="user"), + ] + trials_num = 5 + trials = [WriteCodeByGenerate().run(context=context, temperature=0.0) for _ in range(trials_num)] + trial_results = await asyncio.gather(*trials) + print(*trial_results, sep="\n\n***\n\n") + success = ["load_wine" not in result\ + for result in trial_results] # should reuse iris_data from previous tasks + success_rate = sum(success) / trials_num + logger.info(f"success rate: {success_rate :.2f}") + assert success_rate >= 0.8 From 25c536f3e10f9f08584c07b23ceca16dab85dc0a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 17:44:22 +0800 Subject: [PATCH 051/637] fix: reuse variables in code. --- metagpt/actions/write_analysis_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index ee4555ee1..2b56d6fc1 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -40,7 +40,7 @@ class BaseWriteAnalysisCode(Action): class WriteCodeByGenerate(BaseWriteAnalysisCode): """Write code fully by generation""" - DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Reuse variables in other code directly. Use !pip install in a standalone block to install missing packages.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt + DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def __init__(self, name: str = "", context=None, llm=None) -> str: From 87acf9b4535f6269a1869d0e054e7c713c04b82d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 17:48:24 +0800 Subject: [PATCH 052/637] chore --- tests/metagpt/actions/test_write_analysis_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index 1a727a9e4..e0c3c5230 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -234,7 +234,7 @@ async def test_write_code_reuse_code_long_for_wine(): trials = [WriteCodeByGenerate().run(context=context, temperature=0.0) for _ in range(trials_num)] trial_results = await asyncio.gather(*trials) print(*trial_results, sep="\n\n***\n\n") - success = ["load_wine" not in result\ + success = ["load_wine" not in result and "wine_data" in result\ for result in trial_results] # should reuse iris_data from previous tasks success_rate = sum(success) / trials_num logger.info(f"success rate: {success_rate :.2f}") From 897d1bf0d0c737d465679a38352659485d80e570 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 17:49:38 +0800 Subject: [PATCH 053/637] chore --- .../actions/test_write_analysis_code.py | 202 +++++++++--------- 1 file changed, 101 insertions(+), 101 deletions(-) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index e0c3c5230..211c6ba13 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -6,110 +6,110 @@ from metagpt.actions.execute_code import ExecutePyCode from metagpt.schema import Message from metagpt.logs import logger -# @pytest.mark.asyncio -# async def test_write_code_by_list_plan(): -# write_code = WriteCodeByGenerate() -# execute_code = ExecutePyCode() -# messages = [] -# plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] -# for task in plan: -# print(f"\n任务: {task}\n\n") -# messages.append(Message(task, role='assistant')) -# code = await write_code.run(messages) -# messages.append(Message(code, role='assistant')) -# assert len(code) > 0 -# output = await execute_code.run(code) -# print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") -# messages.append(output[0]) +@pytest.mark.asyncio +async def test_write_code_by_list_plan(): + write_code = WriteCodeByGenerate() + execute_code = ExecutePyCode() + messages = [] + plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] + for task in plan: + print(f"\n任务: {task}\n\n") + messages.append(Message(task, role='assistant')) + code = await write_code.run(messages) + messages.append(Message(code, role='assistant')) + assert len(code) > 0 + output = await execute_code.run(code) + print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") + messages.append(output[0]) -# @pytest.mark.asyncio -# async def test_write_code_to_correct_error(): +@pytest.mark.asyncio +async def test_write_code_to_correct_error(): -# structural_context = """ -# ## User Requirement -# read a dataset test.csv and print its head -# ## Current Plan -# [ -# { -# "task_id": "1", -# "dependent_task_ids": [], -# "instruction": "import pandas and load the dataset from 'test.csv'.", -# "task_type": "", -# "code": "", -# "result": "", -# "is_finished": false -# }, -# { -# "task_id": "2", -# "dependent_task_ids": [ -# "1" -# ], -# "instruction": "Print the head of the dataset to display the first few rows.", -# "task_type": "", -# "code": "", -# "result": "", -# "is_finished": false -# } -# ] -# ## Current Task -# {"task_id": "1", "dependent_task_ids": [], "instruction": "import pandas and load the dataset from 'test.csv'.", "task_type": "", "code": "", "result": "", "is_finished": false} -# """ -# wrong_code = """import pandas as pd\ndata = pd.read_excel('test.csv')\ndata""" # use read_excel to read a csv -# error = """ -# Traceback (most recent call last): -# File "", line 2, in -# File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 478, in read_excel -# io = ExcelFile(io, storage_options=storage_options, engine=engine) -# File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 1500, in __init__ -# raise ValueError( -# ValueError: Excel file format cannot be determined, you must specify an engine manually. -# """ -# context = [ -# Message(content=structural_context, role="user"), -# Message(content=wrong_code, role="assistant"), -# Message(content=error, role="user"), -# ] -# new_code = await WriteCodeByGenerate().run(context=context) -# print(new_code) -# assert "read_csv" in new_code # should correct read_excel to read_csv + structural_context = """ + ## User Requirement + read a dataset test.csv and print its head + ## Current Plan + [ + { + "task_id": "1", + "dependent_task_ids": [], + "instruction": "import pandas and load the dataset from 'test.csv'.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + }, + { + "task_id": "2", + "dependent_task_ids": [ + "1" + ], + "instruction": "Print the head of the dataset to display the first few rows.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + } + ] + ## Current Task + {"task_id": "1", "dependent_task_ids": [], "instruction": "import pandas and load the dataset from 'test.csv'.", "task_type": "", "code": "", "result": "", "is_finished": false} + """ + wrong_code = """import pandas as pd\ndata = pd.read_excel('test.csv')\ndata""" # use read_excel to read a csv + error = """ + Traceback (most recent call last): + File "", line 2, in + File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 478, in read_excel + io = ExcelFile(io, storage_options=storage_options, engine=engine) + File "/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py", line 1500, in __init__ + raise ValueError( + ValueError: Excel file format cannot be determined, you must specify an engine manually. + """ + context = [ + Message(content=structural_context, role="user"), + Message(content=wrong_code, role="assistant"), + Message(content=error, role="user"), + ] + new_code = await WriteCodeByGenerate().run(context=context) + print(new_code) + assert "read_csv" in new_code # should correct read_excel to read_csv -# @pytest.mark.asyncio -# async def test_write_code_reuse_code_simple(): -# structural_context = """ -# ## User Requirement -# read a dataset test.csv and print its head -# ## Current Plan -# [ -# { -# "task_id": "1", -# "dependent_task_ids": [], -# "instruction": "import pandas and load the dataset from 'test.csv'.", -# "task_type": "", -# "code": "import pandas as pd\ndata = pd.read_csv('test.csv')", -# "result": "", -# "is_finished": true -# }, -# { -# "task_id": "2", -# "dependent_task_ids": [ -# "1" -# ], -# "instruction": "Print the head of the dataset to display the first few rows.", -# "task_type": "", -# "code": "", -# "result": "", -# "is_finished": false -# } -# ] -# ## Current Task -# {"task_id": "2", "dependent_task_ids": ["1"], "instruction": "Print the head of the dataset to display the first few rows.", "task_type": "", "code": "", "result": "", "is_finished": false} -# """ -# context = [ -# Message(content=structural_context, role="user"), -# ] -# code = await WriteCodeByGenerate().run(context=context) -# print(code) -# assert "pandas" not in code and "read_csv" not in code # should reuse import and read statement from previous one +@pytest.mark.asyncio +async def test_write_code_reuse_code_simple(): + structural_context = """ + ## User Requirement + read a dataset test.csv and print its head + ## Current Plan + [ + { + "task_id": "1", + "dependent_task_ids": [], + "instruction": "import pandas and load the dataset from 'test.csv'.", + "task_type": "", + "code": "import pandas as pd\ndata = pd.read_csv('test.csv')", + "result": "", + "is_finished": true + }, + { + "task_id": "2", + "dependent_task_ids": [ + "1" + ], + "instruction": "Print the head of the dataset to display the first few rows.", + "task_type": "", + "code": "", + "result": "", + "is_finished": false + } + ] + ## Current Task + {"task_id": "2", "dependent_task_ids": ["1"], "instruction": "Print the head of the dataset to display the first few rows.", "task_type": "", "code": "", "result": "", "is_finished": false} + """ + context = [ + Message(content=structural_context, role="user"), + ] + code = await WriteCodeByGenerate().run(context=context) + print(code) + assert "pandas" not in code and "read_csv" not in code # should reuse import and read statement from previous one @pytest.mark.asyncio async def test_write_code_reuse_code_long(): From f440ff69d04768da1f8183cb4386d36bd9886456 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 30 Nov 2023 18:04:55 +0800 Subject: [PATCH 054/637] chore --- tests/metagpt/actions/test_execute_code.py | 82 +++++++++++----------- 1 file changed, 41 insertions(+), 41 deletions(-) diff --git a/tests/metagpt/actions/test_execute_code.py b/tests/metagpt/actions/test_execute_code.py index 8894f2cb9..73b5886dc 100644 --- a/tests/metagpt/actions/test_execute_code.py +++ b/tests/metagpt/actions/test_execute_code.py @@ -4,57 +4,57 @@ from metagpt.actions.execute_code import ExecutePyCode from metagpt.schema import Message -# @pytest.mark.asyncio -# async def test_code_running(): -# pi = ExecutePyCode() -# output = await pi.run("print('hello world!')") -# assert output[1] is True -# output = await pi.run({"code": "print('hello world!')", "language": "python"}) -# assert output[1] is True -# code_msg = Message("print('hello world!')") -# output = await pi.run(code_msg) -# assert output[1] is True +@pytest.mark.asyncio +async def test_code_running(): + pi = ExecutePyCode() + output = await pi.run("print('hello world!')") + assert output[1] is True + output = await pi.run({"code": "print('hello world!')", "language": "python"}) + assert output[1] is True + code_msg = Message("print('hello world!')") + output = await pi.run(code_msg) + assert output[1] is True -# @pytest.mark.asyncio -# async def test_split_code_running(): -# pi = ExecutePyCode() -# output = await pi.run("x=1\ny=2") -# output = await pi.run("z=x+y") -# output = await pi.run("assert z==3") -# assert output[1] is True +@pytest.mark.asyncio +async def test_split_code_running(): + pi = ExecutePyCode() + output = await pi.run("x=1\ny=2") + output = await pi.run("z=x+y") + output = await pi.run("assert z==3") + assert output[1] is True -# @pytest.mark.asyncio -# async def test_execute_error(): -# pi = ExecutePyCode() -# output = await pi.run("z=1/0") -# assert output[1] is False +@pytest.mark.asyncio +async def test_execute_error(): + pi = ExecutePyCode() + output = await pi.run("z=1/0") + assert output[1] is False -# @pytest.mark.asyncio -# async def test_plotting_code(): -# pi = ExecutePyCode() -# code = """ -# import numpy as np -# import matplotlib.pyplot as plt +@pytest.mark.asyncio +async def test_plotting_code(): + pi = ExecutePyCode() + code = """ + import numpy as np + import matplotlib.pyplot as plt -# # 生成随机数据 -# random_data = np.random.randn(1000) # 生成1000个符合标准正态分布的随机数 + # 生成随机数据 + random_data = np.random.randn(1000) # 生成1000个符合标准正态分布的随机数 -# # 绘制直方图 -# plt.hist(random_data, bins=30, density=True, alpha=0.7, color='blue', edgecolor='black') + # 绘制直方图 + plt.hist(random_data, bins=30, density=True, alpha=0.7, color='blue', edgecolor='black') -# # 添加标题和标签 -# plt.title('Histogram of Random Data') -# plt.xlabel('Value') -# plt.ylabel('Frequency') + # 添加标题和标签 + plt.title('Histogram of Random Data') + plt.xlabel('Value') + plt.ylabel('Frequency') -# # 显示图形 -# plt.show() -# """ -# output = await pi.run(code) -# assert output[1] is True + # 显示图形 + plt.show() + """ + output = await pi.run(code) + assert output[1] is True @pytest.mark.asyncio From 8b3d640dd60b3accce7845744f24522a8ec1bd22 Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 1 Dec 2023 00:44:47 +0800 Subject: [PATCH 055/637] add kaggle manager --- kaggle_team.py | 37 +++++++++ metagpt/roles/kaggle_manager.py | 129 ++++++++++++++++++++++++++++++++ metagpt/schema.py | 1 + 3 files changed, 167 insertions(+) create mode 100644 kaggle_team.py create mode 100644 metagpt/roles/kaggle_manager.py diff --git a/kaggle_team.py b/kaggle_team.py new file mode 100644 index 000000000..0743d445b --- /dev/null +++ b/kaggle_team.py @@ -0,0 +1,37 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +import asyncio + +import fire + +from metagpt.roles.kaggle_manager import KaggleManager +from metagpt.roles.ml_engineer import MLEngineer +from metagpt.team import Team + +async def main( + # competition: str, + # data_desc: str, + # requirement: str, + investment: float = 3.0, + n_round: int = 5, +): + competition, data_desc, requirement = ( + "titanic", + "Training set is train.csv.\nTest set is test.csv. We also include gender_submission.csv, a set of predictions that assume all and only female passengers survive, as an example of what a submission file should look like.", + "Run EDA on the train dataset, train a model to predict survival (20% as validation) and save it, predict the test set using saved model, save the test result according to format", + ) + + team = Team() + team.hire( + [ + KaggleManager(competition=competition, data_desc=data_desc), + MLEngineer(goal=requirement), + ] + ) + + team.invest(investment) + team.start_project(requirement) + await team.run(n_round=n_round) + +if __name__ == '__main__': + fire.Fire(main) diff --git a/metagpt/roles/kaggle_manager.py b/metagpt/roles/kaggle_manager.py new file mode 100644 index 000000000..e902d99a0 --- /dev/null +++ b/metagpt/roles/kaggle_manager.py @@ -0,0 +1,129 @@ +from typing import Dict, List, Union, Tuple +import json +import subprocess + +import fire +import pandas as pd + +from metagpt.const import WORKSPACE_ROOT +from metagpt.roles import Role +from metagpt.actions import Action, BossRequirement +from metagpt.actions.write_analysis_code import AskReview, SummarizeAnalysis +from metagpt.schema import Message, Task, Plan +from metagpt.logs import logger + +import os +os.environ["KAGGLE_USERNAME"] = "xxx" +os.environ["KAGGLE_KEY"] = "xxx" + +def run_command(cmd): + print(cmd) + output = subprocess.run(cmd, shell=True, capture_output=True, text=True) + if output.returncode != 0: + print("Error output:", output.stderr) + exit() + else: + print(output.stdout) + return output.stdout + +class DownloadData(Action): + + async def run(self, competition, data_desc="") -> str: + data_path = WORKSPACE_ROOT / competition + + output = run_command(f"kaggle competitions list --search {competition}") + assert output != "No competitions found", "You must provide the correct competition name" + + run_command(f"kaggle competitions download {competition} --path {WORKSPACE_ROOT}") + + # if not os.path.exists(data_path): + if True: + run_command(f"unzip -o {WORKSPACE_ROOT / '*.zip'} -d {data_path}") # FIXME: not safe + + file_list = run_command(f"ls {data_path}") + + rsp = f""" + Location: + Data downloaded at {data_path} folder, including {file_list} + Data Description: + {data_desc} + """ + return rsp + +class SubmitResult(Action): + PROMPT_TEMPLATE = """ + # Context + {context} + # Your task + Extract the prediction file for test set, return only the path string, e.g., xxx.csv, xxx.xlsx + """ + + def __init__(self, name: str = "", context=None, llm=None) -> str: + super().__init__(name, context, llm) + + async def _parse_submit_file_path(self, context) -> str: + prompt = self.PROMPT_TEMPLATE.format(context=context) + rsp = await self._aask(prompt) + return rsp + + async def run(self, competition, submit_message="") -> str: + submit_file_path = self._parse_submit_file_path(submit_message) + + data_path = WORKSPACE_ROOT / competition + + run_command(f"kaggle competitions submit {competition} -f {submit_file_path} -m '{submit_message}'") + run_command(f"kaggle competitions leaderboard --show --csv {competition} > {data_path / 'leaderboard.csv'}") + run_command(f"kaggle competitions submissions --csv {competition} > {data_path / 'submission.csv'}") + + leaderboard = pd.read_csv(data_path / 'leaderboard.csv') + submission = pd.read_csv(data_path / 'submission.csv') + submission_score = submission.loc[0, "publicScore"] + submission_rank = leaderboard.loc[leaderboard["score"] == submission_score].index[0] + submission_rank_pct = round(submission_rank / len(leaderboard), 4) * 100 + + # best_score = max(submission["publicScore"]) + # best_rank = leaderboard.loc[leaderboard["score"] == best_score].index[0] + + submission_summary = f""" + ## All History + {submission.to_json(orient="records")} + ## Current + Current submission score: {submission_score}, rank: {submission_rank} (top {submission_rank_pct}%); + """ + print(submission_summary) + return submission_summary + + +class KaggleManager(Role): + def __init__( + self, name="ABC", profile="KaggleManager", goal="", competition="titanic", data_desc="" + ): + super().__init__(name=name, profile=profile, goal=goal) + self._init_actions([DownloadData, SubmitResult]) + self._watch([BossRequirement, SummarizeAnalysis]) + self.competition = competition + self.data_desc = data_desc # currently passed in, later can be scrapped down from web by another Role + + async def _think(self): + observed = self.get_memories()[-1].cause_by + if observed == BossRequirement: + self._set_state(0) # DownloadData, get competition of interest from human, download datasets + elif observed == SummarizeAnalysis: + self._set_state(1) # SubmitResult, get prediction from MLEngineer and submit it to Kaggle + elif observed == SubmitResult: + self._set_state(2) # AskReview, ask human for improvement + + async def _act(self): + todo = self._rc.todo + logger.info(f"{self._setting}: ready to {self._rc.todo}") + + if isinstance(todo, DownloadData): + rsp = await todo.run(self.competition, self.data_desc) + + elif isinstance(todo, SubmitResult): + submit_message = self.get_memories()[-1].content # use analysis summary from MLEngineer as submission message + rsp = await todo.run(competition=self.competition, submit_message=submit_message) + + msg = Message(content=rsp, role="user", cause_by=type(todo)) + + return msg diff --git a/metagpt/schema.py b/metagpt/schema.py index e39f54a0c..601bdcea2 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -85,6 +85,7 @@ class Task(BaseModel): class Plan(BaseModel): goal: str + context: str = "" tasks: list[Task] = [] task_map: dict[str, Task] = {} current_task_id = "" From aad201e06f288778ddd1fea20640a761d8afc62e Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 1 Dec 2023 11:57:58 +0800 Subject: [PATCH 056/637] assign task_type for task --- metagpt/actions/write_plan.py | 33 +++++++++++++++++++++++----- metagpt/prompts/ml_engineer.py | 40 +++++++++++++++++++++++++++++----- 2 files changed, 63 insertions(+), 10 deletions(-) diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index dcfa25d55..5e42de199 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -4,12 +4,14 @@ @Author : orange-crow @File : plan.py """ -from typing import List +from typing import List, Dict import json from metagpt.actions import Action +from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_PROMPT, ASSIGN_TASK_TYPE from metagpt.schema import Message, Task -from metagpt.utils.common import CodeParser +from metagpt.utils.common import CodeParser, create_func_config + class WritePlan(Action): PROMPT_TEMPLATE = """ @@ -30,7 +32,28 @@ class WritePlan(Action): ] ``` """ - async def run(self, context: List[Message], max_tasks: int = 5) -> str: + + async def assign_task_type(self, tasks: List[Dict]) -> List[Dict]: + """Assign task type to each task in tasks + + Args: + tasks (List[Dict]): tasks to be assigned task type + + Returns: + List[Dict]: tasks with task type assigned + """ + task_list = "\n".join( + [f"Task {task['task_id']}: {task['instruction']}" for task in tasks] + ) + prompt = ASSIGN_TASK_TYPE_PROMPT.format(task_list=task_list) + tool_config = create_func_config(ASSIGN_TASK_TYPE) + rsp = await self.llm.aask_code(prompt, **tool_config) + task_type_list = rsp["task_type"] + for task, task_type in zip(tasks, task_type_list): + task["task_type"] = task_type + return tasks + + async def run(self, context: List[Message], max_tasks: int = 5) -> List[Dict]: prompt = ( self.PROMPT_TEMPLATE.replace("__context__", "\n".join([str(ct) for ct in context])) # .replace("__current_plan__", current_plan) @@ -38,10 +61,10 @@ class WritePlan(Action): ) rsp = await self._aask(prompt) rsp = CodeParser.parse_code(block=None, text=rsp) + rsp = await self.assign_task_type(json.loads(rsp)) return rsp @staticmethod - def rsp_to_tasks(rsp: str) -> List[Task]: - rsp = json.loads(rsp) + def rsp_to_tasks(rsp: List[Dict]) -> List[Task]: tasks = [Task(**task_config) for task_config in rsp] return tasks diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 70a40ef34..0c4d036fc 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -4,6 +4,35 @@ # @Author : lidanyang # @File : ml_engineer # @Desc : +ASSIGN_TASK_TYPE_PROMPT = """ +## All Task Type: +- **data_preprocess**: Only involve cleaning and preparing data through techniques like imputation, scaling, and encoding, not containing reading data, feature engineering, model training, etc. +- **feature_engineering**: Involves enhancing data features through techniques like encoding, aggregation, time component analysis, and creating polynomial and interaction features, etc. +- **other**: Any tasks that do not fit into the previous categories, such as visualization, summarizing findings, build model, etc. + +Please assign a task type to each task in the list below from the given categories: +{task_list} +""" + +ASSIGN_TASK_TYPE = { + "name": "assign_task_type", + "description": "assign task type to each task by order", + "parameters": { + "type": "object", + "properties": { + "task_type": { + "type": "array", + "description": "List of task type.", + "items": { + "type": "string", + }, + }, + }, + "required": ["task_type"], + }, +} + + TOOL_RECOMMENDATION_PROMPT = """ ## Comprehensive Task Description: {task} @@ -137,11 +166,12 @@ When performing feature engineering, please adhere to the following principles: - Importantly, provide detailed comments explaining the purpose of each feature and how it might enhance model performance, especially when the features are generated based on semantic understanding without clear user directives. """ -CLASSIFICATION_MODEL_PROMPT = """ +MODEL_TRAIN_PROMPT = """ +When selecting and training a model, please follow these guidelines to ensure optimal performance: +- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as lightGBM, XGBoost, CatBoost, etc. +— If user specifies a model, use that model. Otherwise, use the model you believe will best solve the problem. """ -REGRESSION_MODEL_PROMPT = """ -""" DATA_PREPROCESS_OUTPUT_DESC = "Please note that all functions uniformly output a processed pandas.DataFrame, facilitating seamless integration into the broader workflow." @@ -155,8 +185,8 @@ REGRESSION_MODEL_OUTPUT_DESC = "" ML_SPECIFIC_PROMPT = { "data_preprocess": DATA_PREPROCESS_PROMPT, "feature_engineering": FEATURE_ENGINEERING_PROMPT, - "classification_model": CLASSIFICATION_MODEL_PROMPT, - "regression_model": REGRESSION_MODEL_PROMPT, + "classification_model": MODEL_TRAIN_PROMPT, + "regression_model": MODEL_TRAIN_PROMPT, } TOOL_OUTPUT_DESC = { From 35e8a501c54762bd95bddebc1b3c8367a8993238 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 1 Dec 2023 11:59:28 +0800 Subject: [PATCH 057/637] add log print --- metagpt/actions/write_analysis_code.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 6fff1c66f..e81228109 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -4,10 +4,10 @@ @Author : orange-crow @File : write_code_v2.py """ -import json from typing import Dict, List, Union, Tuple from metagpt.actions import Action +from metagpt.logs import logger from metagpt.prompts.ml_engineer import ( TOOL_RECOMMENDATION_PROMPT, SELECT_FUNCTION_TOOLS, @@ -174,6 +174,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): task, task_guide, available_tools ) recommend_tools, tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) + logger.info(f"Recommended tools for every steps: {recommend_tools}") special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") module_name = ML_MODULE_MAP[task_type] From cb8a8ffd5cbf4e13c25bab7ea51ec6736a6a9bcc Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 1 Dec 2023 13:44:24 +0800 Subject: [PATCH 058/637] fix rsp type --- metagpt/actions/write_plan.py | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index 5e42de199..f7c096f2c 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -33,7 +33,7 @@ class WritePlan(Action): ``` """ - async def assign_task_type(self, tasks: List[Dict]) -> List[Dict]: + async def assign_task_type(self, tasks: List[Dict]) -> str: """Assign task type to each task in tasks Args: @@ -51,9 +51,9 @@ class WritePlan(Action): task_type_list = rsp["task_type"] for task, task_type in zip(tasks, task_type_list): task["task_type"] = task_type - return tasks + return json.dumps(tasks) - async def run(self, context: List[Message], max_tasks: int = 5) -> List[Dict]: + async def run(self, context: List[Message], max_tasks: int = 5) -> str: prompt = ( self.PROMPT_TEMPLATE.replace("__context__", "\n".join([str(ct) for ct in context])) # .replace("__current_plan__", current_plan) @@ -65,6 +65,7 @@ class WritePlan(Action): return rsp @staticmethod - def rsp_to_tasks(rsp: List[Dict]) -> List[Task]: + def rsp_to_tasks(rsp: str) -> List[Task]: + rsp = json.loads(rsp) tasks = [Task(**task_config) for task_config in rsp] return tasks From e4a17d122c9c115530375c4f095f5a6be46ec03a Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 1 Dec 2023 14:15:36 +0800 Subject: [PATCH 059/637] fill other task_type --- metagpt/roles/ml_engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 08c5649d4..0ea73a045 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -123,7 +123,7 @@ class MLEngineer(Role): # print("*" * 10) # breakpoint() - if not self.use_tools or self.plan.current_task.task_type == "": + if not self.use_tools or self.plan.current_task.task_type == "other": # code = "print('abc')" code = await WriteCodeByGenerate().run( context=context, plan=self.plan, task_guide=task_guide From 2049f6cd01c66e5ed0402a18bebc20b1a9ceda5d Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 1 Dec 2023 14:29:51 +0800 Subject: [PATCH 060/637] only assign task_type when use_tools --- metagpt/actions/write_plan.py | 7 +++++-- metagpt/roles/ml_engineer.py | 4 +++- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index f7c096f2c..5145ffd68 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -53,7 +53,9 @@ class WritePlan(Action): task["task_type"] = task_type return json.dumps(tasks) - async def run(self, context: List[Message], max_tasks: int = 5) -> str: + async def run( + self, context: List[Message], max_tasks: int = 5, use_tools: bool = False + ) -> str: prompt = ( self.PROMPT_TEMPLATE.replace("__context__", "\n".join([str(ct) for ct in context])) # .replace("__current_plan__", current_plan) @@ -61,7 +63,8 @@ class WritePlan(Action): ) rsp = await self._aask(prompt) rsp = CodeParser.parse_code(block=None, text=rsp) - rsp = await self.assign_task_type(json.loads(rsp)) + if use_tools: + rsp = await self.assign_task_type(json.loads(rsp)) return rsp @staticmethod diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 0ea73a045..8e02e093b 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -173,7 +173,9 @@ class MLEngineer(Role): plan_confirmed = False while not plan_confirmed: context = self.get_useful_memories() - rsp = await WritePlan().run(context, max_tasks=max_tasks) + rsp = await WritePlan().run( + context, max_tasks=max_tasks, use_tools=self.use_tools + ) self.working_memory.add( Message(content=rsp, role="assistant", cause_by=WritePlan) ) From 59af6d96921fadbba22c57ea171ab0725d8e5b0d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Fri, 1 Dec 2023 15:21:40 +0800 Subject: [PATCH 061/637] chore: remove _result. --- metagpt/roles/ml_engineer.py | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index f5bb559e1..ae346579b 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -145,9 +145,8 @@ class MLEngineer(Role): # truncated the result print(truncate(result)) # print(result) - _result = truncate(remove_escape_and_color_codes(result)) self.working_memory.add( - Message(content=_result, role="user", cause_by=ExecutePyCode) + Message(content=truncate(remove_escape_and_color_codes(result)), role="user", cause_by=ExecutePyCode) ) if "!pip" in code: From f1cfbea7728084e14bd93cecbd0b8624c381cbb9 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 1 Dec 2023 15:31:38 +0800 Subject: [PATCH 062/637] add test for write code with tools --- .../actions/test_write_analysis_code.py | 74 ++++++++++++++++++- 1 file changed, 72 insertions(+), 2 deletions(-) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index cde5fa7ad..2319331d4 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -1,8 +1,8 @@ import pytest -from metagpt.actions.write_analysis_code import WriteCodeByGenerate +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.execute_code import ExecutePyCode -from metagpt.schema import Message +from metagpt.schema import Message, Plan, Task # @pytest.mark.asyncio @@ -37,3 +37,73 @@ async def test_write_code_by_list_plan(): output = await execute_code.run(code) print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") messages.append(output[0]) + + +@pytest.mark.asyncio +async def test_tool_recommendation(): + task = "对已经读取的数据集进行数据清洗" + code_steps = """ + step 1: 对数据集进行去重 + step 2: 对数据集进行缺失值处理 + """ + available_tools = [ + { + "name": "fill_missing_value", + "description": "Completing missing values with simple strategies", + }, + { + "name": "split_bins", + "description": "Bin continuous data into intervals and return the bin identifier encoded as an integer value", + }, + ] + write_code = WriteCodeWithTools() + tools = await write_code._tool_recommendation(task, code_steps, available_tools) + + assert len(tools) == 2 + assert tools[0] == [] + assert tools[1] == ["fill_missing_value"] + + +@pytest.mark.asyncio +async def test_write_code_with_tools(): + write_code = WriteCodeWithTools() + messages = [] + task_map = { + "1": Task( + task_id="1", + instruction="随机生成一个pandas DataFrame数据集", + task_type="unknown", + dependent_task_ids=[], + code=""" + import pandas as pd + df = pd.DataFrame({ + 'a': [1, 2, 3, 4, 5], + 'b': [1.1, 2.2, 3.3, 4.4, np.nan], + 'c': ['aa', 'bb', 'cc', 'dd', 'ee'], + 'd': [1, 2, 3, 4, 5] + }) + """, + is_finished=True, + ), + "2": Task( + task_id="2", + instruction="对数据集进行数据清洗", + task_type="data_preprocess", + dependent_task_ids=["1"], + ), + } + plan = Plan( + goal="构造数据集并进行数据清洗", + tasks=list(task_map.values()), + task_map=task_map, + current_task_id="2", + ) + task_guide = """ + step 1: 对数据集进行去重 + step 2: 对数据集进行缺失值处理 + """ + data_desc = "None" + + code = await write_code.run(messages, plan, task_guide, data_desc) + assert len(code) > 0 + print(code) From d3d08fe5f33cf65fcf74442d2dd754ffed1c2b7a Mon Sep 17 00:00:00 2001 From: yzlin Date: Sat, 2 Dec 2023 01:34:22 +0800 Subject: [PATCH 063/637] more plan operation, review update, add kaggle team --- config/config.yaml | 5 +- kaggle_team.py | 3 +- metagpt/actions/ml_da_action.py | 119 +++++++++++++++++++++++++++++ metagpt/actions/write_plan.py | 2 +- metagpt/config.py | 3 + metagpt/prompts/ml_engineer.py | 11 +++ metagpt/roles/kaggle_manager.py | 65 ++++++++++------ metagpt/roles/ml_engineer.py | 129 ++++++++++++++++---------------- metagpt/schema.py | 42 +++++++++++ tests/metagpt/test_schema.py | 39 ++++++++++ 10 files changed, 330 insertions(+), 88 deletions(-) create mode 100644 metagpt/actions/ml_da_action.py diff --git a/config/config.yaml b/config/config.yaml index bed67083c..52a8eb036 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -94,4 +94,7 @@ MODEL_FOR_RESEARCHER_REPORT: gpt-3.5-turbo-16k ### browser path for pyppeteer engine, support Chrome, Chromium,MS Edge #PYPPETEER_EXECUTABLE_PATH: "/usr/bin/google-chrome-stable" -PROMPT_FORMAT: json #json or markdown \ No newline at end of file +PROMPT_FORMAT: json #json or markdown + +KAGGLE_USERNAME: "" +KAGGLE_KEY: "" \ No newline at end of file diff --git a/kaggle_team.py b/kaggle_team.py index 0743d445b..659c4a495 100644 --- a/kaggle_team.py +++ b/kaggle_team.py @@ -12,13 +12,14 @@ async def main( # competition: str, # data_desc: str, # requirement: str, - investment: float = 3.0, + investment: float = 5.0, n_round: int = 5, ): competition, data_desc, requirement = ( "titanic", "Training set is train.csv.\nTest set is test.csv. We also include gender_submission.csv, a set of predictions that assume all and only female passengers survive, as an example of what a submission file should look like.", "Run EDA on the train dataset, train a model to predict survival (20% as validation) and save it, predict the test set using saved model, save the test result according to format", + # "generate a random prediction of the same shape as gender_submission.csv and save", ) team = Team() diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py new file mode 100644 index 000000000..9f903fd22 --- /dev/null +++ b/metagpt/actions/ml_da_action.py @@ -0,0 +1,119 @@ +import json +from typing import Dict, List, Union + +from metagpt.actions import Action +from metagpt.schema import Message, Plan +from metagpt.logs import logger + + +def truncate(result: str, keep_len: int = 1000) -> str: + desc = "Truncated to show only the last 1000 characters\n" + if result.startswith(desc): + result = result[-len(desc) :] + + if len(result) > keep_len: + result = result[-keep_len:] + + if not result.startswith(desc): + return desc + result + return desc + + +class ReviewConst: + TASK_REVIEW_TRIGGER = "task" + CODE_REVIEW_TRIGGER = "code" + CONTINUE_WORD = ["confirm", "continue", "c", "yes", "y"] + CHANGE_WORD = ["change"] + EXIT_WORD = ["exit"] + TASK_REVIEW_INSTRUCTION = ( + f"If you want to change, add, delete a task or merge tasks in the plan, say '{CHANGE_WORD[0]} task task_id or current task, ... (things to change)' " + f"If you confirm the output from the current task and wish to continue, type: {CONTINUE_WORD[0]}" + ) + CODE_REVIEW_INSTRUCTION = ( + f"If you want the codes to be rewritten, say '{CHANGE_WORD[0]} ... (your change advice)' " + f"If you want to leave it as is, type: {CONTINUE_WORD[0]} or {CONTINUE_WORD[1]}" + ) + EXIT_INSTRUCTION = f"If you want to terminate the process, type: {EXIT_WORD[0]}" + + +class AskReview(Action): + async def run( + self, context: List[Message], plan: Plan = None, trigger: str = "task" + ): + logger.info("Current overall plan:") + logger.info( + "\n".join( + [ + f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" + for task in plan.tasks + ] + ) + ) + + logger.info("most recent context:") + latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" + review_instruction = ( + ReviewConst.TASK_REVIEW_INSTRUCTION + if trigger == ReviewConst.TASK_REVIEW_TRIGGER + else ReviewConst.CODE_REVIEW_INSTRUCTION + ) + prompt = ( + f"This is a <{trigger}> review. Please review output from {latest_action}\n" + f"{review_instruction}\n" + f"{ReviewConst.EXIT_INSTRUCTION}\n" + "Please type your review below:\n" + ) + + rsp = input(prompt) + + if rsp.lower() in ReviewConst.EXIT_WORD: + exit() + + confirmed = rsp.lower() in ReviewConst.CONTINUE_WORD + + return rsp, confirmed + + +class SummarizeAnalysis(Action): + PROMPT_TEMPLATE = """ + # Context + {context} + # Summary + Output a 30-word summary on analysis tool and modeling algorithms you have used, and the corresponding result. Make sure to announce the complete path to your test prediction file. Your summary: + """ + + def __init__(self, name: str = "", context=None, llm=None) -> str: + super().__init__(name, context, llm) + + async def run(self, conmpleted_plan: Plan) -> str: + tasks = json.dumps( + [task.dict() for task in conmpleted_plan.tasks], + indent=4, + ensure_ascii=False, + ) # all tasks finished, return all task outputs + prompt = self.PROMPT_TEMPLATE.format(context=tasks) + summary = await self._aask(prompt) + return summary + + +class Reflect(Action): + PROMPT_TEMPLATE = """ + # User Requirement + {user_requirement} + # Context + {context} + # Summary + Above is all your attempts to tackle the user requirement. You plan, act, submit your output, and get the result and feedback. + First, summarize each of your previous trial in a triple of (your methods, the corresponding result, potential improvement), list them out. + # Takeaways + Second, carefully find key takeaways from your summarization in a step-by-step thinking process + # Guidance + Finally, make a concise one-sentence guidance for improving your future plan. + Your response: + """ + + async def run(self, context: str) -> str: + user_requirement = "Score as high as possible in a data modeling competition" + prompt = self.PROMPT_TEMPLATE.format(context=context, user_requirement=user_requirement) + rsp = await self._aask(prompt) + return rsp diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index dcfa25d55..5ff6d965c 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -17,7 +17,7 @@ class WritePlan(Action): __context__ # Task: Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to __max_tasks__ tasks. - If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. + If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan. Output a list of jsons following the format: ```json [ diff --git a/metagpt/config.py b/metagpt/config.py index 3f9e742bd..5973adfc4 100644 --- a/metagpt/config.py +++ b/metagpt/config.py @@ -95,6 +95,9 @@ class Config(metaclass=Singleton): self.prompt_format = self._get("PROMPT_FORMAT", "markdown") + self.kaggle_username = self._get("KAGGLE_USERNAME", "") + self.kaggle_key = self._get("KAGGLE_KEY", "") + def _init_with_config_files_and_env(self, configs: dict, yaml_file): """Load from config/key.yaml, config/config.yaml, and env in decreasing order of priority""" configs.update(os.environ) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 55ac27d82..e78ea4166 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -168,3 +168,14 @@ ML_MODULE_MAP = { "classification_model": "metagpt.tools.functions.libs.machine_learning.ml_model", "regression_model": "metagpt.tools.functions.libs.machine_learning.ml_model", } + +STRUCTURAL_CONTEXT = """ +## User Requirement +{user_requirement} +## Data Description +{data_desc} +## Current Plan +{tasks} +## Current Task +{current_task} +""" diff --git a/metagpt/roles/kaggle_manager.py b/metagpt/roles/kaggle_manager.py index e902d99a0..d20769b92 100644 --- a/metagpt/roles/kaggle_manager.py +++ b/metagpt/roles/kaggle_manager.py @@ -5,16 +5,18 @@ import subprocess import fire import pandas as pd +from metagpt.config import CONFIG from metagpt.const import WORKSPACE_ROOT from metagpt.roles import Role from metagpt.actions import Action, BossRequirement -from metagpt.actions.write_analysis_code import AskReview, SummarizeAnalysis +from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis from metagpt.schema import Message, Task, Plan from metagpt.logs import logger +from metagpt.utils.common import CodeParser import os -os.environ["KAGGLE_USERNAME"] = "xxx" -os.environ["KAGGLE_KEY"] = "xxx" +os.environ["KAGGLE_USERNAME"] = CONFIG.kaggle_username +os.environ["KAGGLE_KEY"] = CONFIG.kaggle_key def run_command(cmd): print(cmd) @@ -38,6 +40,7 @@ class DownloadData(Action): # if not os.path.exists(data_path): if True: + # run_command(f"rm -r {data_path / '*'}") run_command(f"unzip -o {WORKSPACE_ROOT / '*.zip'} -d {data_path}") # FIXME: not safe file_list = run_command(f"ls {data_path}") @@ -52,24 +55,30 @@ class DownloadData(Action): class SubmitResult(Action): PROMPT_TEMPLATE = """ - # Context - {context} + # Summary + __summary__ # Your task - Extract the prediction file for test set, return only the path string, e.g., xxx.csv, xxx.xlsx + Extract the file path for test set prediction from the summary above, output a json following the format: + ```json + {"file_path": str = "the file path, for example, /path/to/the/prediction/file/xxx.csv, /path/to/the/prediction/file/xxx.xlsx"} + ``` """ def __init__(self, name: str = "", context=None, llm=None) -> str: super().__init__(name, context, llm) async def _parse_submit_file_path(self, context) -> str: - prompt = self.PROMPT_TEMPLATE.format(context=context) + prompt = self.PROMPT_TEMPLATE.replace("__summary__", context) rsp = await self._aask(prompt) - return rsp + rsp = CodeParser.parse_code(block=None, text=rsp) + file_path = json.loads(rsp)["file_path"] + return file_path async def run(self, competition, submit_message="") -> str: - submit_file_path = self._parse_submit_file_path(submit_message) + submit_file_path = await self._parse_submit_file_path(submit_message) data_path = WORKSPACE_ROOT / competition + submit_message = submit_message.replace("'", "") run_command(f"kaggle competitions submit {competition} -f {submit_file_path} -m '{submit_message}'") run_command(f"kaggle competitions leaderboard --show --csv {competition} > {data_path / 'leaderboard.csv'}") @@ -77,20 +86,20 @@ class SubmitResult(Action): leaderboard = pd.read_csv(data_path / 'leaderboard.csv') submission = pd.read_csv(data_path / 'submission.csv') - submission_score = submission.loc[0, "publicScore"] - submission_rank = leaderboard.loc[leaderboard["score"] == submission_score].index[0] - submission_rank_pct = round(submission_rank / len(leaderboard), 4) * 100 + print(submission) # submission.to_json(orient="records") - # best_score = max(submission["publicScore"]) - # best_rank = leaderboard.loc[leaderboard["score"] == best_score].index[0] + submission_score = submission.loc[0, "publicScore"] + best_score = max(submission["publicScore"]) # might be min + rank = leaderboard.loc[leaderboard["score"] == best_score].index[0] + rank_pct = round(rank / len(leaderboard), 4) * 100 submission_summary = f""" - ## All History - {submission.to_json(orient="records")} - ## Current - Current submission score: {submission_score}, rank: {submission_rank} (top {submission_rank_pct}%); + # All histories: + {submission.head(5).to_string()} + # Current + Current submission score: {submission_score}, best score: {best_score}, best rank: {rank} (top {rank_pct}%) """ - print(submission_summary) + logger.info(submission_summary) return submission_summary @@ -110,8 +119,6 @@ class KaggleManager(Role): self._set_state(0) # DownloadData, get competition of interest from human, download datasets elif observed == SummarizeAnalysis: self._set_state(1) # SubmitResult, get prediction from MLEngineer and submit it to Kaggle - elif observed == SubmitResult: - self._set_state(2) # AskReview, ask human for improvement async def _act(self): todo = self._rc.todo @@ -127,3 +134,19 @@ class KaggleManager(Role): msg = Message(content=rsp, role="user", cause_by=type(todo)) return msg + +if __name__ == "__main__": + competition, data_desc, requirement = ( + "titanic", + "Training set is train.csv.\nTest set is test.csv. We also include gender_submission.csv, a set of predictions that assume all and only female passengers survive, as an example of what a submission file should look like.", + "Run EDA on the train dataset, train a model to predict survival (20% as validation) and save it, predict the test set using saved model, save the test result according to format", + ) + + summary = "I used Python with pandas for data preprocessing, sklearn's RandomForestClassifier for modeling, and achieved 82.12% accuracy on validation. Predictions saved at '/Users/gary/Desktop/data_agents_opt/workspace/titanic/gender_submission.csv'." + + async def main(requirement: str = requirement): + role = KaggleManager(competition=competition, data_desc=data_desc) + # await role.run(Message(content="", cause_by=BossRequirement)) + await role.run(Message(content=summary, cause_by=SummarizeAnalysis)) + + fire.Fire(main) \ No newline at end of file diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 1e4367372..4536395ba 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -7,55 +7,14 @@ import fire from metagpt.roles import Role from metagpt.actions import Action from metagpt.schema import Message, Task, Plan +from metagpt.memory import Memory from metagpt.logs import logger from metagpt.actions.write_plan import WritePlan from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis, Reflect, ReviewConst, truncate from metagpt.actions.execute_code import ExecutePyCode - -STRUCTURAL_CONTEXT = """ -## User Requirement -{user_requirement} -## Current Plan -{tasks} -## Current Task -{current_task} -""" - - -def truncate(result: str, keep_len: int = 1000) -> str: - desc = "Truncated to show only the last 1000 characters\n" - if result.startswith(desc): - result = result[-len(desc) :] - - if len(result) > keep_len: - result = result[-keep_len:] - - if not result.startswith(desc): - return desc + result - return desc - - -class AskReview(Action): - async def run(self, context: List[Message], plan: Plan = None): - logger.info("Current overall plan:") - logger.info( - "\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks]) - ) - - logger.info("most recent context:") - latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" - prompt = f"\nPlease review output from {latest_action}:\n" \ - "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" \ - "If you confirm the output and wish to continue with the current process, type CONFIRM\n" \ - "If you want to terminate the process, type exit:\n" - rsp = input(prompt) - - if rsp.lower() in ("exit"): - exit() - - confirmed = rsp.lower() in ("confirm", "yes", "y") - - return rsp, confirmed +from metagpt.roles.kaggle_manager import DownloadData, SubmitResult +from metagpt.prompts.ml_engineer import STRUCTURAL_CONTEXT class WriteTaskGuide(Action): @@ -69,13 +28,35 @@ class MLEngineer(Role): ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") + self._watch([DownloadData, SubmitResult]) + self.plan = Plan(goal=goal) self.use_tools = False self.use_task_guide = False self.execute_code = ExecutePyCode() self.auto_run = auto_run + # memory for working on each task, discarded each time a task is done + self.working_memory = Memory() + async def _plan_and_act(self): + + ### Actions in a multi-agent multi-turn setting ### + memories = self.get_memories() + if memories: + latest_event = memories[-1].cause_by + if latest_event == DownloadData: + self.plan.context = memories[-1].content + elif latest_event == SubmitResult: + # get feedback for improvement from human, add to working memory + await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) + # self reflect on previous plan outcomes and think about how to improve the plan, add to working memory + prev_plan_outcomes = memories[-1].content + reflection = await Reflect().run(context=prev_plan_outcomes) + self.working_memory.add(Message(content=reflection, role="assistant")) + + + ### Common Procedure in both single- and multi-agent setting ### # create initial plan and update until confirmation await self._update_plan() @@ -87,7 +68,7 @@ class MLEngineer(Role): code, result, success = await self._write_and_exec_code() # ask for acceptance, users can other refuse and change tasks in the plan - task_result_confirmed = await self._ask_review() + review, task_result_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) if success and task_result_confirmed: # tick off this task and record progress @@ -98,7 +79,16 @@ class MLEngineer(Role): else: # update plan according to user's feedback and to take on changed tasks - await self._update_plan() + await self._update_plan(review) + + completed_plan_memory = self.get_useful_memories() # completed plan as a outcome + self._rc.memory.add(completed_plan_memory[0]) # add to persistent memory + + summary = await SummarizeAnalysis().run(self.plan) + rsp = Message(content=summary, cause_by=SummarizeAnalysis) + self._rc.memory.add(rsp) + + return rsp async def _write_and_exec_code(self, max_retry: int = 3): task_guide = ( @@ -143,23 +133,28 @@ class MLEngineer(Role): if "!pip" in code: success = False - # if not success: - # await self._ask_review() counter += 1 + if not success and counter >= max_retry: + logger.info("coding failed!") + review, _ = await self._ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) + if ReviewConst.CHANGE_WORD in review: + counter = 0 # redo the task again with help of human suggestions + return code, result, success - async def _ask_review(self): - if not self.auto_run: + async def _ask_review(self, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): + auto_run = auto_run or self.auto_run + if not auto_run: context = self.get_useful_memories() - review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan) + review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan, trigger=trigger) if not confirmed: self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) - return confirmed - return True + return review, confirmed + return "", True - async def _update_plan(self, max_tasks: int = 3): + async def _update_plan(self, review: str = "", max_tasks: int = 3): plan_confirmed = False while not plan_confirmed: context = self.get_useful_memories() @@ -167,30 +162,36 @@ class MLEngineer(Role): self.working_memory.add( Message(content=rsp, role="assistant", cause_by=WritePlan) ) - plan_confirmed = await self._ask_review() + + # TODO: precheck plan before asking reviews + + _, plan_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) tasks = WritePlan.rsp_to_tasks(rsp) - self.plan.add_tasks(tasks) - self.working_memory.clear() + if len(tasks) == 1 and self.plan.has_task_id(tasks[0].task_id): + self.plan.replace_task(tasks[0]) + else: + self.plan.add_tasks(tasks) + self.working_memory.clear() def get_useful_memories(self) -> List[Message]: """find useful memories only to reduce context length and improve performance""" user_requirement = self.plan.goal + data_desc = self.plan.context tasks = json.dumps( [task.dict() for task in self.plan.tasks], indent=4, ensure_ascii=False ) current_task = self.plan.current_task.json() if self.plan.current_task else {} context = STRUCTURAL_CONTEXT.format( - user_requirement=user_requirement, tasks=tasks, current_task=current_task + user_requirement=user_requirement, data_desc=data_desc, tasks=tasks, current_task=current_task ) context_msg = [Message(content=context, role="user")] - return context_msg + self.working_memory.get() - - @property - def working_memory(self): - return self._rc.memory + return context_msg + self.get_working_memories() + + def get_working_memories(self) -> List[Message]: + return self.working_memory.get() if __name__ == "__main__": diff --git a/metagpt/schema.py b/metagpt/schema.py index 601bdcea2..9b86a2448 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -156,7 +156,49 @@ class Plan(BaseModel): # Update the task map for quick access to tasks by ID self.task_map = {task.task_id: task for task in self.tasks} + + def reset_task(self, task_id: str): + """ + Clear code and result of the task based on task_id, and set the task as unfinished. + Args: + task_id (str): The ID of the task to be reset. + + Returns: + None + """ + if task_id in self.task_map: + task = self.task_map[task_id] + task.code = "" + task.result = "" + task.is_finished = False + + def replace_task(self, new_task: Task): + """ + Replace an existing task with the new input task based on task_id, and reset all tasks depending on it. + + Args: + new_task (Task): The new task that will replace an existing one. + + Returns: + None + """ + if new_task.task_id in self.task_map: + # Replace the task in the task map and the task list + self.task_map[new_task.task_id] = new_task + for i, task in enumerate(self.tasks): + if task.task_id == new_task.task_id: + self.tasks[i] = new_task + break + + # Reset dependent tasks + for task in self.tasks: + if new_task.task_id in task.dependent_task_ids: + self.reset_task(task.task_id) + + def has_task_id(self, task_id: str) -> bool: + return task_id in self.task_map + @property def current_task(self) -> Task: """Find current task to execute diff --git a/tests/metagpt/test_schema.py b/tests/metagpt/test_schema.py index 8f65d3785..324a083ca 100644 --- a/tests/metagpt/test_schema.py +++ b/tests/metagpt/test_schema.py @@ -104,3 +104,42 @@ class TestPlan: finished_tasks = plan.get_finished_tasks() assert len(finished_tasks) == 1 assert finished_tasks[0].task_id == "1" + + def test_reset_task_existing(self): + plan = Plan(goal="") + task = Task(task_id="1", instruction="Do something", code="print('Hello')", result="Hello", finished=True) + plan.add_tasks([task]) + plan.reset_task("1") + reset_task = plan.task_map["1"] + assert reset_task.code == "" + assert reset_task.result == "" + assert not reset_task.is_finished + + def test_reset_task_non_existing(self): + plan = Plan(goal="") + task = Task(task_id="1", instruction="Do something", code="print('Hello')", result="Hello", finished=True) + plan.add_tasks([task]) + plan.reset_task("2") # Task with ID 2 does not exist + assert "1" in plan.task_map + assert "2" not in plan.task_map + + def test_replace_task_with_dependents(self): + plan = Plan(goal="") + tasks = [Task(task_id="1", instruction="First Task", finished=True), + Task(task_id="2", instruction="Second Task", dependent_task_ids=["1"], finished=True)] + plan.add_tasks(tasks) + new_task = Task(task_id="1", instruction="Updated First Task") + plan.replace_task(new_task) + assert plan.task_map["1"].instruction == "Updated First Task" + assert not plan.task_map["2"].is_finished # Dependent task should be reset + assert plan.task_map["2"].code == "" + assert plan.task_map["2"].result == "" + + def test_replace_task_non_existing(self): + plan = Plan(goal="") + task = Task(task_id="1", instruction="First Task") + plan.add_tasks([task]) + new_task = Task(task_id="2", instruction="New Task") + plan.replace_task(new_task) # Task with ID 2 does not exist in plan + assert "1" in plan.task_map + assert "2" not in plan.task_map From 8d7657f347d51feb3048d6774bdbe17308ecf2ee Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 4 Dec 2023 14:29:47 +0800 Subject: [PATCH 064/637] update reflect on previous plan --- config/config.yaml | 4 ++-- kaggle_team.py | 7 ++++--- metagpt/actions/ml_da_action.py | 37 ++++++++++++++++++++------------- metagpt/roles/kaggle_manager.py | 4 ++-- metagpt/roles/ml_engineer.py | 19 +++++++++++------ 5 files changed, 44 insertions(+), 27 deletions(-) diff --git a/config/config.yaml b/config/config.yaml index 52a8eb036..bf998def7 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -96,5 +96,5 @@ MODEL_FOR_RESEARCHER_REPORT: gpt-3.5-turbo-16k PROMPT_FORMAT: json #json or markdown -KAGGLE_USERNAME: "" -KAGGLE_KEY: "" \ No newline at end of file +# KAGGLE_USERNAME: "" +# KAGGLE_KEY: "" \ No newline at end of file diff --git a/kaggle_team.py b/kaggle_team.py index 659c4a495..e8ab3ec41 100644 --- a/kaggle_team.py +++ b/kaggle_team.py @@ -13,20 +13,21 @@ async def main( # data_desc: str, # requirement: str, investment: float = 5.0, - n_round: int = 5, + n_round: int = 10, + auto_run: bool = False, ): competition, data_desc, requirement = ( "titanic", "Training set is train.csv.\nTest set is test.csv. We also include gender_submission.csv, a set of predictions that assume all and only female passengers survive, as an example of what a submission file should look like.", "Run EDA on the train dataset, train a model to predict survival (20% as validation) and save it, predict the test set using saved model, save the test result according to format", - # "generate a random prediction of the same shape as gender_submission.csv and save", + # "generate a random prediction, replace the Survived column of gender_submission.csv, and save the prediction to a new submission file", ) team = Team() team.hire( [ KaggleManager(competition=competition, data_desc=data_desc), - MLEngineer(goal=requirement), + MLEngineer(goal=requirement, auto_run=auto_run), ] ) diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py index 9f903fd22..a4537dad9 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_da_action.py @@ -3,6 +3,7 @@ from typing import Dict, List, Union from metagpt.actions import Action from metagpt.schema import Message, Plan +from metagpt.utils.common import CodeParser from metagpt.logs import logger @@ -98,22 +99,30 @@ class SummarizeAnalysis(Action): class Reflect(Action): PROMPT_TEMPLATE = """ - # User Requirement - {user_requirement} # Context - {context} + __context__ + # Latest User Requirement + __user_requirement__ # Summary Above is all your attempts to tackle the user requirement. You plan, act, submit your output, and get the result and feedback. - First, summarize each of your previous trial in a triple of (your methods, the corresponding result, potential improvement), list them out. - # Takeaways - Second, carefully find key takeaways from your summarization in a step-by-step thinking process - # Guidance - Finally, make a concise one-sentence guidance for improving your future plan. - Your response: + Output a json following the format: + ```json + { + "summary": str = "summarize each of your previous trial in a triple of (your methods, the corresponding result, potential improvement), list them out", + "takeaways": str = "carefully find key takeaways from your summarization in a step-by-step thinking process", + "reflection": "in one sentence, state executable actions for improving your future plan", + } + ``` """ + REWRITE_PLAN_INSTRUCTION = """When taking this reflection for rewriting plan, modify the current plan in place, replace, add, or delete tasks in the plan, + only make necessary change to the current plan, keep reusable tasks unchanged, provide the complete new plan.""" - async def run(self, context: str) -> str: - user_requirement = "Score as high as possible in a data modeling competition" - prompt = self.PROMPT_TEMPLATE.format(context=context, user_requirement=user_requirement) - rsp = await self._aask(prompt) - return rsp + async def run(self, context: str, user_requirement: str = "") -> str: + user_requirement = user_requirement or "Score as high as possible in a data modeling competition" + # prompt = self.PROMPT_TEMPLATE.format(context=context, user_requirement=user_requirement) + prompt = self.PROMPT_TEMPLATE.replace("__context__", context).replace("__user_requirement__", user_requirement) + rsp_json = await self._aask(prompt) + rsp = CodeParser.parse_code(block=None, text=rsp_json) + reflection = json.loads(rsp)["reflection"] + reflection += self.REWRITE_PLAN_INSTRUCTION + return reflection diff --git a/metagpt/roles/kaggle_manager.py b/metagpt/roles/kaggle_manager.py index d20769b92..354289975 100644 --- a/metagpt/roles/kaggle_manager.py +++ b/metagpt/roles/kaggle_manager.py @@ -38,8 +38,8 @@ class DownloadData(Action): run_command(f"kaggle competitions download {competition} --path {WORKSPACE_ROOT}") - # if not os.path.exists(data_path): - if True: + if not os.path.exists(data_path): + # if True: # run_command(f"rm -r {data_path / '*'}") run_command(f"unzip -o {WORKSPACE_ROOT / '*.zip'} -d {data_path}") # FIXME: not safe diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 4536395ba..abd14c7fb 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -48,13 +48,11 @@ class MLEngineer(Role): if latest_event == DownloadData: self.plan.context = memories[-1].content elif latest_event == SubmitResult: + # self reflect on previous plan outcomes and think about how to improve the plan, add to working memory + await self._reflect() + # get feedback for improvement from human, add to working memory await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - # self reflect on previous plan outcomes and think about how to improve the plan, add to working memory - prev_plan_outcomes = memories[-1].content - reflection = await Reflect().run(context=prev_plan_outcomes) - self.working_memory.add(Message(content=reflection, role="assistant")) - ### Common Procedure in both single- and multi-agent setting ### # create initial plan and update until confirmation @@ -172,7 +170,16 @@ class MLEngineer(Role): self.plan.replace_task(tasks[0]) else: self.plan.add_tasks(tasks) - self.working_memory.clear() + self.working_memory.clear() + + async def _reflect(self): + context = self.get_memories() + context = "\n".join([str(msg) for msg in context]) + # print("*" * 10) + # print(context) + # print("*" * 10) + reflection = await Reflect().run(context=context) + self.working_memory.add(Message(content=reflection, role="assistant")) def get_useful_memories(self) -> List[Message]: """find useful memories only to reduce context length and improve performance""" From acb70fa30c19d068f39f3a295140436bcad7ee82 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 4 Dec 2023 17:14:52 +0800 Subject: [PATCH 065/637] update increment development --- metagpt/actions/refine_design_api.py | 31 +++----- metagpt/actions/refine_prd.py | 46 ++++------- metagpt/actions/refine_project_management.py | 58 ++++---------- metagpt/actions/write_code_guide.py | 45 ++++++----- metagpt/actions/write_code_refine.py | 22 +++--- metagpt/roles/architect.py | 6 +- metagpt/roles/engineer.py | 82 +++++++------------- metagpt/roles/product_manager.py | 4 +- metagpt/roles/project_manager.py | 5 +- metagpt/team.py | 30 +++++++ startup.py | 42 ++-------- 11 files changed, 153 insertions(+), 218 deletions(-) diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py index 909fa6db9..d6a948b43 100644 --- a/metagpt/actions/refine_design_api.py +++ b/metagpt/actions/refine_design_api.py @@ -20,7 +20,7 @@ templates = { # Context {context} -## Legacy Design +## Legacy {legacy} ## Format example @@ -30,9 +30,7 @@ Role: You are an architect; the goal is to perform incremental development and d Requirement: Fill in the following missing information based on the context, each section name is a key in json. Output exactly as shown in the example, including single and double quotes. Max Output: 8192 chars or 2048 tokens. Try to use them up. -## Difference Description: Provide as list, the foremost differences description for system design here based on the previous. - -## Incremental implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. +## Incremental implementation approach: Provide as Python list[str]. Analyze the difficult points of the requirements, select the appropriate open-source framework. Up to 5. ## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores @@ -46,8 +44,7 @@ Output exactly as shown in the example, including single and double quotes, and "FORMAT_EXAMPLE": """ [CONTENT] { - "Difference Description": ["The ..."], - "Incremental implementation approach": "We will ...", + "Incremental implementation approach": ["We will ...",], "Python package name": "new_name", "Data structures and interface definitions": ' classDiagram @@ -72,7 +69,7 @@ Output exactly as shown in the example, including single and double quotes, and # Context {context} -## Legacy Design +## Legacy {legacy} ## Format example @@ -83,9 +80,7 @@ Requirement: Fill in the following missing information based on the context, not Max Output: 8192 chars or 2048 tokens. Try to use them up. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Difference Description: Provide as list, the foremost differences description for system design here based on the previous. - -## Incremental implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. +## Incremental implementation approach: Provide as Python list[str]. Analyze the difficult points of the requirements, select the appropriate open-source framework. Up to 5. ## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores @@ -96,15 +91,13 @@ Attention: Use '##' to split sections, not '#', and '## ' SHOULD W """, "FORMAT_EXAMPLE": """ --- -## Difference Description -```python -[ - "The ...", -] -``` ## Incremental implementation approach -We will ... +```python +[ + "We will ...", +] +``` ## Python package name ```python @@ -135,8 +128,8 @@ sequenceDiagram OUTPUT_MAPPING = { # "Incremental Requirements": (str, ...), - "Difference Description": (Union[List[str], str], ...), - "Incremental implementation approach": (str, ...), + # "Difference Description": (Union[List[str], str], ...), + "Incremental implementation approach": (Union[List[str], str], ...), "Python package name": (str, ...), # "File list": (List[str], ...), "Data structures and interface definitions": (str, ...), diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py index 0f01b9904..1d8bab5f8 100644 --- a/metagpt/actions/refine_prd.py +++ b/metagpt/actions/refine_prd.py @@ -9,10 +9,9 @@ increment_template = { "json": { "PROMPT_TEMPLATE": """ # Context -## User's Incremental Requirements -{new_requirements} +{context} -## Legacy PRD +## Legacy {legacy} ## Search Information @@ -22,13 +21,9 @@ increment_template = { {format_example} ----- Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. -Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive designOnly output one json, nothing else. +Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design. Only output one json, nothing else. -## Incremental Requirements: Provide as str, the foremost incremental requirements for PRD here based on the previous. - -## Difference Description: Provide as str, the foremost differences description for PRD here based on the previous. - -## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple +## Incremental Development Analysis: Provide as Python list[str], up to 5. incremental development analysis and plans based on the context and the legacy. If the requirement itself is simple, the Incremental Development Analysis should also be simple. output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, and only output the json inside this tag, nothing else @@ -36,9 +31,7 @@ and only output the json inside this tag, nothing else "FORMAT_EXAMPLE": """ [CONTENT] { - "Incremental Requirements": "", - "Difference Description": [], - "Incremental Development Plan": [], + "Incremental Development Analysis": [], } [/CONTENT] """, @@ -46,11 +39,9 @@ and only output the json inside this tag, nothing else "markdown": { "PROMPT_TEMPLATE": """ # Context -You need to refine the requirements based on the Incremental Requirements and the existing requirements' output. -## User's Incremental Requirements -{new_requirements} +{context} -## Legacy PRD +## Legacy {legacy} ## Search Information @@ -63,32 +54,21 @@ Role: You are a professional Product Manager tasked with overseeing incremental Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format.Only output one json, nothing else. -## Incremental Requirements: Provide as str, the foremost incremental requirements for PRD here based on the previous. - -## Difference Description: Provide as str, the foremost differences description for PRD here based on the previous. - -## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple +## Incremental Development Analysis: Provide as Python list[str], up to 5. Incremental development analysis and plans based on the context and the legacy. If the requirement itself is simple, the Incremental Development Analysis should also be simple. """, "FORMAT_EXAMPLE": """ --- -## Incremental Requirements -The boss ... -## Difference Description -... - -## Incremental Development Plan +## Incremental Development Analysis [ - "...", + "We will ...", ] """, }, } INCREMENT_OUTPUT_MAPPING = { - "Incremental Requirements": (str, ...), - "Difference Description": (Union[List[str], str], ...), - "Incremental Development Plan": (List[str], ...), + "Incremental Development Analysis": (List[str], ...), } @@ -97,7 +77,7 @@ class RefinePRD(Refine): def __init__(self, name="RefinePRD", context=None, llm=None): super().__init__(name, context, llm) - async def run(self, new_requirements, legacy, format=CONFIG.prompt_format, *args, **kwargs): + async def run(self, context, legacy, format=CONFIG.prompt_format, *args, **kwargs): sas = SearchAndSummarize() rsp = "" info = f"### Search Results\n{sas.result}\n\n### Search Summary\n{rsp}" @@ -107,7 +87,7 @@ class RefinePRD(Refine): prompt_template, format_example = get_template(increment_template, format) prompt = prompt_template.format( - new_requirements=new_requirements, legacy=legacy, search_information=info, + context=context, legacy=legacy, search_information=info, format_example=format_example ) logger.debug(prompt) diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py index dfeeb2db0..2775b1cb6 100644 --- a/metagpt/actions/refine_project_management.py +++ b/metagpt/actions/refine_project_management.py @@ -21,30 +21,23 @@ templates = { ## Format example {format_example} ----- -Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a Task list, and analyze task dependencies to start with the prerequisite modules. +Role: You are a project manager; the goal is to perform incremental development based on the context and the legacy. Break down tasks according to PRD/technical design, provide a Task list, and analyze task dependencies to start with the prerequisite modules. Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. +Output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT]. The following is the attribute description of the JSON object. -## Difference Description: Provide as a python list, the foremost differences description for project management here based on the previous. - -## Incremental Required Python third-party packages: Provided as a python list, the requirements.txt format +## Required Python third-party packages: Provided as a python list, the requirements.txt format ## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. -## Logic Analysis: Only files need to modified, Provided as a Python list[list[str]. If the file has no changes, the file will not be output. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first based on the previous. +## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first. -## Task list: Only files need to modified, provided as Python list[str]. If the file has no changes, the file will not be output. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first - -output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, +Output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, and only output the json inside this tag, nothing else """, "FORMAT_EXAMPLE": ''' { - "Incremental Requirements": "...", - "Difference Description": [ - "...", - ] - "Incremental Required Python third-party packages": [ + "Required Python third-party packages": [ "flask==1.1.2", "bcrypt==3.2.0" ], @@ -53,9 +46,6 @@ and only output the json inside this tag, nothing else ... description: A JSON object ... """, - "Logic Analysis": [ - ["game.py","Contains..."] - ], "Task list": [ "game.py" ] @@ -77,29 +67,16 @@ Role: You are a project manager; the goal is to perform incremental development Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Difference Description: Provided as a python list, the foremost differences description for project management here based on the previous. - -## Incremental Required Python third-party packages: Provided as a python list, the requirements.txt format +## Required Python third-party packages: Provided as a python list, the requirements.txt format ## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. -## Logic Analysis: Only files need to modified, Provided as a Python list[list[str]. If the file has no changes, the file will not be output. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first based on the previous. - -## Task list: Only files need to modified, provided as Python list[str]. If the file has no changes, the file will not be output. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first +## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first. """, "FORMAT_EXAMPLE": ''' --- -## Incremental Requirements -... -## Difference Description -```python -[ - "The ...", -] -``` - -## Incremental Required Python third-party packages +## Required Python third-party packages ```python [ "flask==1.1.2", @@ -116,13 +93,6 @@ description: A JSON object ... """ ``` -## Logic Analysis -```python -[ - ["game.py", "Contains ..."], -] -``` - ## Task list ```python [ @@ -136,16 +106,16 @@ description: A JSON object ... OUTPUT_MAPPING = { # "Incremental Requirements": (str, ...), # ## Incremental Requirements: Provided as a str, the foremost incremental requirements for project management here based on the previous. - "Difference Description": (Union[List[str], str], ...), - "Incremental Required Python third-party packages": (Union[List[str], str], ...), + # "Difference Analysis": (Union[List[str], str], ...), + "Required Python third-party packages": (Union[List[str], str], ...), "Full API spec": (str, ...), - "Logic Analysis": (List[List[str]], ...), + # "Logic Analysis": (List[List[str]], ...), "Task list": (List[str], ...), } class RefineTasks(Action): - def __init__(self, name="CreateTasks", context=None, llm=None): + def __init__(self, name="RefineTasks", context=None, llm=None): super().__init__(name, context, llm) def _save(self, context, rsp): @@ -158,7 +128,7 @@ class RefineTasks(Action): # Write requirements.txt requirements_path = WORKSPACE_ROOT / ws_name / "requirements.txt" - requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Incremental Required Python third-party packages"))) + requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Required Python third-party packages"))) async def run(self, context, legacy, format=CONFIG.prompt_format): prompt_template, format_example = get_template(templates, format) diff --git a/metagpt/actions/write_code_guide.py b/metagpt/actions/write_code_guide.py index 391173bb3..3e51f0d2d 100644 --- a/metagpt/actions/write_code_guide.py +++ b/metagpt/actions/write_code_guide.py @@ -10,24 +10,21 @@ from tenacity import retry, stop_after_attempt, wait_fixed PROMPT_TEMPLATE = """ NOTICE -Role: You are a professional software engineer, and your main task is to conduct incremental development, proposing incremental development plans and code guideance based on context and legacy code. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". +Role: You are a professional software engineer, and your main task is to conduct incremental development, proposing incremental development plans and code guideance based on context and legacy code. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). Output format carefully referenced "Format example". ## Regulations Review: To make the software directly operable without further coding, follow the regulations below during incremental development: +0) Determine the scope of responsibilities of each file and what classes and methods need to be implemented. 1) Import all referenced classes. -2) Implement all methods. -3) Add necessary explanation to all methods. +2) Implement all methods. +3) Add necessary explanation to all methods. 4) Ensure there are no potential bugs. 5) Confirm that the entire project conforms to the tasks proposed by the user. 6) Review the code thoroughly, checking for errors and validating the logic to ensure seamless user interaction without compromising any specified requirements. -## Incremental Development Plan: Proposed the Minimum essential incremental development plan, based on the following context and legacy code by thinking step by step. -... +## Incremental Development Plan: Provided as a Python list containing `filename.py`. Proposed the detail and essential incremental development plan, based on the following context and legacy code by thinking and analyzing step by step. All incremental modules/functions need to be added to the corresponding code files. + +## Code Guidance: Propose the foremost guidelines that how to implement code of modification part for incremental development based on the above context, legacy code and incremental development plan. -## Code guidelines: Propose the foremost guidelines that how to implement code of modification part for incremental development based on the above context, legacy code and incremental development plan. -```python -... -''' ----- # Context {context} @@ -35,20 +32,31 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc ## Legacy Code You are tasked with conducting incremental development in the existing code based on the provided legacy code and above information. ``` -{code} +{legacy} ``` ----- ## Format example ----- -## Incremental Development Guide: -... +## Incremental Development Guide +[ + "`game.py` Contains `Game` and ...", +] -## Code Guidance: -# Implementation the ... + +## Code Guidance +### Implementation `xx` in `xxx.py` ..., else retain the original xxx.py code. ```python +## xxx.py ... -''' +``` +--- +### Implementation of the `Game` in `game.py` ..., else retain the original game.py code. +```python +## game.py +class Game: + ... +``` ----- """ @@ -57,11 +65,10 @@ class WriteCodeGuide(Action): def __init__(self, name="WriteCodeGuide", context: list[Message] = None, llm=None): super().__init__(name, context, llm) - async def run(self, context, code): - prompt = PROMPT_TEMPLATE.format(context=context, code=code) + async def run(self, context, legacy): + prompt = PROMPT_TEMPLATE.format(context=context, legacy=legacy) logger.info(f'Write Code Guide ..') code_guide = await self._aask(prompt) # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) - # self._save(context, filename, code) return code_guide \ No newline at end of file diff --git a/metagpt/actions/write_code_refine.py b/metagpt/actions/write_code_refine.py index 466c30679..5a70b2c68 100644 --- a/metagpt/actions/write_code_refine.py +++ b/metagpt/actions/write_code_refine.py @@ -10,17 +10,19 @@ from tenacity import retry, stop_after_attempt, wait_fixed PROMPT_TEMPLATE = """ NOTICE Role: You are a professional engineer; your primary goal is to write PEP8 compliant, elegant, modular, easy-to-read, and maintainable Python 3.9 code (or any other programming language of your choice). -Requirements: You should modify the corresponding code based on the guidance. Then, output the complete code, fixing all errors according to the context. Ensure that you adhere to the specified guidelines for incremental development and modification of legacy code. +Requirements: Rewrite the complete code based on the Legacy Code so that it can be executed and avoid any potential bugs. You should modify the corresponding code based on the guidance. Output the complete code, fixing all errors according to the context. Ensure that you adhere to the specified guidelines for incremental development and modification of legacy code. ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format should be carefully referenced using the "Format example". Only output the current modified code, nothing else. In the modified code, if unchanged, you should output it, the complete code. -## Code: Only Write {filename}, Write code using triple quotes, based on the following list and context. -1. Do your best to implement THIS ONLY ONE FILE. ONLY USE EXISTING API. IF NO API, IMPLEMENT IT. -2. Requirement: Implement one of the following code files based on the provided context. Return the code in the specified format. Your code will be part of the entire project, so ensure it is complete, reliable, and reusable. -3. Attention1: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. -4. Attention2: YOU MUST FOLLOW "Data structures and interface definitions". DONT CHANGE ANY DESIGN. +## Rewrite Complete Code: Only Write one file {filename}, Write code using triple quotes, based on the following list, context, guidelines and legacy code. +1. Important: Do your best to implement ONLY ONE FILE. ONLY USE EXISTING API. IF NO API, IMPLEMENT IT. +2. Implement one of the following code files based on the provided context. Return the code in the specified format. Your code will be part of the entire project, so ensure it is complete, reliable, and reusable. +3. Attention1: Implement the functions required by the current file scope of responsibility. For example, main only needs to focus on the basic functions of main.py in the legacy code and the incremental functions to be implemented. Reuse existing code as much as possible. You can import functions from other codes instead of reimplementing the function. If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. +4. Attention2: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions, taking into account the guidelines, context, and existing legacy code. Retain the basic function methods from the legacy code, and make sure to preserve the existing code and logic that needs to be retained throughout the incremental development process. Avoid omitting any essential components. 5. Think before writing: What should be implemented and provided in this document? 6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 7. Do not use public member functions that do not exist in your design. +8. The Modified Code is implemented according to the requirements, and there are no issues with the code logic. All functions in the Modified Code are fully implemented; none are omitted or incomplete. + ----- # Context {context} @@ -30,13 +32,13 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format should be carefull {guide} ----- -## Legacy Code: The Legacy Code that needs to be modified. +## Legacy Code: The Legacy Code that needs to be modified. '===' is the separator of each code file in the legacy code. Basic function methods need to be retained in {filename}. {legacy} ----- ## Format example ----- -## Modified/Added Code: {filename} +## Rewrite Complete Code: {filename} ```python # {filename} ... @@ -55,8 +57,8 @@ class WriteCodeRefine(Action): code = CodeParser.parse_code(block="", text=code_rsp) return code - async def run(self, context, code, filename, guide): - prompt = PROMPT_TEMPLATE.format(context=context, legacy=code, filename=filename, guide=guide) + async def run(self, context, legacy, filename, guide): + prompt = PROMPT_TEMPLATE.format(context=context, legacy=legacy, filename=filename, guide=guide) logger.info(f'Code refine {filename}..') code = await self.write_code(prompt) # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) diff --git a/metagpt/roles/architect.py b/metagpt/roles/architect.py index f87f8899d..00ab6de35 100644 --- a/metagpt/roles/architect.py +++ b/metagpt/roles/architect.py @@ -50,7 +50,11 @@ class Architect(Role): async def _act(self) -> Message: if self.increment: logger.info(f"{self._setting}: ready to RefineDesign") - legacy = self._rc.env.get_legacy()["legacy_design"] + legacy_dict = self._rc.env.get_legacy() + legacy_code = "" + for code_dict in legacy_dict.get("legacy_code"): + legacy_code += code_dict.get("code") + "\n===\n" + legacy = "Legacy Design:\n" + legacy_dict.get("legacy_design") + "\nLegacy Code:\n" + legacy_code response = await self._rc.todo.run(self._rc.history, legacy) else: diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 66ce12f83..de144d4ca 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -16,6 +16,7 @@ from metagpt.actions import WriteCode, WriteCodeReview, WriteDesign, WriteTasks, from metagpt.actions.refine_design_api import RefineDesign from metagpt.actions.refine_prd import RefinePRD from metagpt.actions.refine_project_management import RefineTasks +from metagpt.actions.rewrite_code import RewriteCode from metagpt.actions.write_code_refine import WriteCodeRefine from metagpt.actions.write_code_guide import WriteCodeGuide from metagpt.const import WORKSPACE_ROOT @@ -88,7 +89,7 @@ class Engineer(Role): self._init_actions([WriteCode, WriteCodeReview]) if self.increment: - self._init_actions([WriteCodeGuide, WriteCodeRefine, WriteCodeReview]) + self._init_actions([WriteCodeGuide, WriteCodeRefine, RewriteCode]) self._watch([RefineTasks]) else: self._watch([WriteTasks]) @@ -210,18 +211,21 @@ class Engineer(Role): code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later workspace = self.get_workspace() human_str = str(self._rc.memory.get_by_role("Human")[0]) - code = self._rc.env.get_legacy()["legacy_code"] + legacy_code = "" + code_list = self._rc.env.get_legacy().get("legacy_code") + for code_dict in code_list: + legacy_code += code_dict.get("code") + "\n===\n" - # Refine code + # Code guide context = [] - msg = self._rc.memory.get_by_actions([RefineDesign, RefineTasks]) - - for m in msg: - context.append(m.content) + # msg = self._rc.memory.get_by_actions([RefineDesign, RefineTasks]) + msg = self._rc.memory.get_by_actions([RefineTasks]) + for tasks_msg in msg: + context.append(tasks_msg.content) context_str = human_str + "\n".join(context) try: logger.info("Write Code Guide start!") - guide = await WriteCodeGuide().run(context=context_str, code=code) + guide = await WriteCodeGuide().run(context=context_str, legacy=legacy_code) msg = Message(content=guide, role=self.profile, cause_by=WriteCodeGuide) self._rc.memory.add(msg) except Exception as e: @@ -230,19 +234,18 @@ class Engineer(Role): # Write code or Code review for todo in self.todos: - msg = self._rc.memory.get_by_actions([RefineTasks]) - context_str = human_str + "\n".join([m.content for m in msg]) # WriteCodeRefine try: logger.info("Write Code Refine start!") - code = await WriteCodeRefine().run(context=context_str, code=code, filename=todo, guide=guide) + code = await WriteCodeRefine().run(context=context_str, legacy=legacy_code, filename=todo, guide=guide) except Exception as e: logger.error("Write Code Refine failed!", e) pass - # FIXME: Code review Action + # FIXME: Code review Action : 如果有 pass 在代码里面,需要重写 # if self.use_code_review: # try: - # rewrite_code = await WriteCodeReview().run(context=context_str, code=code, filename=todo) + # # 解析成guides context = guide + human_str + # rewrite_code = await RewriteCode().run(context=human_str, code=code, filename=todo, legacy=legacy_code) # code = rewrite_code # except Exception as e: # logger.error("code review failed!", e) @@ -251,48 +254,17 @@ class Engineer(Role): msg = Message(content=code, role=self.profile, cause_by=WriteCodeRefine) self._rc.memory.add(msg) - code_msg = todo + FILENAME_CODE_SEP + str(file_path) - code_msg_all.append(code_msg) - - logger.info(f"Done {self.get_workspace()} generating.") - msg = Message( - content=MSG_SEP.join(code_msg_all), role=self.profile, cause_by=type(self._rc.todo), send_to="QaEngineer" - ) - return msg - - async def _act_bug_fix(self, bug_msgs) -> Message: - code_msg_all = [] # gather all code info, will pass to qa_engineer for tests later - workspace = self.get_workspace() - flag = True - # legacy_codes = legacy.split('---') - for todo in self.todos: - context = [] - - for m in bug_msgs: - if m.sent_from != "Engineer": - context.append(m.content) - context.append(m.content) - context_str = "\n".join(context) - code = [m.content for m in bug_msgs if m.sent_from == "Engineer"] - code = "\n".join(code) - - # Refine code or Write code - # if self.increment and len(legacy_codes) > 0: - # code = legacy_codes.pop(0) - - # Code review - try: - rewrite_code = await WriteCodeGuide().run(context=context_str, code=code, filename=todo) - code = rewrite_code - except Exception as e: - logger.error("code review failed!", e) - pass - - # code = await WriteCode().run(context=context_str, filename=todo) - - file_path = self.write_file(workspace, todo, code) - msg = Message(content=code, role=self.profile, cause_by=WriteCode) - self._rc.memory.add(msg) + # 利用todo作为key,去更新code_list的code + legacy_code = "" + for code_dict in code_list: + if code_dict.get("filename") == todo: + code_dict["code"] = code + legacy_code += code_dict.get("code") + "\n===\n" + # 若code_list中没有todo,则新增一个code_dict + if todo not in [code_dict.get("filename") for code_dict in code_list]: + code_dict = {"filename": todo, "code": code} + code_list.append(code_dict) + print(len(code_list)) code_msg = todo + FILENAME_CODE_SEP + str(file_path) code_msg_all.append(code_msg) diff --git a/metagpt/roles/product_manager.py b/metagpt/roles/product_manager.py index f48720430..2dc306297 100644 --- a/metagpt/roles/product_manager.py +++ b/metagpt/roles/product_manager.py @@ -52,7 +52,9 @@ class ProductManager(Role): async def _act(self) -> Message: if self.increment: logger.info(f"{self._setting}: ready to RefinePRD") - legacy = self._rc.env.get_legacy()["legacy_prd"] + legacy_dict = self._rc.env.get_legacy() + # legacy = "Legacy PRD:\n" + legacy_dict.get("legacy_prd") + "\nLegacy Code:\n" + legacy_dict.get("legacy_code") + legacy = legacy_dict.get("legacy_prd") response = await self._rc.todo.run(self._rc.history, legacy) else: diff --git a/metagpt/roles/project_manager.py b/metagpt/roles/project_manager.py index 5d4820f4a..8cf3e579f 100644 --- a/metagpt/roles/project_manager.py +++ b/metagpt/roles/project_manager.py @@ -57,7 +57,10 @@ class ProjectManager(Role): human_str = "\n".join([msg.content for msg in self._rc.memory.get_by_role("Human")]) # legacy_project_management and legacy_code legacy_dict = self._rc.env.get_legacy() - legacy_str = "Legacy Project Management:\n" + legacy_dict["legacy_project_management"] + "\nLegacy Code:\n" + legacy_dict["legacy_code"] + legacy_code = "" + for code_dict in legacy_dict.get("legacy_code"): + legacy_code += code_dict.get("code") + "\n===\n" + legacy_str = "Legacy Project Management:\n" + legacy_dict["legacy_project_management"] + "\nLegacy Code:\n" + legacy_code response = await self._rc.todo.run(self._rc.history, legacy=legacy_str) else: diff --git a/metagpt/team.py b/metagpt/team.py index ea5de4a12..d71880db9 100644 --- a/metagpt/team.py +++ b/metagpt/team.py @@ -5,6 +5,8 @@ @Author : alexanderwu @File : software_company.py """ +import os + from pydantic import BaseModel, Field from metagpt.actions import BossRequirement @@ -60,3 +62,31 @@ class Team(BaseModel): await self.environment.run() return self.environment.history + def save_legacy(self, project_path): + prd_path = os.path.join(project_path, 'docs/prd.md') + design_path = os.path.join(project_path, 'docs/system_design.md') + api_spec_and_tasks_path = os.path.join(project_path, 'docs/api_spec_and_tasks.md') + code_path = os.path.join(project_path, os.path.basename(project_path)) + with open(prd_path, 'r', encoding='utf-8') as f: + legacy_prd = f.read() + with open(design_path, 'r', encoding='utf-8') as f: + legacy_design = f.read() + with open(api_spec_and_tasks_path, 'r', encoding='utf-8') as f: + legacy_project_management = f.read() + legacy_codes = [] + for root, dirs, files in os.walk(code_path): + for file in files: + legacy_code = {} + if file.endswith('.py'): + with open(os.path.join(root, file), 'r', encoding='utf-8') as f: + legacy_code['filename'] = file + legacy_code['code'] = f.read() + legacy_codes.append(legacy_code) + legacy_dict = { + 'legacy_prd': legacy_prd, + 'legacy_design': legacy_design, + 'legacy_project_management': legacy_project_management, + 'legacy_code': legacy_codes + } + self.environment.set_legacy(legacy_dict) + diff --git a/startup.py b/startup.py index 60298ed36..1ecef65b0 100644 --- a/startup.py +++ b/startup.py @@ -20,48 +20,20 @@ from metagpt.utils.special_tokens import MSG_SEP async def startup( idea: str, - difference_description: str = "", - project_path: str = "", investment: float = 3.0, n_round: int = 5, code_review: bool = False, run_tests: bool = False, implement: bool = True, increment: bool = False, + path: str = "", + difference: str = "", ): """Run a startup. Be a boss.""" company = Team() if increment: - # 读取文件 - prd_path = os.path.join(project_path, 'docs/prd.md') - design_path = os.path.join(project_path, 'docs/system_design.md') - api_spec_and_tasks_path = os.path.join(project_path, 'docs/api_spec_and_tasks.md') - code_path = os.path.join(project_path, os.path.basename(project_path)) - - with open(prd_path, 'r', encoding='utf-8') as f: - legacy_prd = f.read() - - with open(design_path, 'r', encoding='utf-8') as f: - legacy_design = f.read() - - with open(api_spec_and_tasks_path, 'r', encoding='utf-8') as f: - legacy_project_management = f.read() - - legacy_code = '' - for root, dirs, files in os.walk(code_path): - for file in files: - if file.endswith('.py'): - with open(os.path.join(root, file), 'r', encoding='utf-8') as f: - legacy_code += f.read() + '\n---\n' - - legacy_dict = { - 'legacy_prd': legacy_prd, - 'legacy_design': legacy_design, - 'legacy_project_management': legacy_project_management, - 'legacy_code': legacy_code - } - company.environment.set_legacy(legacy_dict) + company.save_legacy(path) company.hire( [ ProductManager(increment=increment), @@ -85,20 +57,20 @@ async def startup( company.hire([QaEngineer()]) company.invest(investment) - company.start_project(idea+"\n"+difference_description) + company.start_project(idea +"\n" + difference) await company.run(n_round=n_round) def main( idea: str, - difference_description: str = "", - project_path: str = "", investment: float = 3.0, n_round: int = 5, code_review: bool = True, run_tests: bool = False, implement: bool = True, increment: bool = False, + path: str = "", + difference: str = "", ): """ We are a software startup comprised of AI. By investing in us, @@ -111,7 +83,7 @@ def main( :return: """ asyncio.run( - startup(idea, difference_description, project_path, investment, n_round, code_review, run_tests, implement, increment)) + startup(idea, investment, n_round, code_review, run_tests, implement, increment, path, difference)) if __name__ == "__main__": From 4304dd28cae93e3a2c597bf139bcd2d7783b3dad Mon Sep 17 00:00:00 2001 From: wubinhao <15754305168@163.com> Date: Tue, 5 Dec 2023 17:57:56 +0800 Subject: [PATCH 066/637] update write task guide, add code plan --- metagpt/actions/write_task_guide.py | 82 +++++++++++++++++++++++++++++ metagpt/roles/ml_engineer.py | 21 ++++---- metagpt/schema.py | 1 + 3 files changed, 92 insertions(+), 12 deletions(-) create mode 100644 metagpt/actions/write_task_guide.py diff --git a/metagpt/actions/write_task_guide.py b/metagpt/actions/write_task_guide.py new file mode 100644 index 000000000..eff53feef --- /dev/null +++ b/metagpt/actions/write_task_guide.py @@ -0,0 +1,82 @@ + +import json +from typing import Dict, List, Union + +from metagpt.actions import Action +from metagpt.schema import Message, Task, Plan + + +TASK_GUIDE_PROMPT_TEMPLATE = """ +# Context +{context} + +## Format example +1. +2. +3. +... + +----- +Tasks are all code development tasks. +You are a professional engineer, the main goal is to plan out concise solution steps for Current Task before coding. +A planning process can reduce the difficulty and improve the quality of coding. +You may be given some code plans for the tasks ahead, but you don't have to follow the existing plan when planning the current task. +The output plan should following the subsequent principles: +1.The plan is a rough checklist of steps outlining the entire program's structure.Try to keep the number of steps fewer than 5. +2.The steps should be written concisely and at a high level, avoiding overly detailed implementation specifics. +3.The execution of the plan happens sequentially, but the plan can incorporate conditional (if) and looping(loop) keywords for more complex structures. +4.Output carefully referenced "Format example" in format. +""" + +STRUCTURAL_CONTEXT = """ +## User Requirement +{user_requirement} +## Current Plan +{tasks} +## Current Task +{current_task} +""" + + +class WriteTaskGuide(Action): + + async def run(self, plan: Plan) -> str: + """Run of a task guide writing action, used in ml engineer + + Args: + plan (plan): task plan + useful_memories (list): useful_memories + Returns: + str: The dataset_descriptions string. + """ + + context = self.get_context(plan) + task_guide_prompt = TASK_GUIDE_PROMPT_TEMPLATE.format( + context=context, + ) + task_guide = await self._aask(task_guide_prompt) + return task_guide + + def get_context(self, plan: Plan): + user_requirement = plan.goal + task_rename_map = { + 'task_id': 'task_id', + 'instruction': 'instruction', + 'is_finished': 'is_finished', + # 'task_guide': 'code_plan' + } + + def process_task(task): + task_dict = task.dict() + ptask = {task_rename_map[k]: task_dict[k] for k in task_dict if k in task_rename_map} + return ptask + tasks = json.dumps( + [process_task(task) for task in plan.tasks], indent=4, ensure_ascii=False + ) + current_task = json.dumps(process_task(plan.current_task)) if plan.current_task else {} + context = STRUCTURAL_CONTEXT.format( + user_requirement=user_requirement, tasks=tasks, current_task=current_task + ) + # print(context) + return context + diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 65583638e..d905b7bfd 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -12,6 +12,7 @@ from metagpt.logs import logger from metagpt.actions.write_plan import WritePlan from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.write_task_guide import WriteTaskGuide STRUCTURAL_CONTEXT = """ ## User Requirement @@ -66,11 +67,6 @@ class AskReview(Action): return rsp, confirmed -class WriteTaskGuide(Action): - async def run(self, task_instruction: str, data_desc: str = "") -> str: - return "" - - class MLEngineer(Role): def __init__( self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False @@ -79,7 +75,7 @@ class MLEngineer(Role): self._set_react_mode(react_mode="plan_and_act") self.plan = Plan(goal=goal) self.use_tools = False - self.use_task_guide = False + self.use_task_guide = True self.execute_code = ExecutePyCode() self.auto_run = auto_run @@ -92,7 +88,7 @@ class MLEngineer(Role): logger.info(f"ready to take on task {task}") # take on current task - code, result, success = await self._write_and_exec_code() + code, result, success, task_guide = await self._write_and_exec_code() # ask for acceptance, users can other refuse and change tasks in the plan task_result_confirmed = await self._ask_review() @@ -101,6 +97,7 @@ class MLEngineer(Role): # tick off this task and record progress task.code = code task.result = result + task.task_guide = task_guide self.plan.finish_current_task() self.working_memory.clear() @@ -110,7 +107,7 @@ class MLEngineer(Role): async def _write_and_exec_code(self, max_retry: int = 3): task_guide = ( - await WriteTaskGuide().run(self.plan.current_task.instruction) + await WriteTaskGuide().run(self.plan) if self.use_task_guide else "" ) @@ -156,7 +153,7 @@ class MLEngineer(Role): counter += 1 - return code, result, success + return code, result, success, task_guide async def _ask_review(self): if not self.auto_run: @@ -185,7 +182,7 @@ class MLEngineer(Role): def get_useful_memories(self) -> List[Message]: """find useful memories only to reduce context length and improve performance""" - + # TODO dataset description , code steps user_requirement = self.plan.goal tasks = json.dumps( [task.dict() for task in self.plan.tasks], indent=4, ensure_ascii=False @@ -204,9 +201,9 @@ class MLEngineer(Role): if __name__ == "__main__": - requirement = "Run data analysis on sklearn Iris dataset, include a plot" + # requirement = "Run data analysis on sklearn Iris dataset, include a plot" # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" - # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" + requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" diff --git a/metagpt/schema.py b/metagpt/schema.py index e39f54a0c..db6861280 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -81,6 +81,7 @@ class Task(BaseModel): code: str = "" result: str = "" is_finished: bool = False + task_guide: str = "" class Plan(BaseModel): From 7436150849344945de0d7783538f9e7d7f44fb41 Mon Sep 17 00:00:00 2001 From: wubinhao <15754305168@163.com> Date: Tue, 5 Dec 2023 18:02:02 +0800 Subject: [PATCH 067/637] add code plan --- metagpt/actions/write_task_guide.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_task_guide.py b/metagpt/actions/write_task_guide.py index eff53feef..75067d33c 100644 --- a/metagpt/actions/write_task_guide.py +++ b/metagpt/actions/write_task_guide.py @@ -63,7 +63,7 @@ class WriteTaskGuide(Action): 'task_id': 'task_id', 'instruction': 'instruction', 'is_finished': 'is_finished', - # 'task_guide': 'code_plan' + 'task_guide': 'code_plan' } def process_task(task): From b561b2f98252c9174f885f4c82fc1c9eb4ee83df Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 5 Dec 2023 18:58:16 +0800 Subject: [PATCH 068/637] fix: change keep length of result from 1000 to 2000. --- metagpt/roles/ml_engineer.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 65583638e..e2203c4fb 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -143,10 +143,12 @@ class MLEngineer(Role): result, success = await self.execute_code.run(code) # truncated the result - print(truncate(result)) + _keep_result_len = 2000 + truncate_result = truncate(remove_escape_and_color_codes(result), keep_len=_keep_result_len) + print(truncate_result) # print(result) self.working_memory.add( - Message(content=truncate(remove_escape_and_color_codes(result)), role="user", cause_by=ExecutePyCode) + Message(content=truncate_result, keep_len=_keep_result_len), role="user", cause_by=ExecutePyCode) ) if "!pip" in code: From 2e7abe7d0342c13f782c878662f065a7a1b829eb Mon Sep 17 00:00:00 2001 From: wubinhao <15754305168@163.com> Date: Wed, 6 Dec 2023 11:24:24 +0800 Subject: [PATCH 069/637] change task_guide to code_steps --- metagpt/actions/write_analysis_code.py | 12 ++++---- ...rite_task_guide.py => write_code_steps.py} | 21 +++++-------- metagpt/llm.py | 2 +- metagpt/roles/ml_engineer.py | 30 +++++++++---------- metagpt/schema.py | 2 +- 5 files changed, 31 insertions(+), 36 deletions(-) rename metagpt/actions/{write_task_guide.py => write_code_steps.py} (80%) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index db0df2f90..1127dc78b 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -85,7 +85,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): self, context: [List[Message]], plan: Plan = None, - task_guide: str = "", + code_steps: str = "", system_msg: str = None, **kwargs, ) -> str: @@ -155,7 +155,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): self, context: List[Message], plan: Plan = None, - task_guide: str = "", + code_steps: str = "", data_desc: str = "", ) -> str: task_type = plan.current_task.task_type @@ -165,12 +165,12 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): {k: tool[k] for k in ["name", "description"] if k in tool} for tool in available_tools ] - task_guide = "\n".join( - [f"Step {step.strip()}" for step in task_guide.split("\n")] + code_steps = "\n".join( + [f"Step {step.strip()}" for step in code_steps.split("\n")] ) recommend_tools = await self._tool_recommendation( - task, task_guide, available_tools + task, code_steps, available_tools ) recommend_tools, tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) logger.info(f"Recommended tools for every steps: {recommend_tools}") @@ -194,7 +194,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): completed_code=completed_code, data_desc=data_desc, special_prompt=special_prompt, - code_steps=task_guide, + code_steps=code_steps, module_name=module_name, output_desc=output_desc, available_tools=recommend_tools, diff --git a/metagpt/actions/write_task_guide.py b/metagpt/actions/write_code_steps.py similarity index 80% rename from metagpt/actions/write_task_guide.py rename to metagpt/actions/write_code_steps.py index 75067d33c..47ea0b1df 100644 --- a/metagpt/actions/write_task_guide.py +++ b/metagpt/actions/write_code_steps.py @@ -6,7 +6,7 @@ from metagpt.actions import Action from metagpt.schema import Message, Task, Plan -TASK_GUIDE_PROMPT_TEMPLATE = """ +CODE_STEPS_PROMPT_TEMPLATE = """ # Context {context} @@ -38,7 +38,7 @@ STRUCTURAL_CONTEXT = """ """ -class WriteTaskGuide(Action): +class WriteCodeSteps(Action): async def run(self, plan: Plan) -> str: """Run of a task guide writing action, used in ml engineer @@ -51,24 +51,19 @@ class WriteTaskGuide(Action): """ context = self.get_context(plan) - task_guide_prompt = TASK_GUIDE_PROMPT_TEMPLATE.format( + code_steps_prompt = CODE_STEPS_PROMPT_TEMPLATE.format( context=context, ) - task_guide = await self._aask(task_guide_prompt) - return task_guide + code_steps = await self._aask(code_steps_prompt) + return code_steps def get_context(self, plan: Plan): user_requirement = plan.goal - task_rename_map = { - 'task_id': 'task_id', - 'instruction': 'instruction', - 'is_finished': 'is_finished', - 'task_guide': 'code_plan' - } + select_task_keys = ['task_id', 'instruction', 'is_finished', 'code_steps'] def process_task(task): task_dict = task.dict() - ptask = {task_rename_map[k]: task_dict[k] for k in task_dict if k in task_rename_map} + ptask = {k: task_dict[k] for k in task_dict if k in select_task_keys} return ptask tasks = json.dumps( [process_task(task) for task in plan.tasks], indent=4, ensure_ascii=False @@ -77,6 +72,6 @@ class WriteTaskGuide(Action): context = STRUCTURAL_CONTEXT.format( user_requirement=user_requirement, tasks=tasks, current_task=current_task ) - # print(context) + print(context) return context diff --git a/metagpt/llm.py b/metagpt/llm.py index 4edcd7a83..c8ddf9a26 100644 --- a/metagpt/llm.py +++ b/metagpt/llm.py @@ -11,7 +11,7 @@ from metagpt.config import CONFIG from metagpt.provider.anthropic_api import Claude2 as Claude from metagpt.provider.openai_api import OpenAIGPTAPI from metagpt.provider.zhipuai_api import ZhiPuAIGPTAPI -from metagpt.provider.spark_api import SparkAPI +# from metagpt.provider.spark_api import SparkAPI from metagpt.provider.human_provider import HumanProvider diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index d905b7bfd..e957d66c4 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -12,7 +12,7 @@ from metagpt.logs import logger from metagpt.actions.write_plan import WritePlan from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_task_guide import WriteTaskGuide +from metagpt.actions.write_code_steps import WriteCodeSteps STRUCTURAL_CONTEXT = """ ## User Requirement @@ -75,7 +75,7 @@ class MLEngineer(Role): self._set_react_mode(react_mode="plan_and_act") self.plan = Plan(goal=goal) self.use_tools = False - self.use_task_guide = True + self.use_code_steps = True self.execute_code = ExecutePyCode() self.auto_run = auto_run @@ -88,7 +88,7 @@ class MLEngineer(Role): logger.info(f"ready to take on task {task}") # take on current task - code, result, success, task_guide = await self._write_and_exec_code() + code, result, success, code_steps = await self._write_and_exec_code() # ask for acceptance, users can other refuse and change tasks in the plan task_result_confirmed = await self._ask_review() @@ -97,7 +97,7 @@ class MLEngineer(Role): # tick off this task and record progress task.code = code task.result = result - task.task_guide = task_guide + task.code_steps = code_steps self.plan.finish_current_task() self.working_memory.clear() @@ -106,9 +106,9 @@ class MLEngineer(Role): await self._update_plan() async def _write_and_exec_code(self, max_retry: int = 3): - task_guide = ( - await WriteTaskGuide().run(self.plan) - if self.use_task_guide + code_steps = ( + await WriteCodeSteps().run(self.plan) + if self.use_code_steps else "" ) @@ -123,14 +123,14 @@ class MLEngineer(Role): # breakpoint() if not self.use_tools or self.plan.current_task.task_type == "other": - # code = "print('abc')" - code = await WriteCodeByGenerate().run( - context=context, plan=self.plan, task_guide=task_guide, temperature=0.0 - ) + code = "print('abc')" + # code = await WriteCodeByGenerate().run( + # context=context, plan=self.plan, code_steps=code_steps, temperature=0.0 + # ) cause_by = WriteCodeByGenerate else: code = await WriteCodeWithTools().run( - context=context, plan=self.plan, task_guide=task_guide, data_desc="" + context=context, plan=self.plan, code_steps=code_steps, data_desc="" ) cause_by = WriteCodeWithTools @@ -153,7 +153,7 @@ class MLEngineer(Role): counter += 1 - return code, result, success, task_guide + return code, result, success, code_steps async def _ask_review(self): if not self.auto_run: @@ -203,9 +203,9 @@ class MLEngineer(Role): if __name__ == "__main__": # requirement = "Run data analysis on sklearn Iris dataset, include a plot" # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" - requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" + # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" - # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" + requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" async def main(requirement: str = requirement, auto_run: bool = False): role = MLEngineer(goal=requirement, auto_run=auto_run) diff --git a/metagpt/schema.py b/metagpt/schema.py index db6861280..2e4260096 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -81,7 +81,7 @@ class Task(BaseModel): code: str = "" result: str = "" is_finished: bool = False - task_guide: str = "" + code_steps: str = "" class Plan(BaseModel): From 962632cd15e76ba142d89ef086467be97f6ba7f0 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 6 Dec 2023 14:16:48 +0800 Subject: [PATCH 070/637] add GenerateDataDesc action --- metagpt/roles/ml_engineer.py | 131 ++++++++++++++++++++++++++++++----- 1 file changed, 112 insertions(+), 19 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 65583638e..15edb2b06 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,25 +1,38 @@ -from typing import Dict, List, Union +import glob import json -import subprocess +from typing import List import fire +import pandas as pd import re -from metagpt.roles import Role from metagpt.actions import Action -from metagpt.schema import Message, Task, Plan -from metagpt.logs import logger -from metagpt.actions.write_plan import WritePlan -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.write_plan import WritePlan +from metagpt.actions.write_task_guide import WriteTaskGuide +from metagpt.logs import logger +from metagpt.prompts.ml_engineer import GEN_DATA_DESC_PROMPT +from metagpt.roles import Role +from metagpt.schema import Message, Plan +from metagpt.utils.common import CodeParser STRUCTURAL_CONTEXT = """ ## User Requirement {user_requirement} +## Dataset Description +{data_desc} ## Current Plan {tasks} ## Current Task {current_task} +## Packages Installed +scikit-learn +pandas +numpy +lightgbm +xgboost +catboost """ @@ -43,6 +56,50 @@ def remove_escape_and_color_codes(input_str): return result +def read_data(file: str) -> pd.DataFrame: + if file.endswith(".csv"): + df = pd.read_csv(file, sep=",") + sep_list = [";", "\t", ":", " ", "|"] + for sep in sep_list: + if df.shape[1] == 1: + df = pd.read_csv(file, sep=sep) + else: + break + else: + raise ValueError(f"Unsupported file type: {file}") + return df + + +def get_samples(df: pd.DataFrame) -> str: + data = [] + + if len(df) > 5: + df_ = df.sample(5, random_state=0) + else: + df_ = df + + for i in list(df_): + nan_freq = float("%.2g" % (df[i].isna().mean() * 100)) + n_unique = df[i].nunique() + s = df_[i].tolist() + + if str(df[i].dtype) == "float64": + s = [round(sample, 2) if not pd.isna(sample) else None for sample in s] + + data.append([df_[i].name, df[i].dtype, nan_freq, n_unique, s]) + samples = pd.DataFrame( + data, + columns=[ + "Column_name", + "Data_type", + "NaN_Frequency(%)", + "N_unique", + "Samples", + ], + ) + return samples.to_string(index=False) + + class AskReview(Action): async def run(self, context: List[Message], plan: Plan = None): logger.info("Current overall plan:") @@ -66,24 +123,47 @@ class AskReview(Action): return rsp, confirmed -class WriteTaskGuide(Action): - async def run(self, task_instruction: str, data_desc: str = "") -> str: - return "" +# class WriteTaskGuide(Action): +# async def run(self, task_instruction: str, data_desc: dict = None) -> str: +# return "" + + +class GenerateDataDesc(Action): + async def run(self, files: list) -> dict: + data_desc = {} + for file in files: + df = read_data(file) + file_name = file.split("/")[-1] + data_head = df.head().to_dict(orient="list") + data_head = json.dumps(data_head, indent=4, ensure_ascii=False) + prompt = GEN_DATA_DESC_PROMPT.replace("{data_head}", data_head) + rsp = await self._aask(prompt) + rsp = CodeParser.parse_code(block=None, text=rsp) + data_desc[file_name] = {} + data_desc[file_name]["path"] = file + data_desc[file_name]["description"] = rsp + data_desc[file_name]["column_info"] = get_samples(df) + return data_desc class MLEngineer(Role): def __init__( - self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False + self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False, data_path: str = None ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") self.plan = Plan(goal=goal) - self.use_tools = False - self.use_task_guide = False + self.use_tools = True + self.use_task_guide = True self.execute_code = ExecutePyCode() self.auto_run = auto_run + self.data_path = data_path + self.data_desc = {} async def _plan_and_act(self): + if self.data_path: + self.data_desc = await self._generate_data_desc() + # create initial plan and update until confirmation await self._update_plan() @@ -108,9 +188,14 @@ class MLEngineer(Role): # update plan according to user's feedback and to take on changed tasks await self._update_plan() + async def _generate_data_desc(self): + files = glob.glob(self.data_path + "/*.csv") + data_desc = await GenerateDataDesc().run(files=files) + return data_desc + async def _write_and_exec_code(self, max_retry: int = 3): task_guide = ( - await WriteTaskGuide().run(self.plan.current_task.instruction) + await WriteTaskGuide().run(self.plan) if self.use_task_guide else "" ) @@ -126,14 +211,16 @@ class MLEngineer(Role): # breakpoint() if not self.use_tools or self.plan.current_task.task_type == "other": + logger.info("Write code with pure generation") # code = "print('abc')" code = await WriteCodeByGenerate().run( context=context, plan=self.plan, task_guide=task_guide, temperature=0.0 ) cause_by = WriteCodeByGenerate else: + logger.info("Write code with tools") code = await WriteCodeWithTools().run( - context=context, plan=self.plan, task_guide=task_guide, data_desc="" + context=context, plan=self.plan, task_guide=task_guide ) cause_by = WriteCodeWithTools @@ -192,7 +279,10 @@ class MLEngineer(Role): ) current_task = self.plan.current_task.json() if self.plan.current_task else {} context = STRUCTURAL_CONTEXT.format( - user_requirement=user_requirement, tasks=tasks, current_task=current_task + user_requirement=user_requirement, + data_desc=self.data_desc, + tasks=tasks, + current_task=current_task ) context_msg = [Message(content=context, role="user")] @@ -204,14 +294,17 @@ class MLEngineer(Role): if __name__ == "__main__": - requirement = "Run data analysis on sklearn Iris dataset, include a plot" + # requirement = "Run data analysis on sklearn Iris dataset, include a plot.." # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" - async def main(requirement: str = requirement, auto_run: bool = False): - role = MLEngineer(goal=requirement, auto_run=auto_run) + requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." + data_path = "/data/lidanyang/tabular_data/titanic" + + async def main(requirement: str = requirement, auto_run: bool = True, data_path: str = data_path): + role = MLEngineer(goal=requirement, auto_run=auto_run, data_path=data_path) await role.run(requirement) fire.Fire(main) From 6edbed8fb6e9ea19c2fa37de8d7f74888b83b903 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 6 Dec 2023 14:17:29 +0800 Subject: [PATCH 071/637] refine schema --- metagpt/tools/functions/schemas/feature_engineering.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/functions/schemas/feature_engineering.py b/metagpt/tools/functions/schemas/feature_engineering.py index c14bb933e..df2eebff6 100644 --- a/metagpt/tools/functions/schemas/feature_engineering.py +++ b/metagpt/tools/functions/schemas/feature_engineering.py @@ -20,7 +20,7 @@ class PolynomialExpansion(ToolSchema): class OneHotEncoding(ToolSchema): - """Apply one-hot encoding to specified categorical columns in a DataFrame.""" + """Apply one-hot encoding to specified categorical columns, the original columns will be dropped.""" df: pd.DataFrame = tool_field(description="DataFrame to process.") cols: list = tool_field(description="Categorical columns to be one-hot encoded.") From 0b918eb224e07621525d2518dba8e417de6fab8a Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 6 Dec 2023 14:18:38 +0800 Subject: [PATCH 072/637] Standardize the process with or without task guide --- metagpt/actions/write_analysis_code.py | 147 ++++++++++------------- metagpt/prompts/ml_engineer.py | 159 +++++++++++-------------- metagpt/tools/functions/__init__.py | 1 + 3 files changed, 136 insertions(+), 171 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index db0df2f90..646b4f3f1 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -23,28 +23,8 @@ from metagpt.utils.common import create_func_config class BaseWriteAnalysisCode(Action): - async def run( - self, context: List[Message], plan: Plan = None, task_guide: str = "" - ) -> str: - """Run of a code writing action, used in data analysis or modeling - - Args: - context (List[Message]): Action output history, source action denoted by Message.cause_by - plan (Plan, optional): Overall plan. Defaults to None. - task_guide (str, optional): suggested step breakdown for the current task. Defaults to "". - - Returns: - str: The code string. - """ - - -class WriteCodeByGenerate(BaseWriteAnalysisCode): - """Write code fully by generation""" - DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt - # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" - - def __init__(self, name: str = "", context=None, llm=None) -> str: - super().__init__(name, context, llm) + DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt + REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG @@ -81,6 +61,27 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): } return messages + async def run( + self, context: List[Message], plan: Plan = None, task_guide: str = "" + ) -> str: + """Run of a code writing action, used in data analysis or modeling + + Args: + context (List[Message]): Action output history, source action denoted by Message.cause_by + plan (Plan, optional): Overall plan. Defaults to None. + task_guide (str, optional): suggested step breakdown for the current task. Defaults to "". + + Returns: + str: The code string. + """ + + +class WriteCodeByGenerate(BaseWriteAnalysisCode): + """Write code fully by generation""" + + def __init__(self, name: str = "", context=None, llm=None) -> str: + super().__init__(name, context, llm) + async def run( self, context: [List[Message]], @@ -89,7 +90,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): system_msg: str = None, **kwargs, ) -> str: - # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) + context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) prompt = self.process_msg(context, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) return code_content["code"] @@ -99,7 +100,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" @staticmethod - def _parse_recommend_tools(module: str, recommend_tools: list) -> Tuple[Dict, List[Dict]]: + def _parse_recommend_tools(module: str, recommend_tools: list) -> List[Dict]: """ Parses and validates a list of recommended tools, and retrieves their schema from registry. @@ -108,44 +109,40 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): recommend_tools (list): A list of lists of recommended tools for each step. Returns: - Tuple[Dict, List[Dict]]: - - valid_tools: A dict of lists of valid tools for each step. - - tool_catalog: A list of dicts of unique tool schemas. + List[Dict]: A list of dicts of valid tool schemas. """ - valid_tools = {} + valid_tools = [] available_tools = registry.get_all_by_module(module).keys() - for index, tools in enumerate(recommend_tools): - key = f"Step {index + 1}" - tools = [tool for tool in tools if tool in available_tools] - valid_tools[key] = tools + for tool in recommend_tools: + if tool in available_tools: + valid_tools.append(tool) - unique_tools = set() - for tools in valid_tools.values(): - unique_tools.update(tools) - tool_catalog = registry.get_schemas(module, unique_tools) - return valid_tools, tool_catalog + tool_catalog = registry.get_schemas(module, valid_tools) + return tool_catalog async def _tool_recommendation( - self, task: str, data_desc: str, code_steps: str, available_tools: list + self, + context: [List[Message]], + code_steps: str, + available_tools: list ) -> list: """ - Recommend tools for each step of the specified task + Recommend tools for the specified task. Args: - task (str): the task description - data_desc (str): the description of the dataset for the task + context (List[Message]): Action output history, source action denoted by Message.cause_by code_steps (str): the code steps to generate the full code for the task available_tools (list): the available tools for the task Returns: - list: recommended tools for each step of the specified task + list: recommended tools for the specified task """ - prompt = TOOL_RECOMMENDATION_PROMPT.format( - task=task, - data_desc=data_desc, + system_prompt = TOOL_RECOMMENDATION_PROMPT.format( code_steps=code_steps, available_tools=available_tools, ) + prompt = self.process_msg(context, system_prompt) + tool_config = create_func_config(SELECT_FUNCTION_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) recommend_tools = rsp["recommend_tools"] @@ -156,50 +153,36 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): context: List[Message], plan: Plan = None, task_guide: str = "", - data_desc: str = "", ) -> str: task_type = plan.current_task.task_type - task = plan.current_task.instruction available_tools = registry.get_all_schema_by_module(task_type) - available_tools = [ - {k: tool[k] for k in ["name", "description"] if k in tool} - for tool in available_tools - ] - task_guide = "\n".join( - [f"Step {step.strip()}" for step in task_guide.split("\n")] - ) - - recommend_tools = await self._tool_recommendation( - task, task_guide, available_tools - ) - recommend_tools, tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) - logger.info(f"Recommended tools for every steps: {recommend_tools}") - special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") - module_name = ML_MODULE_MAP[task_type] - output_desc = TOOL_OUTPUT_DESC.get(task_type, "") - all_tasks = "" - completed_code = "" - for i, task in enumerate(plan.tasks): - stats = "DONE" if task.is_finished else "TODO" - all_tasks += f"Subtask {task.task_id}: {task.instruction}({stats})\n" + if len(available_tools) > 0: + available_tools = [ + {k: tool[k] for k in ["name", "description"] if k in tool} + for tool in available_tools + ] - for task in plan.tasks: - if task.code: - completed_code += task.code + "\n" + recommend_tools = await self._tool_recommendation(context, task_guide, available_tools) + tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) + logger.info(f"Recommended tools: \n{recommend_tools}") - prompt = TOO_ORGANIZATION_PROMPT.format( - all_tasks=all_tasks, - completed_code=completed_code, - data_desc=data_desc, - special_prompt=special_prompt, - code_steps=task_guide, - module_name=module_name, - output_desc=output_desc, - available_tools=recommend_tools, - tool_catalog=tool_catalog, - ) + module_name = ML_MODULE_MAP[task_type] + output_desc = TOOL_OUTPUT_DESC.get(task_type, "") + prompt = TOO_ORGANIZATION_PROMPT.format( + special_prompt=special_prompt, + code_steps=task_guide, + module_name=module_name, + output_desc=output_desc, + function_catalog=tool_catalog, + ) + context.append(Message(content=prompt, role="user")) + else: + context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) + context.append(Message(content=special_prompt, role="user")) + + prompt = self.process_msg(context) tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) return rsp["code"] diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 0c4d036fc..d568bdd1f 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -4,25 +4,46 @@ # @Author : lidanyang # @File : ml_engineer # @Desc : -ASSIGN_TASK_TYPE_PROMPT = """ -## All Task Type: -- **data_preprocess**: Only involve cleaning and preparing data through techniques like imputation, scaling, and encoding, not containing reading data, feature engineering, model training, etc. -- **feature_engineering**: Involves enhancing data features through techniques like encoding, aggregation, time component analysis, and creating polynomial and interaction features, etc. -- **other**: Any tasks that do not fit into the previous categories, such as visualization, summarizing findings, build model, etc. +GEN_DATA_DESC_PROMPT = """ +Here is the head 5 rows of the dataset: +{data_head} +Please provide a brief one-sentence background of the dataset, and concise descriptions for each column. Keep descriptions short yet informative. + +Output the information in a JSON format, as shown in this example: +```json +{ + "data_desc": "Brief dataset background.", + "column_desc": { + "column_name1": "Description of the first column.", + "column_name2": "Description of the second column.", + ... + } +} +``` +""" + + +ASSIGN_TASK_TYPE_PROMPT = """ Please assign a task type to each task in the list below from the given categories: {task_list} + +## All Task Type: +- **feature_engineering**: Only for creating new columns for input data. +- **data_preprocess**: Only for changing value inplace. +- **model_train**: Only for training model. +- **other**: Any tasks that do not fit into the previous categories, such as visualization, summarizing findings, build model, etc. """ ASSIGN_TASK_TYPE = { "name": "assign_task_type", - "description": "assign task type to each task by order", + "description": "Assign task type to each task by order.", "parameters": { "type": "object", "properties": { "task_type": { "type": "array", - "description": "List of task type.", + "description": "List of task type. The length should as long as task list", "items": { "type": "string", }, @@ -34,43 +55,32 @@ ASSIGN_TASK_TYPE = { TOOL_RECOMMENDATION_PROMPT = """ -## Comprehensive Task Description: -{task} - -## Dataset Description: -Details about the dataset for the project: -{data_desc} - -This task is divided into several steps, and you need to select the most suitable tools for each step. A tool means a function that can be used to help you solve the task. - -## Detailed Code Steps for the Task: -{code_steps} +Your are a tool recommender, the main goal is to recommend suitable tools for current task before coding. A tool means a function that can be used to help you solve the task. ## List of Available Tools: {available_tools} +This is a task guide for the current task, including detailed code steps. You can refer to it when recommending tools. +{code_steps} + ## Tool Selection and Instructions: -- For each code step listed above, choose up to five tools that are most likely to be useful in solving the task. -- If you believe that no tools are suitable for a step, indicate with an empty list. +- For the task, choose up to five tools that are most likely to be useful in solving the task. +- If you believe that no tools are suitable, indicate with an empty list. - Only list the names of the tools, not the full schema of each tool. - The result should only contain tool names that are in the list of available tools. -- The result list should be in the same order as the code steps. """ SELECT_FUNCTION_TOOLS = { "name": "select_function_tools", - "description": "Given code steps to generate full code for a task, select suitable tools for each step by order.", + "description": "For current task, select suitable tools for it.", "parameters": { "type": "object", "properties": { "recommend_tools": { "type": "array", - "description": "List of tool names for each code step. Empty list if no tool is suitable.", + "description": "List of tool names. Empty list if no tool is suitable.", "items": { - "type": "array", - "items": { - "type": "string", - }, + "type": "string", }, }, }, @@ -81,13 +91,13 @@ SELECT_FUNCTION_TOOLS = { CODE_GENERATOR_WITH_TOOLS = { "name": "add_subtask_code", - "description": "Add new code of current subtask to the end of an active Jupyter notebook.", + "description": "Add new code cell of current task to the end of an active Jupyter notebook.", "parameters": { "type": "object", "properties": { "code": { "type": "string", - "description": "The code to be added.", + "description": "The code to be added to a new cell in jupyter.", }, }, "required": ["code"], @@ -95,84 +105,60 @@ CODE_GENERATOR_WITH_TOOLS = { } TOO_ORGANIZATION_PROMPT = """ -As a senior data scientist, your role involves developing code for a specific sub-task within a larger project. This project is divided into several sub-tasks, which may either be new challenges or extensions of previous work. +The previous conversation has provided all tasks step-by-step for the use goal and their statuses. +Now, begin writing code for the current task. This code should writen strictly on the basis of all previous completed tasks code, not a standalone code. And avoid writing duplicate code that has already been written in previous tasks, such as repeated import of packages, reading data, etc. +Specifically, {special_prompt} +You can utilize pre-defined tools in 'Available Tools' if the tools are sufficient. And you should combine the use of other public packages if necessary, like sklearn, numpy, pandas, etc.. -## Sub-tasks Overview -Here's a list of all the sub-tasks, indicating their current status (DONE or TODO). Your responsibility is the first TODO task on this list. -{all_tasks} - -## Historical Code (Previously Done Sub-tasks): -This code, already executed in the Jupyter notebook, is critical for understanding the background and foundation for your current task. -```python -{completed_code} -``` - -## Dataset Description: -Details about the dataset for the project: -{data_desc} - -## Current Task Notion: -{special_prompt} - -## Code Steps for Your Sub-task: -Follow these steps to complete your current TODO task. You may use external Python functions or write custom code as needed. Ensure your code is self-contained. +## Code Steps for Current Task: +Follow steps below when you writing code if it's convenient. {code_steps} -When you call a function, you should import the function from `{module_name}` first, e.g.: -```python -from metagpt.tools.functions.libs.feature_engineering import fill_missing_value -``` - -## Available Functions for Each Step: -Here's a list of all available functions for each step. You can find more details about each function in [## Function Catalog] -{available_tools} - -## Function Catalog: +## Available Tools: Each function is described in JSON format, including the function name and parameters. {output_desc} {function_catalog} -## Your Output Format: -Generate the complete code for every step, listing any used function tools at the beginning of the step: +When you call a function above, you should import the function from `{module_name}` first, e.g.: ```python -# Step 1 -# Tools used: [function names or 'none'] - +from metagpt.tools.functions.libs.data_preprocess import fill_missing_value +```end -# Step 2 +## Your Output Format: +Generate the complete code for this task: +```python # Tools used: [function names or 'none'] - - -# Continue with additional steps, following the same format... + ```end *** Important Rules *** -- Use only the tools designated for each code step. -- Your output should only include code for the current sub-task. Don't repeat historical code. -- Only mention functions in comments if used in the code. -- Ensure the output new code is executable in the current Jupyter notebook environment, with all historical code executed. +- If you use tool not in the list, you should implement it by yourself. +- Ensure the output new code is executable in the same Jupyter notebook environment with previous tasks code have been executed. +- When write code for current task, remember the code should be coherent with previous tasks code. +- Remember that don't process the columns have been processed in previous tasks and don't mock data yourself. +- Prioritize using tools for the same functionality. """ - DATA_PREPROCESS_PROMPT = """ -In data preprocessing, closely monitor each column's data type. Apply suitable methods for various types (numerical, categorical, datetime, textual, etc.) to ensure the pandas.DataFrame is correctly formatted. +The current task is about data preprocessing, closely monitor each column's data type. Apply suitable methods for various types (numerical, categorical, datetime, textual, etc.) to ensure the pandas.DataFrame is correctly formatted. Additionally, ensure that the columns being processed must be the ones that actually exist in the dataset. +Don't write processed data to files. """ FEATURE_ENGINEERING_PROMPT = """ -When performing feature engineering, please adhere to the following principles: -- For specific user requests (such as removing a feature, creating a new feature based on existing data), directly generate the corresponding code. -- In cases of unclear user requirements, write feature engineering code that you believe will most improve model performance. This may include feature transformation, combination, aggregation, etc., with a limit of five features at a time. +The current task is about feature engineering. when performing it, please adhere to the following principles: - Ensure that the feature you're working with is indeed present in the dataset and consider the data type (numerical, categorical, etc.) and application scenario (classification, regression tasks, etc.). -- Importantly, provide detailed comments explaining the purpose of each feature and how it might enhance model performance, especially when the features are generated based on semantic understanding without clear user directives. +- When generate new features, you should combine real world knowledge and decide what features are useful for the task. +- Generate as diverse features as possible to improve the model's performance. +- Before generating a new feature, ensure the used features are already processed and ready to use. """ MODEL_TRAIN_PROMPT = """ -When selecting and training a model, please follow these guidelines to ensure optimal performance: +The current task is about training a model, please ensure high performance: - Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as lightGBM, XGBoost, CatBoost, etc. -— If user specifies a model, use that model. Otherwise, use the model you believe will best solve the problem. +- Before training, first check not is_numeric_dtype columns and use label encoding to convert them to numeric columns. +- Use the data from previous task result directly, do not mock or reload data yourself. """ - DATA_PREPROCESS_OUTPUT_DESC = "Please note that all functions uniformly output a processed pandas.DataFrame, facilitating seamless integration into the broader workflow." FEATURE_ENGINEERING_OUTPUT_DESC = "Please note that all functions uniformly output updated pandas.DataFrame with feature engineering applied." @@ -185,20 +171,15 @@ REGRESSION_MODEL_OUTPUT_DESC = "" ML_SPECIFIC_PROMPT = { "data_preprocess": DATA_PREPROCESS_PROMPT, "feature_engineering": FEATURE_ENGINEERING_PROMPT, - "classification_model": MODEL_TRAIN_PROMPT, - "regression_model": MODEL_TRAIN_PROMPT, + "model_train": MODEL_TRAIN_PROMPT, } TOOL_OUTPUT_DESC = { "data_preprocess": DATA_PREPROCESS_OUTPUT_DESC, "feature_engineering": FEATURE_ENGINEERING_OUTPUT_DESC, - "classification_model": CLASSIFICATION_MODEL_OUTPUT_DESC, - "regression_model": REGRESSION_MODEL_OUTPUT_DESC, } ML_MODULE_MAP = { - "data_preprocess": "metagpt.tools.functions.libs.machine_learning.data_preprocess", - "feature_engineering": "metagpt.tools.functions.libs.machine_learning.feature_engineering", - "classification_model": "metagpt.tools.functions.libs.machine_learning.ml_model", - "regression_model": "metagpt.tools.functions.libs.machine_learning.ml_model", + "data_preprocess": "metagpt.tools.functions.libs.data_preprocess", + "feature_engineering": "metagpt.tools.functions.libs.feature_engineering", } diff --git a/metagpt/tools/functions/__init__.py b/metagpt/tools/functions/__init__.py index b81e85833..30ee10827 100644 --- a/metagpt/tools/functions/__init__.py +++ b/metagpt/tools/functions/__init__.py @@ -6,3 +6,4 @@ # @Desc : from metagpt.tools.functions.register.register import registry import metagpt.tools.functions.libs.feature_engineering +import metagpt.tools.functions.libs.data_preprocess From 58e8e4c87936d6bf721f91109d4595a864a23203 Mon Sep 17 00:00:00 2001 From: wubinhao <15754305168@163.com> Date: Wed, 6 Dec 2023 15:56:26 +0800 Subject: [PATCH 073/637] fix --- metagpt/actions/write_code_steps.py | 2 +- metagpt/llm.py | 2 +- metagpt/roles/ml_engineer.py | 8 ++++---- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py index 47ea0b1df..d3f6e5553 100644 --- a/metagpt/actions/write_code_steps.py +++ b/metagpt/actions/write_code_steps.py @@ -72,6 +72,6 @@ class WriteCodeSteps(Action): context = STRUCTURAL_CONTEXT.format( user_requirement=user_requirement, tasks=tasks, current_task=current_task ) - print(context) + # print(context) return context diff --git a/metagpt/llm.py b/metagpt/llm.py index c8ddf9a26..4edcd7a83 100644 --- a/metagpt/llm.py +++ b/metagpt/llm.py @@ -11,7 +11,7 @@ from metagpt.config import CONFIG from metagpt.provider.anthropic_api import Claude2 as Claude from metagpt.provider.openai_api import OpenAIGPTAPI from metagpt.provider.zhipuai_api import ZhiPuAIGPTAPI -# from metagpt.provider.spark_api import SparkAPI +from metagpt.provider.spark_api import SparkAPI from metagpt.provider.human_provider import HumanProvider diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index e957d66c4..ce0689497 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -123,10 +123,10 @@ class MLEngineer(Role): # breakpoint() if not self.use_tools or self.plan.current_task.task_type == "other": - code = "print('abc')" - # code = await WriteCodeByGenerate().run( - # context=context, plan=self.plan, code_steps=code_steps, temperature=0.0 - # ) + # code = "print('abc')" + code = await WriteCodeByGenerate().run( + context=context, plan=self.plan, code_steps=code_steps, temperature=0.0 + ) cause_by = WriteCodeByGenerate else: code = await WriteCodeWithTools().run( From 98b14bbcc38fd99d39731fe38342e6e2fac96961 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 6 Dec 2023 16:44:14 +0800 Subject: [PATCH 074/637] chore --- metagpt/roles/ml_engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index e2203c4fb..34bd81110 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -148,7 +148,7 @@ class MLEngineer(Role): print(truncate_result) # print(result) self.working_memory.add( - Message(content=truncate_result, keep_len=_keep_result_len), role="user", cause_by=ExecutePyCode) + Message(content=truncate_result, role="user", cause_by=ExecutePyCode) ) if "!pip" in code: From 029adbc6d6fcc10e1cd553e2412b2355de36f2e8 Mon Sep 17 00:00:00 2001 From: wubinhao <15754305168@163.com> Date: Wed, 6 Dec 2023 16:48:31 +0800 Subject: [PATCH 075/637] update functions --- .../tools/functions/libs/data_preprocess.py | 123 +++++++++++ metagpt/tools/functions/libs/ml_model.py | 196 ++++++++++++++++++ .../functions/schemas/data_preprocess.py | 62 ++++++ metagpt/tools/functions/schemas/ml_model.py | 55 +++++ 4 files changed, 436 insertions(+) create mode 100644 metagpt/tools/functions/libs/data_preprocess.py create mode 100644 metagpt/tools/functions/libs/ml_model.py create mode 100644 metagpt/tools/functions/schemas/data_preprocess.py create mode 100644 metagpt/tools/functions/schemas/ml_model.py diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py new file mode 100644 index 000000000..68c96bbc9 --- /dev/null +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -0,0 +1,123 @@ + +import pandas as pd +import numpy as np + +from sklearn.impute import SimpleImputer +from sklearn.preprocessing import LabelEncoder +from sklearn.preprocessing import KBinsDiscretizer +from sklearn.preprocessing import MinMaxScaler +from sklearn.preprocessing import StandardScaler +from sklearn.preprocessing import MaxAbsScaler +from sklearn.preprocessing import RobustScaler +from sklearn.preprocessing import OrdinalEncoder + +from metagpt.tools.functions import registry +from metagpt.tools.functions.schemas.data_preprocess import * + + +@registry.register("data_preprocess", FillMissingValue) +def fill_missing_value(df: pd.DataFrame, features: list, strategy: str = 'mean', fill_value=None,): + df[features] = SimpleImputer(strategy=strategy, fill_value=fill_value).fit_transform(df[features]) + return df + + +# @registry.register("data_preprocess", FillMissingValue) +# def label_encode(df: pd.DataFrame, features: list,): +# for col in features: +# df[col] = LabelEncoder().fit_transform(df[col]) +# return df + + +@registry.register("data_preprocess", SplitBins) +def split_bins(df: pd.DataFrame, features: list, strategy: str = 'quantile',): + df[features] = KBinsDiscretizer(strategy=strategy, encode='ordinal').fit_transform(df[features]) + return df + + +@registry.register("data_preprocess", MinMaxScale) +def min_max_scale(df: pd.DataFrame, features: list, ): + df[features] = MinMaxScaler().fit_transform(df[features]) + return df + + +@registry.register("data_preprocess", StandardScale) +def standard_scale(df: pd.DataFrame, features: list, ): + df[features] = StandardScaler().fit_transform(df[features]) + return df + + +@registry.register("data_preprocess", LogTransform) +def log_transform(df: pd.DataFrame, features: list, ): + for col in features: + if df[col].min() <= 0: + df[col] = df[col] - df[col].min() + 2 + df[col] = np.log(df[col]) + return df + + +@registry.register("data_preprocess", MaxAbsScale) +def max_abs_scale(df: pd.DataFrame, features: list, ): + df[features] = MaxAbsScaler().fit_transform(df[features]) + return df + + +@registry.register("data_preprocess", RobustScale) +def robust_scale(df: pd.DataFrame, features: list, ): + df[features] = RobustScaler().fit_transform(df[features]) + return df + + +@registry.register("data_preprocess", OrdinalEncode) +def ordinal_encode(df: pd.DataFrame, features: list,): + df[features] = OrdinalEncoder().fit_transform(df[features]) + return df + + +if __name__ == '__main__': + def run(): + V = { + 'a': [-1, 2, 3, 6, 5, 4], + 'b': [1.1, 2.2, 3.3, 6.6, 5.5, 4.4], + 'c': ['aa', 'bb', 'cc', 'dd', 'ee', 'ff'], + 'd': [1, None, 3, None, 5, 4], + 'e': [1.1, np.NAN, 3.3, None, 5.5, 4.4], + 'f': ['aa', np.NAN, 'cc', None, '', 'ff'], + + } + + df = pd.DataFrame(V) + print(df.dtypes) + + numeric_features = ['a', 'b', 'd', 'e'] + numeric_features_wo_miss = ['a', 'b', ] + categorial_features = ['c', 'f'] + + df_ = fill_missing_value(df.copy(), numeric_features) + print(df_) + df_ = fill_missing_value(df.copy(), categorial_features, strategy='constant', fill_value='hehe') + print(df_) + + df_ = fill_missing_value(df.copy(), numeric_features, strategy='constant', fill_value=999) + print(df_) + + # df_ = label_encode(df.copy(), numeric_features + categorial_features, ) + # print(df_) + + df_ = split_bins(df.copy(), numeric_features_wo_miss, strategy='quantile') + print(df_) + + df_ = min_max_scale(df.copy(), numeric_features, ) + print(df_) + + df_ = standard_scale(df.copy(), numeric_features, ) + print(df_) + + df_ = log_transform(df.copy(), numeric_features, ) + print(df_) + + df_ = max_abs_scale(df.copy(), numeric_features, ) + print(df_) + + df_ = robust_scale(df.copy(), numeric_features, ) + print(df_) + run() \ No newline at end of file diff --git a/metagpt/tools/functions/libs/ml_model.py b/metagpt/tools/functions/libs/ml_model.py new file mode 100644 index 000000000..b669de2c1 --- /dev/null +++ b/metagpt/tools/functions/libs/ml_model.py @@ -0,0 +1,196 @@ +from sklearn.model_selection import train_test_split +from sklearn.preprocessing import LabelEncoder + +from sklearn.linear_model import LogisticRegression +from sklearn.ensemble import RandomForestClassifier +from sklearn.ensemble import GradientBoostingClassifier + + +from sklearn.linear_model import LinearRegression +from sklearn.ensemble import RandomForestRegressor +from sklearn.ensemble import GradientBoostingRegressor + +from metagpt.tools.functions import registry +from metagpt.tools.functions.schemas.ml_model import * + + +######### +## 分类 ## +######### + + +@registry.register("classification_model", LogisticRegressionClassification) +def logistic_regression_classification(df, label, test_size=0.2, penalty='l2', dual=False): + nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] + for col in nonnumeric_columns: + df[col] = LabelEncoder().fit_transform(df[col]) + df = df.fillna(0) + + features = [col for col in df if col != label] + x, y = df[features], df[label] + tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) + + model = LogisticRegression(penalty=penalty, dual=dual) + model.fit(tr_x, tr_y, ) + te_pred_prob = model.predict_proba(te_x) + + res = { + 'te_pred_prob': te_pred_prob + } + return res + + +@registry.register("classification_model", RandomForestClassification) +def random_forest_classification(df, label, test_size=0.2, n_estimators=100, criterion='gini'): + nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] + for col in nonnumeric_columns: + df[col] = LabelEncoder().fit_transform(df[col]) + df = df.fillna(0) + + features = [col for col in df if col != label] + x, y = df[features], df[label] + tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) + model = RandomForestClassifier(n_estimators=n_estimators, criterion=criterion) + model.fit(tr_x, tr_y, ) + te_pred_prob = model.predict_proba(te_x) + + res = { + 'te_pred_prob': te_pred_prob + } + return res + + +@registry.register("classification_model", GradientBoostingClassification) +def gradient_boosting_classification(df, label, test_size=0.2, n_estimators=100, learning_rate=0.1): + nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] + for col in nonnumeric_columns: + df[col] = LabelEncoder().fit_transform(df[col]) + df = df.fillna(0) + + features = [col for col in df if col != label] + x, y = df[features], df[label] + tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) + model = GradientBoostingClassifier(n_estimators=n_estimators, learning_rate=learning_rate) + model.fit(tr_x, tr_y, ) + te_pred_prob = model.predict_proba(te_x) + + res = { + 'te_pred_prob': te_pred_prob + } + return res + + + +######### +## 回归 ## +######### + + +@registry.register("regression_model", LinearRegressionRegression) +def linear_regression(df, label, test_size=0.2, ): + nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] + for col in nonnumeric_columns: + df[col] = LabelEncoder().fit_transform(df[col]) + df = df.fillna(0) + + features = [col for col in df if col != label] + x, y = df[features], df[label] + tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) + + model = LinearRegression() + model.fit(tr_x, tr_y, ) + te_pred_prob = model.predict(te_x) + + res = { + 'te_pred_prob': te_pred_prob + } + return res + + +@registry.register("regression_model", RandomForestRegression) +def random_forest_regression(df, label, test_size=0.2, n_estimators=100, criterion='squared_error'): + nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] + for col in nonnumeric_columns: + df[col] = LabelEncoder().fit_transform(df[col]) + df = df.fillna(0) + + features = [col for col in df if col != label] + x, y = df[features], df[label] + tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) + model = RandomForestRegressor(n_estimators=n_estimators, criterion=criterion) + model.fit(tr_x, tr_y, ) + te_pred_prob = model.predict(te_x) + + res = { + 'te_pred_prob': te_pred_prob + } + return res + + +@registry.register("regression_model", GradientBoostingRegression) +def gradient_boosting_regression(df, label, test_size=0.2, n_estimators=100, learning_rate=0.1): + nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] + for col in nonnumeric_columns: + df[col] = LabelEncoder().fit_transform(df[col]) + df = df.fillna(0) + + features = [col for col in df if col != label] + x, y = df[features], df[label] + tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) + model = GradientBoostingRegressor(n_estimators=n_estimators, learning_rate=learning_rate) + model.fit(tr_x, tr_y, ) + te_pred_prob = model.predict(te_x) + + res = { + 'te_pred_prob': te_pred_prob + } + return res + + +if __name__ == '__main__': + def run(): + from sklearn.datasets import load_iris + loader = load_iris(as_frame=True) + df = loader['data'] + df['target'] = loader['target'] + + df[df.columns[0]] = df[df.columns[0]].astype(str) + df[df.columns[1]] = df[df.columns[1]].astype(int) + df['target'] = df['target'].astype(str) + + print(df) + print('####'*5) + res = logistic_regression_classification(df, 'target', test_size=0.25, penalty='l2', dual=False) + print(res['te_pred_prob']) + + print('####'*5) + res = random_forest_classification(df, 'target', test_size=0.25, n_estimators=100, criterion='gini') + print(res['te_pred_prob']) + + print('####'*5) + res = gradient_boosting_classification(df, 'target', test_size=0.25, n_estimators=100, learning_rate=0.1) + print(res['te_pred_prob']) + + from sklearn.datasets import make_regression + import pandas as pd + loader = make_regression() + df = pd.DataFrame(loader[0]) + df['target'] = loader[1] + + df[df.columns[0]] = df[df.columns[0]].astype(str) + df[df.columns[1]] = df[df.columns[1]].astype(int) + # df['target'] = df['target'].astype(str) + + print(df) + print('####' * 5) + res = linear_regression(df, 'target', test_size=0.25, ) + print(res['te_pred_prob']) + + print('####' * 5) + res = random_forest_regression(df, 'target', test_size=0.25, n_estimators=100, criterion='squared_error') + print(res['te_pred_prob']) + + print('####' * 5) + res = gradient_boosting_regression(df, 'target', test_size=0.25, n_estimators=100, learning_rate=0.1) + print(res['te_pred_prob']) + run() \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/data_preprocess.py b/metagpt/tools/functions/schemas/data_preprocess.py new file mode 100644 index 000000000..40e1d64e0 --- /dev/null +++ b/metagpt/tools/functions/schemas/data_preprocess.py @@ -0,0 +1,62 @@ + +import pandas as pd + +from metagpt.tools.functions.schemas.base import tool_field, ToolSchema + + +class FillMissingValue(ToolSchema): + """Completing missing values with simple strategies""" + df: pd.DataFrame = tool_field(description="input dataframe") + features: list = tool_field(description="columns to be processed") + strategy: str = tool_field(description="the imputation strategy", default='mean') + fill_value: int = tool_field(description="fill_value is used to replace all occurrences of missing_values", default=None) + + +# class LabelEncode(ToolSchema): +# """Completing missing values with simple strategies""" +# df: pd.DataFrame = tool_field(description="input dataframe") +# features: list = tool_field(description="columns to be processed") + + +class SplitBins(ToolSchema): + """Bin continuous data into intervals and return the bin identifier encoded as an integer value""" + df: pd.DataFrame = tool_field(description="input dataframe") + features: list = tool_field(description="columns to be processed") + strategy: str = tool_field(description="Strategy used to define the widths of the bins", default='quantile') + + +class MinMaxScale(ToolSchema): + """Transform features by scaling each feature to a range, witch is (0, 1)""" + df: pd.DataFrame = tool_field(description="input dataframe") + features: list = tool_field(description="columns to be processed") + + +class StandardScale(ToolSchema): + """Standardize features by removing the mean and scaling to unit variance""" + df: pd.DataFrame = tool_field(description="input dataframe") + features: list = tool_field(description="columns to be processed") + + +class LogTransform(ToolSchema): + """Performs a logarithmic transformation on the specified columns""" + df: pd.DataFrame = tool_field(description="input dataframe") + features: list = tool_field(description="columns to be processed") + + +class MaxAbsScale(ToolSchema): + """Scale each feature by its maximum absolute value""" + df: pd.DataFrame = tool_field(description="input dataframe") + features: list = tool_field(description="columns to be processed") + + +class RobustScale(ToolSchema): + """Scale features using statistics that are robust to outliers, the quantile_range is (25.0, 75.0)""" + df: pd.DataFrame = tool_field(description="input dataframe") + features: list = tool_field(description="columns to be processed") + + +class OrdinalEncode(ToolSchema): + """Encode categorical features as an integer array""" + df: pd.DataFrame = tool_field(description="input dataframe") + features: list = tool_field(description="columns to be processed") + diff --git a/metagpt/tools/functions/schemas/ml_model.py b/metagpt/tools/functions/schemas/ml_model.py new file mode 100644 index 000000000..9268156af --- /dev/null +++ b/metagpt/tools/functions/schemas/ml_model.py @@ -0,0 +1,55 @@ +import pandas as pd + +from metagpt.tools.functions.schemas.base import tool_field, ToolSchema + + +class LogisticRegressionClassification(ToolSchema): + """Logistic Regression (aka logit, MaxEnt) classifier""" + df: pd.DataFrame = tool_field(description="input dataframe") + label: str = tool_field(description="target name") + test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) + penalty: str = tool_field(description="Specify the norm of the penalty", default="l2") + dual: bool = tool_field(description="Dual (constrained) or primal (regularized) formulation", default="l2") + + +class RandomForestClassification(ToolSchema): + """random forest is a meta estimator that fits a number of decision tree classifiers on various sub-samples of the dataset and uses averaging to improve the predictive accuracy and control over-fitting""" + df: pd.DataFrame = tool_field(description="input dataframe") + label: str = tool_field(description="target name") + test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) + n_estimators: int = tool_field(description="The number of trees in the forest", default=100) + criterion: str = tool_field(description="The function to measure the quality of a split", default="gini") + + +class GradientBoostingClassification(ToolSchema): + """Gradient Boosting for classification.This algorithm builds an additive model in a forward stage-wise fashion""" + df: pd.DataFrame = tool_field(description="input dataframe") + label: str = tool_field(description="target name") + test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) + n_estimators: int = tool_field(description="The number of boosting stages to perform", default=100) + learning_rate: float = tool_field(description="Learning rate shrinks the contribution of each tree by learning_rate", default=0.1) + + +class LinearRegressionRegression(ToolSchema): + """Ordinary least squares Linear Regression.""" + df: pd.DataFrame = tool_field(description="input dataframe") + label: str = tool_field(description="target name") + test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) + + +class RandomForestRegression(ToolSchema): + """random forest is a meta estimator that fits a number of decision tree on various sub-samples of the dataset and uses averaging to improve the predictive accuracy and control over-fitting""" + df: pd.DataFrame = tool_field(description="input dataframe") + label: str = tool_field(description="target name") + test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) + n_estimators: int = tool_field(description="The number of trees in the forest", default=100) + criterion: str = tool_field(description="The function to measure the quality of a split", default="squared_error") + + +class GradientBoostingRegression(ToolSchema): + """Gradient Boosting for regression.This estimator builds an additive model in a forward stage-wise fashion""" + df: pd.DataFrame = tool_field(description="input dataframe") + label: str = tool_field(description="target name") + test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) + n_estimators: int = tool_field(description="The number of boosting stages to perform", default=100) + learning_rate: float = tool_field(description="Learning rate shrinks the contribution of each tree by learning_rate", default=0.1) From 21d97a23bb65b92a0379ff101ecbd497bd6e8537 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 6 Dec 2023 17:31:51 +0800 Subject: [PATCH 076/637] output code_steps to json --- metagpt/actions/write_analysis_code.py | 1 - metagpt/actions/write_code_steps.py | 25 ++++++++++++++----------- metagpt/roles/ml_engineer.py | 2 +- 3 files changed, 15 insertions(+), 13 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index cfec95deb..71467edd0 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -153,7 +153,6 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): context: List[Message], plan: Plan = None, code_steps: str = "", - data_desc: str = "", ) -> str: task_type = plan.current_task.task_type available_tools = registry.get_all_schema_by_module(task_type) diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py index d3f6e5553..0bfb9c225 100644 --- a/metagpt/actions/write_code_steps.py +++ b/metagpt/actions/write_code_steps.py @@ -4,18 +4,12 @@ from typing import Dict, List, Union from metagpt.actions import Action from metagpt.schema import Message, Task, Plan - +from metagpt.utils.common import CodeParser CODE_STEPS_PROMPT_TEMPLATE = """ # Context {context} -## Format example -1. -2. -3. -... - ----- Tasks are all code development tasks. You are a professional engineer, the main goal is to plan out concise solution steps for Current Task before coding. @@ -25,7 +19,16 @@ The output plan should following the subsequent principles: 1.The plan is a rough checklist of steps outlining the entire program's structure.Try to keep the number of steps fewer than 5. 2.The steps should be written concisely and at a high level, avoiding overly detailed implementation specifics. 3.The execution of the plan happens sequentially, but the plan can incorporate conditional (if) and looping(loop) keywords for more complex structures. -4.Output carefully referenced "Format example" in format. + +Output the code steps in a JSON format, as shown in this example: +```json +{ + "Step 1": "", + "Step 2": "", + "Step 3": "", + ... +} +``` """ STRUCTURAL_CONTEXT = """ @@ -51,10 +54,11 @@ class WriteCodeSteps(Action): """ context = self.get_context(plan) - code_steps_prompt = CODE_STEPS_PROMPT_TEMPLATE.format( - context=context, + code_steps_prompt = CODE_STEPS_PROMPT_TEMPLATE.replace( + "{context}", context ) code_steps = await self._aask(code_steps_prompt) + code_steps = CodeParser.parse_code(block=None, text=code_steps) return code_steps def get_context(self, plan: Plan): @@ -74,4 +78,3 @@ class WriteCodeSteps(Action): ) # print(context) return context - diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 148851e9e..c2841be4c 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -294,7 +294,7 @@ if __name__ == "__main__": # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" - requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" + # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." data_path = "/data/lidanyang/tabular_data/titanic" From 757174366e49cb3f0a8c460b8ba8075baedc2ac7 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 6 Dec 2023 20:45:37 +0800 Subject: [PATCH 077/637] update locally --- config/config.yaml | 15 ++++++++------- metagpt/roles/ml_engineer.py | 18 +++++++++++------- metagpt/tools/functions/__init__.py | 2 +- metagpt/tools/web_browser_engine.py | 2 +- metagpt/utils/__init__.py | 4 ++-- requirements.txt | 2 -- 6 files changed, 23 insertions(+), 20 deletions(-) diff --git a/config/config.yaml b/config/config.yaml index bed67083c..694251f17 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -5,7 +5,7 @@ ## The official OPENAI_API_BASE is https://api.openai.com/v1 ## If the official OPENAI_API_BASE is not available, we recommend using the [openai-forward](https://github.com/beidongjiedeguang/openai-forward). ## Or, you can configure OPENAI_PROXY to access official OPENAI_API_BASE. -OPENAI_API_BASE: "https://api.openai.com/v1" +#OPENAI_API_BASE: "https://api.openai.com/v1" #OPENAI_PROXY: "http://127.0.0.1:8118" #OPENAI_API_KEY: "YOUR_API_KEY" # set the value to sk-xxx if you host the openai interface for open llm model OPENAI_API_MODEL: "gpt-4" @@ -24,12 +24,13 @@ RPM: 10 #### if AZURE, check https://github.com/openai/openai-cookbook/blob/main/examples/azure/chat.ipynb #### You can use ENGINE or DEPLOYMENT mode -#OPENAI_API_TYPE: "azure" -#OPENAI_API_BASE: "YOUR_AZURE_ENDPOINT" -#OPENAI_API_KEY: "YOUR_AZURE_API_KEY" -#OPENAI_API_VERSION: "YOUR_AZURE_API_VERSION" -#DEPLOYMENT_NAME: "YOUR_DEPLOYMENT_NAME" -#DEPLOYMENT_ID: "YOUR_DEPLOYMENT_ID" +OPENAI_API_TYPE: "azure" +OPENAI_API_BASE: "https://deepwisdom.openai.azure.com/" +OPENAI_API_KEY: "02ae6058d09849c691176befeae2107c" +#OPENAI_API_VERSION: "2023-05-15" +OPENAI_API_VERSION: "2023-07-01-preview" +DEPLOYMENT_ID: "GPT-4" +OPENAI_API_ENGINE: "gpt-4" #### if zhipuai from `https://open.bigmodel.cn`. You can set here or export API_KEY="YOUR_API_KEY" # ZHIPUAI_API_KEY: "YOUR_API_KEY" diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 15edb2b06..c088ff104 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -10,7 +10,7 @@ from metagpt.actions import Action from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.write_plan import WritePlan -from metagpt.actions.write_task_guide import WriteTaskGuide +# from metagpt.actions.write_task_guide import WriteTaskGuide from metagpt.logs import logger from metagpt.prompts.ml_engineer import GEN_DATA_DESC_PROMPT from metagpt.roles import Role @@ -39,7 +39,7 @@ catboost def truncate(result: str, keep_len: int = 1000) -> str: desc = "Truncated to show only the last 1000 characters\n" if result.startswith(desc): - result = result[-len(desc) :] + result = result[-len(desc):] if len(result) > keep_len: result = result[-keep_len:] @@ -110,9 +110,9 @@ class AskReview(Action): logger.info("most recent context:") latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" prompt = f"\nPlease review output from {latest_action}:\n" \ - "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" \ - "If you confirm the output and wish to continue with the current process, type CONFIRM\n" \ - "If you want to terminate the process, type exit:\n" + "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" \ + "If you confirm the output and wish to continue with the current process, type CONFIRM\n" \ + "If you want to terminate the process, type exit:\n" rsp = input(prompt) if rsp.lower() in ("exit"): @@ -148,7 +148,7 @@ class GenerateDataDesc(Action): class MLEngineer(Role): def __init__( - self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False, data_path: str = None + self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False, data_path: str = None ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") @@ -300,11 +300,15 @@ if __name__ == "__main__": # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" + from metagpt.const import DATA_PATH + requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." - data_path = "/data/lidanyang/tabular_data/titanic" + data_path = f"{DATA_PATH}/titanic" + async def main(requirement: str = requirement, auto_run: bool = True, data_path: str = data_path): role = MLEngineer(goal=requirement, auto_run=auto_run, data_path=data_path) await role.run(requirement) + fire.Fire(main) diff --git a/metagpt/tools/functions/__init__.py b/metagpt/tools/functions/__init__.py index 30ee10827..d4a1ff73b 100644 --- a/metagpt/tools/functions/__init__.py +++ b/metagpt/tools/functions/__init__.py @@ -6,4 +6,4 @@ # @Desc : from metagpt.tools.functions.register.register import registry import metagpt.tools.functions.libs.feature_engineering -import metagpt.tools.functions.libs.data_preprocess +# import metagpt.tools.functions.libs.data_preprocess diff --git a/metagpt/tools/web_browser_engine.py b/metagpt/tools/web_browser_engine.py index 453d87f31..7228ae9cf 100644 --- a/metagpt/tools/web_browser_engine.py +++ b/metagpt/tools/web_browser_engine.py @@ -7,7 +7,7 @@ from typing import Any, Callable, Coroutine, Literal, overload from metagpt.config import CONFIG from metagpt.tools import WebBrowserEngineType -from metagpt.utils.parse_html import WebPage +# from metagpt.utils.parse_html import WebPage class WebBrowserEngine: diff --git a/metagpt/utils/__init__.py b/metagpt/utils/__init__.py index f13175cf8..86cac50db 100644 --- a/metagpt/utils/__init__.py +++ b/metagpt/utils/__init__.py @@ -6,7 +6,7 @@ @File : __init__.py """ -from metagpt.utils.read_document import read_docx +# from metagpt.utils.read_document import read_docx from metagpt.utils.singleton import Singleton from metagpt.utils.token_counter import ( TOKEN_COSTS, @@ -16,7 +16,7 @@ from metagpt.utils.token_counter import ( __all__ = [ - "read_docx", + # "read_docx", "Singleton", "TOKEN_COSTS", "count_message_tokens", diff --git a/requirements.txt b/requirements.txt index 1d1bc95a1..9b75fd200 100644 --- a/requirements.txt +++ b/requirements.txt @@ -35,7 +35,6 @@ tqdm==4.64.0 # webdriver_manager<3.9 anthropic==0.3.6 typing-inspect==0.8.0 -typing_extensions==4.5.0 libcst==1.0.1 qdrant-client==1.4.0 pytest-mock==3.11.1 @@ -46,7 +45,6 @@ wrapt==1.15.0 websocket-client==0.58.0 zhipuai==1.0.7 rich==13.6.0 -nbclient==0.9.0 nbformat==5.9.2 ipython==8.17.2 ipykernel==6.27.0 From f26b2c135922eeb539fc4c907b086bbefdddff19 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 7 Dec 2023 19:21:27 +0800 Subject: [PATCH 078/637] =?UTF-8?q?=E5=8F=96=E6=B6=88=E6=B3=A8=E9=87=8A?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/tools/functions/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/functions/__init__.py b/metagpt/tools/functions/__init__.py index d4a1ff73b..30ee10827 100644 --- a/metagpt/tools/functions/__init__.py +++ b/metagpt/tools/functions/__init__.py @@ -6,4 +6,4 @@ # @Desc : from metagpt.tools.functions.register.register import registry import metagpt.tools.functions.libs.feature_engineering -# import metagpt.tools.functions.libs.data_preprocess +import metagpt.tools.functions.libs.data_preprocess From 204cda844fba774910baaa21417a40c9ae8171d8 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 7 Dec 2023 19:22:19 +0800 Subject: [PATCH 079/637] fix typo --- metagpt/actions/write_analysis_code.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index c8a28edd1..957d35f7e 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -192,7 +192,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): output_desc = TOOL_OUTPUT_DESC.get(task_type, "") hist_info = f"Previous finished code is \n\n ```Python {final_code} ``` \n\n " \ - f"Conde runtime result is {result} \n\n" + f"Runtime result is {result} \n\n" prompt = TOOL_USAGE_PROMPT.format( goal=plan.current_task.instruction, @@ -213,7 +213,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): else: hist_info = f"Previous finished code is \n\n ```Python {code_context} ``` \n\n " \ - f"Conde runtime result is {result} \n\n" + f"runtime result is {result} \n\n" prompt = GENERATE_CODE_PROMPT.format( goal=plan.current_task.instruction, From ba6a62f55aa5546d9ac274db1416d90b91c17bfb Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 7 Dec 2023 19:24:21 +0800 Subject: [PATCH 080/637] update ignore --- .gitignore | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.gitignore b/.gitignore index e03eab3d3..d01469a36 100644 --- a/.gitignore +++ b/.gitignore @@ -148,6 +148,9 @@ allure-results .DS_Store .vscode +# Config +config/config.yaml + log.txt docs/scripts/set_env.sh key.yaml From 7e343a100b449a8441ab55063ad76661d0391f46 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 7 Dec 2023 20:45:08 +0800 Subject: [PATCH 081/637] update ml functions --- .../tools/functions/libs/data_preprocess.py | 29 ++++++------- .../functions/libs/feature_engineering.py | 42 +++++++++++++------ .../functions/schemas/data_preprocess.py | 21 ++++++---- .../functions/schemas/feature_engineering.py | 36 ++++++++++------ 4 files changed, 80 insertions(+), 48 deletions(-) diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index 68c96bbc9..5579c5bd8 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -1,15 +1,12 @@ - -import pandas as pd import numpy as np - from sklearn.impute import SimpleImputer -from sklearn.preprocessing import LabelEncoder from sklearn.preprocessing import KBinsDiscretizer -from sklearn.preprocessing import MinMaxScaler -from sklearn.preprocessing import StandardScaler from sklearn.preprocessing import MaxAbsScaler -from sklearn.preprocessing import RobustScaler +from sklearn.preprocessing import MinMaxScaler +from sklearn.preprocessing import OneHotEncoder from sklearn.preprocessing import OrdinalEncoder +from sklearn.preprocessing import RobustScaler +from sklearn.preprocessing import StandardScaler from metagpt.tools.functions import registry from metagpt.tools.functions.schemas.data_preprocess import * @@ -21,13 +18,6 @@ def fill_missing_value(df: pd.DataFrame, features: list, strategy: str = 'mean', return df -# @registry.register("data_preprocess", FillMissingValue) -# def label_encode(df: pd.DataFrame, features: list,): -# for col in features: -# df[col] = LabelEncoder().fit_transform(df[col]) -# return df - - @registry.register("data_preprocess", SplitBins) def split_bins(df: pd.DataFrame, features: list, strategy: str = 'quantile',): df[features] = KBinsDiscretizer(strategy=strategy, encode='ordinal').fit_transform(df[features]) @@ -73,6 +63,17 @@ def ordinal_encode(df: pd.DataFrame, features: list,): return df +@registry.register("data_preprocess", OneHotEncoding) +def one_hot_encoding(df, cols): + enc = OneHotEncoder(handle_unknown="ignore", sparse=False) + ts_data = enc.fit_transform(df[cols]) + new_columns = enc.get_feature_names_out(cols) + ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) + df.drop(cols, axis=1, inplace=True) + df = pd.concat([df, ts_data], axis=1) + return df + + if __name__ == '__main__': def run(): V = { diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index 0573f362d..4780e4fa0 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -8,7 +8,8 @@ import itertools from dateutil.relativedelta import relativedelta from pandas.api.types import is_numeric_dtype -from sklearn.preprocessing import PolynomialFeatures, OneHotEncoder +from sklearn.model_selection import KFold +from sklearn.preprocessing import PolynomialFeatures from metagpt.tools.functions import registry from metagpt.tools.functions.schemas.feature_engineering import * @@ -29,17 +30,6 @@ def polynomial_expansion(df, cols, degree=2): return df -@registry.register("feature_engineering", OneHotEncoding) -def one_hot_encoding(df, cols): - enc = OneHotEncoder(handle_unknown="ignore", sparse=False) - ts_data = enc.fit_transform(df[cols]) - new_columns = enc.get_feature_names_out(cols) - ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) - df.drop(cols, axis=1, inplace=True) - df = pd.concat([df, ts_data], axis=1) - return df - - @registry.register("feature_engineering", FrequencyEncoding) def frequency_encoding(df, cols): for col in cols: @@ -48,6 +38,31 @@ def frequency_encoding(df, cols): return df +@registry.register("feature_engineering", TargetMeanEncoder) +def target_mean_encoder(df, col, label): + encoder_dict = df.groupby(col)[label].mean().to_dict() + df[f"{col}_target_mean"] = df[col].map(encoder_dict) + return df + + +@registry.register("feature_engineering", KFoldTargetMeanEncoder) +def k_fold_target_mean_encoder(df, col, label, n_splits=5, random_state=2021): + tmp = df.copy() + kf = KFold(n_splits=n_splits, shuffle=True, random_state=random_state) + + global_mean = tmp[label].mean() + col_name = f"{col}_kf_target_mean" + for trn_idx, val_idx in kf.split(tmp, tmp[label]): + _trn, _val = tmp.iloc[trn_idx], tmp.iloc[val_idx] + tmp.loc[tmp.index[val_idx], col_name] = _val[col].map( + _trn.groupby(col)[label].mean() + ) + tmp[col_name].fillna(global_mean, inplace=True) + encoder_dict = tmp.groupby(col)[col_name].mean().to_dict() + df[f"{col}_kf_target_mean"] = df[col].map(encoder_dict) + return df + + @registry.register("feature_engineering", CatCross) def cat_cross(df, cols, max_cat_num=100): for col in cols: @@ -56,7 +71,8 @@ def cat_cross(df, cols, max_cat_num=100): for col1, col2 in itertools.combinations(cols, 2): cross_col = f"{col1}_cross_{col2}" - df[cross_col] = df[col1].astype(str) + "_" + df[col2].astype(str) + crossed = df[col1].astype(str) + "_" + df[col2].astype(str) + df[cross_col] = crossed.astype('category').cat.codes return df diff --git a/metagpt/tools/functions/schemas/data_preprocess.py b/metagpt/tools/functions/schemas/data_preprocess.py index 40e1d64e0..16b97aeac 100644 --- a/metagpt/tools/functions/schemas/data_preprocess.py +++ b/metagpt/tools/functions/schemas/data_preprocess.py @@ -8,14 +8,13 @@ class FillMissingValue(ToolSchema): """Completing missing values with simple strategies""" df: pd.DataFrame = tool_field(description="input dataframe") features: list = tool_field(description="columns to be processed") - strategy: str = tool_field(description="the imputation strategy", default='mean') - fill_value: int = tool_field(description="fill_value is used to replace all occurrences of missing_values", default=None) - - -# class LabelEncode(ToolSchema): -# """Completing missing values with simple strategies""" -# df: pd.DataFrame = tool_field(description="input dataframe") -# features: list = tool_field(description="columns to be processed") + strategy: str = tool_field( + description="the imputation strategy", + default='mean', + enum=['mean', 'median', 'most_frequent', 'constant'] + ) + fill_value: int = tool_field( + description="fill_value is used to replace all occurrences of missing_values", default=None) class SplitBins(ToolSchema): @@ -60,3 +59,9 @@ class OrdinalEncode(ToolSchema): df: pd.DataFrame = tool_field(description="input dataframe") features: list = tool_field(description="columns to be processed") + +class OneHotEncoding(ToolSchema): + """Apply one-hot encoding to specified categorical columns, the original columns will be dropped.""" + + df: pd.DataFrame = tool_field(description="DataFrame to process.") + cols: list = tool_field(description="Categorical columns to be one-hot encoded and dropped.") diff --git a/metagpt/tools/functions/schemas/feature_engineering.py b/metagpt/tools/functions/schemas/feature_engineering.py index df2eebff6..5c89d9b16 100644 --- a/metagpt/tools/functions/schemas/feature_engineering.py +++ b/metagpt/tools/functions/schemas/feature_engineering.py @@ -12,29 +12,39 @@ from metagpt.tools.functions.schemas.base import ToolSchema, tool_field class PolynomialExpansion(ToolSchema): - """Generate polynomial and interaction features from selected columns, excluding the bias column.""" + """Add polynomial and interaction features from selected numeric columns, excluding the bias column.""" df: pd.DataFrame = tool_field(description="DataFrame to process.") cols: list = tool_field(description="Columns for polynomial expansion.") degree: int = tool_field(description="Degree of polynomial features.", default=2) -class OneHotEncoding(ToolSchema): - """Apply one-hot encoding to specified categorical columns, the original columns will be dropped.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - cols: list = tool_field(description="Categorical columns to be one-hot encoded.") - - class FrequencyEncoding(ToolSchema): - """Convert categorical columns to frequency encoding.""" + """Add value counts of categorical columns as new features.""" df: pd.DataFrame = tool_field(description="DataFrame to process.") cols: list = tool_field(description="Categorical columns to be frequency encoded.") +class TargetMeanEncoder(ToolSchema): + """Encodes a categorical column by the mean of the label column, and adds the result as a new feature.""" + + df: pd.DataFrame = tool_field(description="DataFrame to process.") + col: str = tool_field(description="Column to be mean encoded.") + label: str = tool_field(description="Predicted label column.") + + +class KFoldTargetMeanEncoder(ToolSchema): + """Adds a new feature to the DataFrame by k-fold mean encoding of a categorical column using the label column.""" + df: pd.DataFrame = tool_field(description="DataFrame to process.") + col: str = tool_field(description="Column to be k-fold mean encoded.") + label: str = tool_field(description="Predicted label column.") + n_splits: int = tool_field(description="Number of splits for K-fold.", default=5) + random_state: int = tool_field(description="Random seed.", default=2021) + + class CatCross(ToolSchema): - """Create pairwise crossed features from categorical columns, joining values with '_'.""" + """Add pairwise crossed features and convert them to numerical features.""" df: pd.DataFrame = tool_field(description="DataFrame to process.") cols: list = tool_field(description="Columns to be pairwise crossed.") @@ -44,7 +54,7 @@ class CatCross(ToolSchema): class GroupStat(ToolSchema): - """Perform aggregation operations on a specified column grouped by certain categories.""" + """Aggregate specified column in a DataFrame grouped by another column, adding new features named '__by_'.""" df: pd.DataFrame = tool_field(description="DataFrame to process.") group_col: str = tool_field(description="Column used for grouping.") @@ -56,7 +66,7 @@ class GroupStat(ToolSchema): class ExtractTimeComps(ToolSchema): - """Extract specific time components from a designated time column in a DataFrame.""" + """Extract and add specific time components as new features from a designated time column.""" df: pd.DataFrame = tool_field(description="DataFrame to process.") time_col: str = tool_field( @@ -69,7 +79,7 @@ class ExtractTimeComps(ToolSchema): class FeShiftByTime(ToolSchema): - """Shift column values in a DataFrame based on specified time intervals.""" + """Shift column values based on specified time intervals and add the resulting new features to the DataFrame. New features are named in the format of '__lag__'.""" df: pd.DataFrame = tool_field(description="DataFrame to process.") time_col: str = tool_field(description="Column for time-based shifting.") From fe2b79fedc407afe72ad855ea6187afe11108beb Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 7 Dec 2023 20:48:00 +0800 Subject: [PATCH 082/637] refine ml prompt --- metagpt/actions/write_analysis_code.py | 114 +++++++++--------------- metagpt/prompts/ml_engineer.py | 118 ++++++++++++++++++++++--- metagpt/roles/ml_engineer.py | 75 ++++++---------- metagpt/utils/common.py | 14 +++ 4 files changed, 192 insertions(+), 129 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 957d35f7e..f96ade1b4 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -15,15 +15,11 @@ from metagpt.prompts.ml_engineer import ( TOO_ORGANIZATION_PROMPT, ML_SPECIFIC_PROMPT, ML_MODULE_MAP, - TOOL_OUTPUT_DESC, - TOOL_USAGE_PROMPT, + TOOL_OUTPUT_DESC, DATA_PROCESS_PROMPT, ) from metagpt.schema import Message, Plan from metagpt.tools.functions import registry -from metagpt.utils.common import create_func_config -from metagpt.prompts.ml_engineer import GEN_DATA_DESC_PROMPT, GENERATE_CODE_PROMPT -from metagpt.utils.common import CodeParser -from metagpt.actions.execute_code import ExecutePyCode +from metagpt.utils.common import create_func_config, remove_comments class BaseWriteAnalysisCode(Action): @@ -51,13 +47,13 @@ class BaseWriteAnalysisCode(Action): # 添加默认的提示词 if ( - default_system_msg not in messages[0]["content"] - and messages[0]["role"] != "system" + default_system_msg not in messages[0]["content"] + and messages[0]["role"] != "system" ): messages.insert(0, {"role": "system", "content": default_system_msg}) elif ( - default_system_msg not in messages[0]["content"] - and messages[0]["role"] == "system" + default_system_msg not in messages[0]["content"] + and messages[0]["role"] == "system" ): messages[0] = { "role": "system", @@ -66,7 +62,7 @@ class BaseWriteAnalysisCode(Action): return messages async def run( - self, context: List[Message], plan: Plan = None, code_steps: str = "" + self, context: List[Message], plan: Plan = None, code_steps: str = "" ) -> str: """Run of a code writing action, used in data analysis or modeling @@ -87,12 +83,12 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): super().__init__(name, context, llm) async def run( - self, - context: [List[Message]], - plan: Plan = None, - code_steps: str = "", - system_msg: str = None, - **kwargs, + self, + context: [List[Message]], + plan: Plan = None, + code_steps: str = "", + system_msg: str = None, + **kwargs, ) -> str: context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) prompt = self.process_msg(context, system_msg) @@ -102,7 +98,6 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - execute_code = ExecutePyCode() @staticmethod def _parse_recommend_tools(module: str, recommend_tools: list) -> List[Dict]: @@ -126,10 +121,10 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): return tool_catalog async def _tool_recommendation( - self, - context: [List[Message]], - code_steps: str, - available_tools: list + self, + task: str, + code_steps: str, + available_tools: list ) -> list: """ Recommend tools for the specified task. @@ -142,86 +137,63 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): Returns: list: recommended tools for the specified task """ - system_prompt = TOOL_RECOMMENDATION_PROMPT.format( + prompt = TOOL_RECOMMENDATION_PROMPT.format( + current_task=task, code_steps=code_steps, available_tools=available_tools, ) - prompt = self.process_msg(context, system_prompt) - tool_config = create_func_config(SELECT_FUNCTION_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) recommend_tools = rsp["recommend_tools"] return recommend_tools - async def run( - self, - context: List[Message], - plan: Plan = None, - code_steps: str = "", - **kwargs, + self, + context: List[Message], + plan: Plan = None, + code_steps: str = "", + column_info: str = "", ) -> str: task_type = plan.current_task.task_type - logger.info(f"task_type is: {task_type}") available_tools = registry.get_all_schema_by_module(task_type) - - # special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") + special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") finished_tasks = plan.get_finished_tasks() - code_context = [task.code for task in finished_tasks] - + code_context = [remove_comments(task.code) for task in finished_tasks] code_context = "\n\n".join(code_context) - ### add runtime info - result, success = await self.execute_code.run(code_context) - logger.info(result) - if len(available_tools) > 0: available_tools = [ {k: tool[k] for k in ["name", "description"] if k in tool} for tool in available_tools ] - final_code = code_context - - recommend_tools = await self._tool_recommendation(context, code_steps, available_tools) + recommend_tools = await self._tool_recommendation( + plan.current_task.instruction, + code_steps, + available_tools + ) tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) logger.info(f"Recommended tools: \n{recommend_tools}") module_name = ML_MODULE_MAP[task_type] output_desc = TOOL_OUTPUT_DESC.get(task_type, "") - - hist_info = f"Previous finished code is \n\n ```Python {final_code} ``` \n\n " \ - f"Runtime result is {result} \n\n" - - prompt = TOOL_USAGE_PROMPT.format( - goal=plan.current_task.instruction, - context=hist_info, + prompt = DATA_PROCESS_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + special_prompt=special_prompt, code_steps=code_steps, module_name=module_name, output_desc=output_desc, function_catalog=tool_catalog, ) - - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - - rsp = await self.llm.aask_code(prompt, **tool_config) - logger.info(f"rsp is: {rsp}") - final_code = final_code + "\n\n" + rsp["code"] - - return final_code - else: - hist_info = f"Previous finished code is \n\n ```Python {code_context} ``` \n\n " \ - f"runtime result is {result} \n\n" + context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) + context.append(Message(content=special_prompt, role="user")) + prompt = self.process_msg(context) - prompt = GENERATE_CODE_PROMPT.format( - goal=plan.current_task.instruction, - context=hist_info, - ) - - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - logger.info(f"prompt is: {prompt}") - rsp = await self.llm.aask_code(prompt, **tool_config) - logger.info(f"rsp is: {rsp}") - return rsp["code"] + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + rsp = await self.llm.aask_code(prompt, **tool_config) + return rsp['code'] diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index b68dadc9a..88cebf68a 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -8,19 +8,22 @@ GEN_DATA_DESC_PROMPT = """ Here is the head 5 rows of the dataset: {data_head} -Please provide a brief one-sentence background of the dataset, and concise descriptions for each column. Keep descriptions short yet informative. +Please provide a brief one-sentence background of the dataset, and concise meaning for each column. Keep descriptions short. Output the information in a JSON format, as shown in this example: ```json { "data_desc": "Brief dataset background.", "column_desc": { - "column_name1": "Description of the first column.", - "column_name2": "Description of the second column.", + "column_name1": "Abstract meaning of the first column.", + "column_name2": "Abstract meaning of the second column.", ... } } ``` + +# Constraints: +- Don't contain specific values or examples found in the data column. """ ASSIGN_TASK_TYPE_PROMPT = """ @@ -53,19 +56,22 @@ ASSIGN_TASK_TYPE = { } TOOL_RECOMMENDATION_PROMPT = """ -Your are a tool recommender, the main goal is to recommend suitable tools for current task before coding. A tool means a function that can be used to help you solve the task. +## User Requirement: +{current_task} -## List of Available Tools: -{available_tools} - -This is a task guide for the current task, including detailed code steps. You can refer to it when recommending tools. +## Task +Recommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. +This is a detailed code steps for current task. You can refer to it when recommending tools. {code_steps} +## Available Tools: +{available_tools} + ## Tool Selection and Instructions: -- For the task, choose up to five tools that are most likely to be useful in solving the task. +- Select tools most relevant to completing the 'User Requirement'. - If you believe that no tools are suitable, indicate with an empty list. - Only list the names of the tools, not the full schema of each tool. -- The result should only contain tool names that are in the list of available tools. +- Ensure selected tools are listed in 'Available Tools'. """ SELECT_FUNCTION_TOOLS = { @@ -149,6 +155,34 @@ Finish your coding tasks as a helpful programmer based on the tools. """ +TOOL_USAGE_PROMPT = """ +## Target +{goal} + +## History Info +{context} + +## Available Tools: +Each function is described in JSON format, including the function name and parameters. {output_desc} +{function_catalog} + +When you call a function above, you should import the function from `{module_name}` first, e.g.: +```python +from metagpt.tools.functions.libs.data_preprocess import fill_missing_value +```end + +## Your Output Format: +Generate the complete code for this task: +```python +# Tools used: [function names or 'none'] + +```end + +## Attention: +Make sure use the columns from the dataset columns +Finish your coding tasks as a helpful programmer based on the tools. +""" + TOO_ORGANIZATION_PROMPT = """ The previous conversation has provided all tasks step-by-step for the use goal and their statuses. Now, begin writing code for the current task. This code should writen strictly on the basis of all previous completed tasks code, not a standalone code. And avoid writing duplicate code that has already been written in previous tasks, such as repeated import of packages, reading data, etc. @@ -197,6 +231,66 @@ The current task is about feature engineering. when performing it, please adhere - Before generating a new feature, ensure the used features are already processed and ready to use. """ +DATA_PROCESS_PROMPT = """ +# Background +As a data scientist, you need to help user to achieve the goal [{user_requirement}] step-by-step in an continuous Jupyter notebook. + +## Done Tasks +```python +{history_code} +```end + +## Current Task +{current_task} + +# Latest Data Info +Latest data info after previous tasks: +{column_info} + +# Task +Write a Python function for 'Current Task'. Start by copying the input DataFrame. Avoid duplicating code from 'Done Tasks'. +Specifically, {special_prompt} + +# Code Steps: +Follow steps below when you writing code if it's convenient. +{code_steps} + +# Capabilities +- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of python functions. +- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. +- You can do anything about data preprocessing, feature engineering, model training, etc.. + +# Available Tools: +Each function tool is described in JSON format. {output_desc} +When you call a function below, import the function from `{module_name}` first. +{function_catalog} + +# Output Example: +when current task is "fill missing value and handle outliers", the output code be like: +```python +from metagpt.tools.functions.libs.data_preprocess import fill_missing_value + +def function_name(df): + df_processed = df.copy() + num_cols = df_processed.select_dtypes(include='number').columns.tolist() + df_processed = fill_missing_value(df_processed, num_cols, 'mean') + + for col in num_cols: + low, high = df_processed[col].quantile([0.01, 0.99]) + df_processed[col] = df_processed[col].clip(low, high) + return df_processed + +df_processed = function_name(df) +print(df_processed.info()) +```end + +# Constraints: +- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. +- Prioritize using pre-defined tools for the same functionality. +- Return DataFrame should always be named `df_processed`, while the input DataFrame should based on the done tasks' output DataFrame. +- Limit to one print statement for the output DataFrame's info. +""" + MODEL_TRAIN_PROMPT = """ The current task is about training a model, please ensure high performance: - Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as lightGBM, XGBoost, CatBoost, etc. @@ -204,9 +298,9 @@ The current task is about training a model, please ensure high performance: - Use the data from previous task result directly, do not mock or reload data yourself. """ -DATA_PREPROCESS_OUTPUT_DESC = "Please note that all functions uniformly output a processed pandas.DataFrame, facilitating seamless integration into the broader workflow." +DATA_PREPROCESS_OUTPUT_DESC = "Please note that all functions output a updated pandas.DataFrame after data preprocessing." -FEATURE_ENGINEERING_OUTPUT_DESC = "Please note that all functions uniformly output updated pandas.DataFrame with feature engineering applied." +FEATURE_ENGINEERING_OUTPUT_DESC = "Please note that all functions output a updated pandas.DataFrame with new features added or existing features modified." CLASSIFICATION_MODEL_OUTPUT_DESC = "" diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index deb76f0a9..4ad24df52 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,21 +1,21 @@ -import glob import json +import re from typing import List import fire import pandas as pd -import re from metagpt.actions import Action from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.actions.write_plan import WritePlan +from metagpt.const import DATA_PATH from metagpt.logs import logger from metagpt.prompts.ml_engineer import GEN_DATA_DESC_PROMPT from metagpt.roles import Role from metagpt.schema import Message, Plan from metagpt.utils.common import CodeParser -from metagpt.actions.write_code_steps import WriteCodeSteps STRUCTURAL_CONTEXT = """ ## User Requirement @@ -70,32 +70,16 @@ def read_data(file: str) -> pd.DataFrame: return df -def get_samples(df: pd.DataFrame) -> str: +def get_column_info(df: pd.DataFrame) -> str: data = [] - - if len(df) > 5: - df_ = df.sample(5, random_state=0) - else: - df_ = df - - for i in list(df_): + for i in df.columns: nan_freq = float("%.2g" % (df[i].isna().mean() * 100)) n_unique = df[i].nunique() - s = df_[i].tolist() + data.append([i, df[i].dtype, nan_freq, n_unique]) - if str(df[i].dtype) == "float64": - s = [round(sample, 2) if not pd.isna(sample) else None for sample in s] - - data.append([df_[i].name, df[i].dtype, nan_freq, n_unique, s]) samples = pd.DataFrame( data, - columns=[ - "Column_name", - "Data_type", - "NaN_Frequency(%)", - "N_unique", - "Samples", - ], + columns=["Column_name", "Data_type", "NaN_Frequency(%)", "N_unique"], ) return samples.to_string(index=False) @@ -124,20 +108,19 @@ class AskReview(Action): class GenerateDataDesc(Action): - async def run(self, files: list) -> dict: + async def run(self, file: str) -> dict: data_desc = {} - for file in files: - df = read_data(file) - file_name = file.split("/")[-1] - data_head = df.head().to_dict(orient="list") - data_head = json.dumps(data_head, indent=4, ensure_ascii=False) - prompt = GEN_DATA_DESC_PROMPT.replace("{data_head}", data_head) - rsp = await self._aask(prompt) - rsp = CodeParser.parse_code(block=None, text=rsp) - data_desc[file_name] = {} - data_desc[file_name]["path"] = file - data_desc[file_name]["description"] = rsp - data_desc[file_name]["column_info"] = get_samples(df) + df = read_data(file) + data_head = df.head().to_dict(orient="list") + data_head = json.dumps(data_head, indent=4, ensure_ascii=False) + prompt = GEN_DATA_DESC_PROMPT.replace("{data_head}", data_head) + rsp = await self._aask(prompt) + rsp = CodeParser.parse_code(block=None, text=rsp) + rsp = json.loads(rsp) + data_desc["path"] = file + data_desc["data_desc"] = rsp["data_desc"] + data_desc["column_desc"] = rsp["column_desc"] + data_desc["column_info"] = get_column_info(df) return data_desc @@ -159,7 +142,6 @@ class MLEngineer(Role): if self.data_path: self.data_desc = await self._generate_data_desc() - # create initial plan and update until confirmation await self._update_plan() @@ -181,13 +163,14 @@ class MLEngineer(Role): self.plan.finish_current_task() self.working_memory.clear() + if "print(df_processed.info())" in code: + self.data_desc["column_info"] = result else: # update plan according to user's feedback and to take on changed tasks await self._update_plan() async def _generate_data_desc(self): - files = glob.glob(self.data_path + "/*.csv") - data_desc = await GenerateDataDesc().run(files=files) + data_desc = await GenerateDataDesc().run(self.data_path) return data_desc async def _write_and_exec_code(self, max_retry: int = 3): @@ -201,9 +184,11 @@ class MLEngineer(Role): success = False while not success and counter < max_retry: context = self.get_useful_memories() - # breakpoint() - column_names_dict = {key: value["column_info"] for key,value in self.data_desc.items()} + # print("*" * 10) + # print(context) + # print("*" * 10) + # breakpoint() if not self.use_tools or self.plan.current_task.task_type == "other": logger.info("Write code with pure generation") @@ -214,9 +199,9 @@ class MLEngineer(Role): cause_by = WriteCodeByGenerate else: logger.info("Write code with tools") - + column_info = self.data_desc['column_info'] code = await WriteCodeWithTools().run( - context=context, plan=self.plan, code_steps=code_steps, **{"column_names": column_names_dict} + context=context, plan=self.plan, code_steps=code_steps, column_info=column_info ) cause_by = WriteCodeWithTools @@ -296,10 +281,8 @@ if __name__ == "__main__": # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" - from metagpt.const import DATA_PATH - requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." - data_path = f"{DATA_PATH}/titanic" + data_path = f"{DATA_PATH}/titanic.csv" async def main(requirement: str = requirement, auto_run: bool = True, data_path: str = data_path): role = MLEngineer(goal=requirement, auto_run=auto_run, data_path=data_path) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 8f8edbc6d..168966ef7 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -315,3 +315,17 @@ def create_func_config(func_schema: dict) -> dict: "tools": tools, "tool_choice": tool_choice, } + + +def remove_comments(code_str): + """Remove comments from code.""" + pattern = r"(\".*?\"|\'.*?\')|(\#.*?$)" + def replace_func(match): + if match.group(2) is not None: + return "" + else: + return match.group(1) + + clean_code = re.sub(pattern, replace_func, code_str, flags=re.MULTILINE) + clean_code = os.linesep.join([s.rstrip() for s in clean_code.splitlines() if s.strip()]) + return clean_code From 13e2b058125f45f43ff998483a7e175ddaeb5883 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 8 Dec 2023 11:01:13 +0800 Subject: [PATCH 083/637] add reflection change write code internal ppl --- metagpt/actions/debug_code.py | 111 +++++++++++++++++++++++++ metagpt/actions/write_analysis_code.py | 65 ++++++++------- metagpt/prompts/ml_engineer.py | 25 ++++-- metagpt/roles/ml_engineer.py | 68 +++++++++++---- 4 files changed, 219 insertions(+), 50 deletions(-) create mode 100644 metagpt/actions/debug_code.py diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py new file mode 100644 index 000000000..3d460fa40 --- /dev/null +++ b/metagpt/actions/debug_code.py @@ -0,0 +1,111 @@ +from typing import Dict, List, Union, Tuple, Optional, Any + +from metagpt.actions import Action +from metagpt.logs import logger +from metagpt.schema import Message, Plan +from metagpt.utils.common import CodeParser +from metagpt.actions.write_analysis_code import BaseWriteAnalysisCode + +DEBUG_REFLECTION_EXAMPLE = '''Example 1: + [previous impl]: + ```python + def add(a: int, b: int) -> int: + """ + Given integers a and b, return the total value of a and b. + """ + return a - b + ``` + + [runtime Error]: + Tested passed: + + Tests failed: + assert add(1, 2) == 3 # output: -1 + assert add(1, 2) == 4 # output: -1 + + [reflection on previous impl]: + The implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input. + + [improved impl]: + ```python + def add(a: int, b: int) -> int: + """ + Given integers a and b, return the total value of a and b. + """ + return a + b + ``` + ''' + +REFLECTION_PROMPT = """ + Here is an example for you. + {debug_example} + [requirement] + {goal} + [previous impl] + {code} + [runtime Error] + {runtime_result} + + Analysis the error step by step, provide me improve method. Do not repeat [previous impl] + [reflection on previous impl]: + xxx + + """ + + +def message_to_str(message: Message) -> str: + return f"{message.role}: {message.content}" + + +def messages_to_str(messages: List[Message]) -> str: + return "\n".join([message_to_str(message) for message in messages]) + + +class DebugCode(BaseWriteAnalysisCode): + name: str = "debugcode" + context: Optional[str] = None + llm: None + + def __init__(self, **kwargs: Any): + super().__init__(**kwargs) + + async def run_reflection(self, plan, code, runtime_result) -> str: + info = [] + reflection_prompt = REFLECTION_PROMPT.format(debug_example=DEBUG_REFLECTION_EXAMPLE, + goal=plan.goal, + code=code, + runtime_result=runtime_result + ) + system_prompt = "You are an AI Python assistant. You will be given your previous implementation of a function, runtime error results, and a hint to change the implementation appropriately. Write your full implementation " + info.append(Message(role="system", content=system_prompt)) + info.append(Message(role="assistant", content=reflection_prompt)) + + msg = messages_to_str(info) + resp = await self.llm.aask(msg=msg) + logger.info(f"reflection is {resp}") + return resp + + async def rewrite_code(self, reflection: str = "") -> str: + """ + 根据reflection重写代码 + """ + info = [] + info.append(Message(role="assistant", content=f"[reflection]: \n {reflection}")) + info.append(Message(role="user", content=f"[improved impl]:\n Return in Python block")) + msg = messages_to_str(info) + resp = await self.llm.aask(msg=msg) + logger.info(f"improve code is {resp}") + improv_code = CodeParser.parse_code(block=None, text=resp) + return improv_code + + async def run(self, + plan: Plan = None, + code: str = "", + runtime_result: str = "") -> str: + """ + 根据当前运行代码和报错信息进行reflection和纠错 + """ + reflection = await self.run_reflection(plan, code, runtime_result) + # 根据reflection结果重写代码 + improv_code = await self.rewrite_code(reflection) + return improv_code diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 957d35f7e..777064f93 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -4,7 +4,7 @@ @Author : orange-crow @File : write_code_v2.py """ -from typing import Dict, List, Union, Tuple +from typing import Dict, List, Union, Tuple, Optional, Any from metagpt.actions import Action from metagpt.logs import logger @@ -12,7 +12,7 @@ from metagpt.prompts.ml_engineer import ( TOOL_RECOMMENDATION_PROMPT, SELECT_FUNCTION_TOOLS, CODE_GENERATOR_WITH_TOOLS, - TOO_ORGANIZATION_PROMPT, + TOOL_ORGANIZATION_PROMPT, ML_SPECIFIC_PROMPT, ML_MODULE_MAP, TOOL_OUTPUT_DESC, @@ -22,10 +22,13 @@ from metagpt.schema import Message, Plan from metagpt.tools.functions import registry from metagpt.utils.common import create_func_config from metagpt.prompts.ml_engineer import GEN_DATA_DESC_PROMPT, GENERATE_CODE_PROMPT -from metagpt.utils.common import CodeParser + from metagpt.actions.execute_code import ExecutePyCode + + + class BaseWriteAnalysisCode(Action): DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" @@ -80,6 +83,8 @@ class BaseWriteAnalysisCode(Action): """ + + class WriteCodeByGenerate(BaseWriteAnalysisCode): """Write code fully by generation""" @@ -153,7 +158,6 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): recommend_tools = rsp["recommend_tools"] return recommend_tools - async def run( self, context: List[Message], @@ -164,25 +168,23 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): task_type = plan.current_task.task_type logger.info(f"task_type is: {task_type}") available_tools = registry.get_all_schema_by_module(task_type) + special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") - # special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") - + column_names = kwargs.get("column_names", {}) finished_tasks = plan.get_finished_tasks() code_context = [task.code for task in finished_tasks] code_context = "\n\n".join(code_context) - ### add runtime info - result, success = await self.execute_code.run(code_context) - logger.info(result) - if len(available_tools) > 0: available_tools = [ {k: tool[k] for k in ["name", "description"] if k in tool} for tool in available_tools ] - final_code = code_context + final_code = {} + new_code = "" + code_steps_dict = eval(code_steps) recommend_tools = await self._tool_recommendation(context, code_steps, available_tools) tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) @@ -191,33 +193,40 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name = ML_MODULE_MAP[task_type] output_desc = TOOL_OUTPUT_DESC.get(task_type, "") - hist_info = f"Previous finished code is \n\n ```Python {final_code} ``` \n\n " \ - f"Runtime result is {result} \n\n" - prompt = TOOL_USAGE_PROMPT.format( - goal=plan.current_task.instruction, - context=hist_info, - code_steps=code_steps, - module_name=module_name, - output_desc=output_desc, - function_catalog=tool_catalog, - ) + for idx, tool in enumerate(recommend_tools): + hist_info = f"Previous finished code is \n\n ```Python {code_context} ``` \n\n " - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + prompt = TOOL_USAGE_PROMPT.format( + goal=plan.current_task.instruction, + context=hist_info, + code_steps=code_steps, + column_names=column_names, + special_prompt=special_prompt, + module_name=module_name, + output_desc=output_desc, + function_catalog=tool_catalog[idx], + ) - rsp = await self.llm.aask_code(prompt, **tool_config) - logger.info(f"rsp is: {rsp}") - final_code = final_code + "\n\n" + rsp["code"] + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - return final_code + rsp = await self.llm.aask_code(prompt, **tool_config) + logger.info(f"rsp is: {rsp}") + # final_code = final_code + "\n\n" + rsp["code"] + # final_code[key] = rsp["code"] + new_code = new_code + "\n\n" + rsp["code"] + code_context = code_context + "\n\n" + rsp["code"] + return new_code else: - hist_info = f"Previous finished code is \n\n ```Python {code_context} ``` \n\n " \ - f"runtime result is {result} \n\n" + hist_info = f"Previous finished code is \n\n ```Python {code_context} ``` \n\n " prompt = GENERATE_CODE_PROMPT.format( goal=plan.current_task.instruction, context=hist_info, + code_steps=code_steps, + special_prompt=special_prompt, + # column_names=column_names ) tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index b68dadc9a..9a234478c 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -105,9 +105,15 @@ TOOL_USAGE_PROMPT = """ ## Target {goal} +Specifically, {special_prompt} + ## History Info {context} +## Code Steps for Current Task: +Follow steps below when you writing code if it's convenient. +{code_steps} + ## Available Tools: Each function is described in JSON format, including the function name and parameters. {output_desc} {function_catalog} @@ -125,7 +131,7 @@ Generate the complete code for this task: ```end ## Attention: -Make sure use the columns from the dataset columns +Make sure use the columns from the dataset columns: {column_names} Finish your coding tasks as a helpful programmer based on the tools. """ @@ -133,23 +139,30 @@ GENERATE_CODE_PROMPT = """ ## Target {goal} +Specifically, {special_prompt} + + ## History Info {context} +## Code Steps for Current Task: +Follow steps below when you writing code if it's convenient. +{code_steps} + ## Your Output Format: Generate the complete code for this task: ```python -# Tools used: [function names or 'none'] - -```end +import pandas as pd + +``` ## Attention: Make sure use the columns from the dataset columns -Finish your coding tasks as a helpful programmer based on the tools. +Finish your coding tasks as a helpful programmer based on the code. """ -TOO_ORGANIZATION_PROMPT = """ +TOOL_ORGANIZATION_PROMPT = """ The previous conversation has provided all tasks step-by-step for the use goal and their statuses. Now, begin writing code for the current task. This code should writen strictly on the basis of all previous completed tasks code, not a standalone code. And avoid writing duplicate code that has already been written in previous tasks, such as repeated import of packages, reading data, etc. Specifically, {special_prompt} diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index deb76f0a9..b5904213c 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -16,6 +16,7 @@ from metagpt.roles import Role from metagpt.schema import Message, Plan from metagpt.utils.common import CodeParser from metagpt.actions.write_code_steps import WriteCodeSteps +from metagpt.actions.debug_code import DebugCode STRUCTURAL_CONTEXT = """ ## User Requirement @@ -36,10 +37,13 @@ catboost """ + + + def truncate(result: str, keep_len: int = 1000) -> str: desc = "Truncated to show only the last 1000 characters\n" if result.startswith(desc): - result = result[-len(desc) :] + result = result[-len(desc):] if len(result) > keep_len: result = result[-keep_len:] @@ -110,9 +114,9 @@ class AskReview(Action): logger.info("most recent context:") latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" prompt = f"\nPlease review output from {latest_action}:\n" \ - "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" \ - "If you confirm the output and wish to continue with the current process, type CONFIRM\n" \ - "If you want to terminate the process, type exit:\n" + "If you want to change a task in the plan, say 'change task task_id, ... (things to change)'\n" \ + "If you confirm the output and wish to continue with the current process, type CONFIRM\n" \ + "If you want to terminate the process, type exit:\n" rsp = input(prompt) if rsp.lower() in ("exit"): @@ -143,7 +147,7 @@ class GenerateDataDesc(Action): class MLEngineer(Role): def __init__( - self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False, data_path: str = None + self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False, data_path: str = None ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") @@ -159,7 +163,6 @@ class MLEngineer(Role): if self.data_path: self.data_desc = await self._generate_data_desc() - # create initial plan and update until confirmation await self._update_plan() @@ -185,6 +188,15 @@ class MLEngineer(Role): # update plan according to user's feedback and to take on changed tasks await self._update_plan() + + finished_tasks = self.plan.get_finished_tasks() + if len(finished_tasks) == len(self.plan.tasks): + code_context = [task.code for task in finished_tasks] + code_context = "\n\n".join(code_context) + result, success = await self.execute_code.run(code_context) + # truncated the result + print(truncate(result)) + async def _generate_data_desc(self): files = glob.glob(self.data_path + "/*.csv") data_desc = await GenerateDataDesc().run(files=files) @@ -198,16 +210,29 @@ class MLEngineer(Role): ) counter = 0 + improve_code = "" success = False + + finished_tasks = self.plan.get_finished_tasks() + code_context = [task.code for task in finished_tasks] + code_context = "\n\n".join(code_context) + while not success and counter < max_retry: - context = self.get_useful_memories() + if counter == 0: + context = self.get_useful_memories() + else: + # improve_code = await DebugCode().run(plan=self.plan, + # code= code_context + "\n\n" + code, + # runtime_result=self.working_memory.get()) + improve_code = "" + # breakpoint() - column_names_dict = {key: value["column_info"] for key,value in self.data_desc.items()} + column_names_dict = {key: value["column_info"] for key, value in self.data_desc.items()} if not self.use_tools or self.plan.current_task.task_type == "other": logger.info("Write code with pure generation") - # code = "print('abc')" + code = await WriteCodeByGenerate().run( context=context, plan=self.plan, code_steps=code_steps, temperature=0.0 ) @@ -215,16 +240,24 @@ class MLEngineer(Role): else: logger.info("Write code with tools") - code = await WriteCodeWithTools().run( - context=context, plan=self.plan, code_steps=code_steps, **{"column_names": column_names_dict} - ) - cause_by = WriteCodeWithTools + if improve_code!="": + code = improve_code + logger.info(f"new code {code}") + cause_by = DebugCode + else: + code = await WriteCodeWithTools().run( + context=context, plan=self.plan, code_steps=code_steps, **{"column_names": column_names_dict} + ) + + cause_by = WriteCodeWithTools self.working_memory.add( Message(content=code, role="assistant", cause_by=cause_by) ) - result, success = await self.execute_code.run(code) + # debug on code, run on runcode with finished code and new_df + runcode = code_context + "\n\n" + code + result, success = await self.execute_code.run(runcode) # truncated the result print(truncate(result)) # print(result) @@ -266,6 +299,7 @@ class MLEngineer(Role): self.plan.add_tasks(tasks) self.working_memory.clear() + def get_useful_memories(self) -> List[Message]: """find useful memories only to reduce context length and improve performance""" # TODO dataset description , code steps @@ -298,11 +332,13 @@ if __name__ == "__main__": from metagpt.const import DATA_PATH - requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." + # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." data_path = f"{DATA_PATH}/titanic" + requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - async def main(requirement: str = requirement, auto_run: bool = True, data_path: str = data_path): + async def main(requirement: str = requirement, auto_run: bool = True, data_path: str = ""): role = MLEngineer(goal=requirement, auto_run=auto_run, data_path=data_path) await role.run(requirement) + fire.Fire(main) From 1265d3d924b0e1553591c6628d0c2de2a18d5722 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Fri, 8 Dec 2023 12:37:06 +0800 Subject: [PATCH 084/637] feat: make_tools by function. --- metagpt/actions/make_tools.py | 49 ++++++++++++++++++++++++ metagpt/provider/base_gpt_api.py | 2 +- tests/metagpt/actions/test_make_tools.py | 18 +++++++++ 3 files changed, 68 insertions(+), 1 deletion(-) create mode 100644 metagpt/actions/make_tools.py create mode 100644 tests/metagpt/actions/test_make_tools.py diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py new file mode 100644 index 000000000..7fd05751e --- /dev/null +++ b/metagpt/actions/make_tools.py @@ -0,0 +1,49 @@ +from typing import List, Dict +from pathlib import Path +import re + +from tenacity import retry, stop_after_attempt, wait_fixed + +from metagpt.logs import logger +from metagpt.schema import Message +from metagpt.actions.write_analysis_code import WriteCodeByGenerate + + +class MakeTools(WriteCodeByGenerate): + DEFAULT_SYSTEM_MSG = """Please Create a General Function Code startswith `def` from any codes you got.\n + **Notice:The import statement must be written after `def`, it is very important for you. + Reflect on whether it meets the requirements of function. Must Write example code, and we will execute the example code.** + """ + + def __init__(self, name: str = '', context=None, llm=None, workspace: str = None): + super().__init__(name, context, llm) + self.workspace = workspace or "." + self.file_suffix = '.py' + + def parse_function_name(self, function_code: str) -> str: + # 定义正则表达式模式 + pattern = r'\bdef\s+([a-zA-Z_]\w*)\s*\(' + # 在代码中搜索匹配的模式 + match = re.search(pattern, function_code) + # 如果找到匹配项,则返回匹配的函数名;否则返回None + if match: + return match.group(1) + else: + return None + + def save(self, tool_code: str) -> None: + func_name = self.parse_function_name(tool_code) + if func_name is None: + raise ValueError(f"No function name found in {tool_code}") + saved_path = Path(self.workspace).joinpath(func_name+self.file_suffix) + logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") + saved_path.write_text(tool_code, encoding='utf-8') + + @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) + async def run(self, code_message: List[Message | Dict], **kwargs) -> str: + msgs = self.process_msg(code_message) + logger.info(f"Ask: {msgs[-1]}") + tool_code = await self.llm.aask_code(msgs, **kwargs) + logger.info(f"Respond: Got {tool_code} from llm.") + self.save(tool_code['code']) + return tool_code["code"] diff --git a/metagpt/provider/base_gpt_api.py b/metagpt/provider/base_gpt_api.py index b6b034329..5516ceb7c 100644 --- a/metagpt/provider/base_gpt_api.py +++ b/metagpt/provider/base_gpt_api.py @@ -150,7 +150,7 @@ class BaseGPTAPI(BaseChatbot): :return dict: return the first function arguments of choice, for example, {'language': 'python', 'code': "print('Hello, World!')"} """ - return json.loads(self.get_choice_function(rsp)["arguments"]) + return json.loads(self.get_choice_function(rsp)["arguments"], strict=False) def messages_to_prompt(self, messages: list[dict]): """[{"role": "user", "content": msg}] to user: etc.""" diff --git a/tests/metagpt/actions/test_make_tools.py b/tests/metagpt/actions/test_make_tools.py new file mode 100644 index 000000000..2c5168bf1 --- /dev/null +++ b/tests/metagpt/actions/test_make_tools.py @@ -0,0 +1,18 @@ +import pytest + +from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.make_tools import MakeTools + + +@pytest.mark.asyncio +async def test_make_tools(): + code = "import yfinance as yf\n\n# Collect Alibaba stock data\nalibaba = yf.Ticker('BABA')\ndata = alibaba.history(period='1d', start='2022-01-01', end='2022-12-31')\nprint(data.head())" + msgs = [{'role': 'assistant', 'content': code}] + mt = MakeTools() + tool_code = await mt.run(msgs) + print(tool_code) + ep = ExecutePyCode() + tool_code = "!pip install yfinance\n" + tool_code + result, res_type = await ep.run(tool_code) + assert res_type is True + print(result) From ab020adec4c10e400410fb43c5dc7972e4cf0477 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 14:39:14 +0800 Subject: [PATCH 085/637] update: add refactor code for make_tools. --- metagpt/actions/make_tools.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index 7fd05751e..9da829e1f 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -11,8 +11,10 @@ from metagpt.actions.write_analysis_code import WriteCodeByGenerate class MakeTools(WriteCodeByGenerate): DEFAULT_SYSTEM_MSG = """Please Create a General Function Code startswith `def` from any codes you got.\n - **Notice:The import statement must be written after `def`, it is very important for you. - Reflect on whether it meets the requirements of function. Must Write example code, and we will execute the example code.** + **Notice:1. The import statement must be written after `def`, it is very important for you. + 2. Reflect on whether it meets the requirements of function. + 3. Refactor your code with the best performance when dealing with big data. + 4. Must Write example code, and it could be execute in the user machine.** """ def __init__(self, name: str = '', context=None, llm=None, workspace: str = None): From 402ec5bcb44528f9c2ce7505e75f30998cd87024 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 14:39:57 +0800 Subject: [PATCH 086/637] add new test for make tools. --- tests/metagpt/actions/test_make_tools.py | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/tests/metagpt/actions/test_make_tools.py b/tests/metagpt/actions/test_make_tools.py index 2c5168bf1..4f7d7859a 100644 --- a/tests/metagpt/actions/test_make_tools.py +++ b/tests/metagpt/actions/test_make_tools.py @@ -16,3 +16,19 @@ async def test_make_tools(): result, res_type = await ep.run(tool_code) assert res_type is True print(result) + + +@pytest.mark.asyncio +async def test_make_tools2(): + code = '''import pandas as pd\npath = "./tests/data/test.csv"\ndf = pd.read_csv(path)\ndata = df.copy()\n + data['started_at'] = data['started_at'].apply(lambda r: pd.to_datetime(r))\n + data['ended_at'] = data['ended_at'].apply(lambda r: pd.to_datetime(r))\ndata.head()''' + msgs = [{'role': 'assistant', 'content': code}] + mt = MakeTools() + tool_code = await mt.run(msgs) + print(tool_code) + ep = ExecutePyCode() + tool_code = tool_code + result, res_type = await ep.run(tool_code) + assert res_type is True + print(result) From d9342025cdd01730f87ede2f0f9e10aaedd7dda6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 14:40:16 +0800 Subject: [PATCH 087/637] update typing-extensions. --- requirements.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/requirements.txt b/requirements.txt index 1d1bc95a1..1ca309762 100644 --- a/requirements.txt +++ b/requirements.txt @@ -51,4 +51,4 @@ nbformat==5.9.2 ipython==8.17.2 ipykernel==6.27.0 scikit_learn==1.3.2 -typing-extensions==4.8.0 \ No newline at end of file +typing-extensions==4.9.0 \ No newline at end of file From 3ea4b3200bef5bbdc1b656b34093a74f03d4d334 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 15:00:28 +0800 Subject: [PATCH 088/637] update DEFAULT_SYSTEM_MSG. --- metagpt/actions/make_tools.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index 9da829e1f..0b5d09d8c 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -13,8 +13,8 @@ class MakeTools(WriteCodeByGenerate): DEFAULT_SYSTEM_MSG = """Please Create a General Function Code startswith `def` from any codes you got.\n **Notice:1. The import statement must be written after `def`, it is very important for you. 2. Reflect on whether it meets the requirements of function. - 3. Refactor your code with the best performance when dealing with big data. - 4. Must Write example code, and it could be execute in the user machine.** + 3. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. + 4. Write example code by using old varibales in old code, and make sure it could be execute in the user's machine.** """ def __init__(self, name: str = '', context=None, llm=None, workspace: str = None): From 65db6683e6069501a669e73c1eaad3bae7566a09 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 15:18:26 +0800 Subject: [PATCH 089/637] add new test instance. --- tests/metagpt/actions/test_make_tools.py | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/tests/metagpt/actions/test_make_tools.py b/tests/metagpt/actions/test_make_tools.py index 4f7d7859a..7811cf7ab 100644 --- a/tests/metagpt/actions/test_make_tools.py +++ b/tests/metagpt/actions/test_make_tools.py @@ -32,3 +32,20 @@ async def test_make_tools2(): result, res_type = await ep.run(tool_code) assert res_type is True print(result) + + +@pytest.mark.asyncio +async def test_make_tools3(): + code = '''import pandas as pd\npath = "./tests/data/test.csv"\ndf = pd.read_csv(path)\ndata = df.copy()\n + data['started_at'] = data['started_at'].apply(lambda r: pd.to_datetime(r))\n + data['ended_at'] = data['ended_at'].apply(lambda r: pd.to_datetime(r))\n + data['duration_hour'] = (data['ended_at'] - data['started_at']).dt.seconds/3600\ndata.head()''' + msgs = [{'role': 'assistant', 'content': code}] + mt = MakeTools() + tool_code = await mt.run(msgs) + print(tool_code) + ep = ExecutePyCode() + tool_code = tool_code + result, res_type = await ep.run(tool_code) + assert res_type is True + print(result) From 5a01fdb0e2b00f597a702b45ff818977fd9dba9f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 15:18:59 +0800 Subject: [PATCH 090/637] update DEFAULT_SYSTEM_MSG. --- metagpt/actions/make_tools.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index 0b5d09d8c..9ab7fd922 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -10,9 +10,9 @@ from metagpt.actions.write_analysis_code import WriteCodeByGenerate class MakeTools(WriteCodeByGenerate): - DEFAULT_SYSTEM_MSG = """Please Create a General Function Code startswith `def` from any codes you got.\n + DEFAULT_SYSTEM_MSG = """Please Create a very General Function Code startswith `def` from any codes you got.\n **Notice:1. The import statement must be written after `def`, it is very important for you. - 2. Reflect on whether it meets the requirements of function. + 2. Reflect on whether it meets the requirements of a general function. 3. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. 4. Write example code by using old varibales in old code, and make sure it could be execute in the user's machine.** """ From 76c95f8428ac08b5bd1a12f4e742c108fbae08eb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 16:00:09 +0800 Subject: [PATCH 091/637] chore: add logger.debug(). --- tests/metagpt/actions/test_make_tools.py | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/tests/metagpt/actions/test_make_tools.py b/tests/metagpt/actions/test_make_tools.py index 7811cf7ab..264599439 100644 --- a/tests/metagpt/actions/test_make_tools.py +++ b/tests/metagpt/actions/test_make_tools.py @@ -2,6 +2,7 @@ import pytest from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.make_tools import MakeTools +from metagpt.logs import logger @pytest.mark.asyncio @@ -10,12 +11,12 @@ async def test_make_tools(): msgs = [{'role': 'assistant', 'content': code}] mt = MakeTools() tool_code = await mt.run(msgs) - print(tool_code) + logger.debug(tool_code) ep = ExecutePyCode() tool_code = "!pip install yfinance\n" + tool_code result, res_type = await ep.run(tool_code) assert res_type is True - print(result) + logger.debug(result) @pytest.mark.asyncio @@ -26,12 +27,12 @@ async def test_make_tools2(): msgs = [{'role': 'assistant', 'content': code}] mt = MakeTools() tool_code = await mt.run(msgs) - print(tool_code) + logger.debug(tool_code) ep = ExecutePyCode() tool_code = tool_code result, res_type = await ep.run(tool_code) assert res_type is True - print(result) + logger.debug(result) @pytest.mark.asyncio @@ -43,9 +44,9 @@ async def test_make_tools3(): msgs = [{'role': 'assistant', 'content': code}] mt = MakeTools() tool_code = await mt.run(msgs) - print(tool_code) + logger.debug(tool_code) ep = ExecutePyCode() tool_code = tool_code result, res_type = await ep.run(tool_code) assert res_type is True - print(result) + logger.debug(result) From c2f0e547ee2db3332fdb4408ef8f2c179243735d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 16:03:22 +0800 Subject: [PATCH 092/637] =?UTF-8?q?chore:=20=E5=B1=9E=E6=80=A7=E6=B3=A8?= =?UTF-8?q?=E9=87=8A=EF=BC=8C=E4=BB=A5=E5=8F=8A=E5=85=A5=E5=8F=82=E7=9A=84?= =?UTF-8?q?=E6=95=B0=E6=8D=AE=E7=B1=BB=E5=9E=8B=E5=AE=9A=E4=B9=89?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/actions/make_tools.py | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index 9ab7fd922..f7e385138 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -4,6 +4,7 @@ import re from tenacity import retry, stop_after_attempt, wait_fixed +from metagpt.llm import LLM from metagpt.logs import logger from metagpt.schema import Message from metagpt.actions.write_analysis_code import WriteCodeByGenerate @@ -17,10 +18,10 @@ class MakeTools(WriteCodeByGenerate): 4. Write example code by using old varibales in old code, and make sure it could be execute in the user's machine.** """ - def __init__(self, name: str = '', context=None, llm=None, workspace: str = None): + def __init__(self, name: str = '', context: list[Message] = None, llm: LLM = None, workspace: str = None): super().__init__(name, context, llm) self.workspace = workspace or "." - self.file_suffix = '.py' + self.file_suffix: str = '.py' def parse_function_name(self, function_code: str) -> str: # 定义正则表达式模式 From 4b58942159342c74c053a235b473e578f3147dbc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 16:05:50 +0800 Subject: [PATCH 093/637] =?UTF-8?q?chore:=20=E5=B1=9E=E6=80=A7=E6=B3=A8?= =?UTF-8?q?=E9=87=8A=EF=BC=8C=E4=BB=A5=E5=8F=8A=E5=85=A5=E5=8F=82=E7=9A=84?= =?UTF-8?q?=E6=95=B0=E6=8D=AE=E7=B1=BB=E5=9E=8B=E5=AE=9A=E4=B9=89.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/actions/make_tools.py | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index f7e385138..aa2ebe501 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -19,6 +19,12 @@ class MakeTools(WriteCodeByGenerate): """ def __init__(self, name: str = '', context: list[Message] = None, llm: LLM = None, workspace: str = None): + """ + :param str name: name, defaults to '' + :param list[Message] context: context, defaults to None + :param LLM llm: llm, defaults to None + :param str workspace: tools code saved file path dir, defaults to None + """ super().__init__(name, context, llm) self.workspace = workspace or "." self.file_suffix: str = '.py' From 4231e0a11e7775d22c35ec9f8f4dfc1a233cb925 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 11 Dec 2023 16:13:34 +0800 Subject: [PATCH 094/637] kaggle iterative trial done --- kaggle_team.py | 3 +- metagpt/actions/execute_code.py | 28 ++++++++++++++-- metagpt/actions/ml_da_action.py | 17 +++++----- metagpt/actions/write_plan.py | 38 ++++++++++++++++++---- metagpt/roles/kaggle_manager.py | 3 +- metagpt/roles/ml_engineer.py | 34 ++++++++++++++------ metagpt/schema.py | 39 +++++++++++++++++----- tests/metagpt/actions/test_write_plan.py | 20 ++++++------ tests/metagpt/test_schema.py | 41 ++++++++++++++++++++++++ 9 files changed, 178 insertions(+), 45 deletions(-) diff --git a/kaggle_team.py b/kaggle_team.py index e8ab3ec41..50a8f7288 100644 --- a/kaggle_team.py +++ b/kaggle_team.py @@ -19,8 +19,9 @@ async def main( competition, data_desc, requirement = ( "titanic", "Training set is train.csv.\nTest set is test.csv. We also include gender_submission.csv, a set of predictions that assume all and only female passengers survive, as an example of what a submission file should look like.", - "Run EDA on the train dataset, train a model to predict survival (20% as validation) and save it, predict the test set using saved model, save the test result according to format", + # "Run EDA on the train dataset, train a model to predict survival (20% as validation) and save it, predict the test set using saved model, save the test result according to format", # "generate a random prediction, replace the Survived column of gender_submission.csv, and save the prediction to a new submission file", + "Score as high as possible for the provided dataset, save the test prediction to a csv with two columns PassengerId and Survived" ) team = Team() diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 981aa894c..9c2b8d96c 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -8,6 +8,7 @@ from abc import ABC, abstractmethod from pathlib import Path from typing import Dict, List, Tuple, Union import traceback +import re import nbformat from nbclient import NotebookClient @@ -171,11 +172,34 @@ class ExecutePyCode(ExecuteCode, Action): # TODO: add max_tries for run code. cell_index = len(self.nb.cells) - 1 await self.nb_client.async_execute_cell(self.nb.cells[-1], cell_index) - return self.parse_outputs(self.nb.cells[-1].outputs), True + outputs = self.parse_outputs(self.nb.cells[-1].outputs) + success = True except Exception as e: # FIXME: CellExecutionError is hard to read. for example `1\0` raise ZeroDivisionError: # CellExecutionError('An error occurred while executing the following cell:\n------------------\nz=1/0\n------------------\n\n\n\x1b[0;31m---------------------------------------------------------------------------\x1b[0m\n\x1b[0;31mZeroDivisionError\x1b[0m Traceback (most recent call last)\nCell \x1b[0;32mIn[1], line 1\x1b[0m\n\x1b[0;32m----> 1\x1b[0m z\x1b[38;5;241m=\x1b[39m\x1b[38;5;241;43m1\x1b[39;49m\x1b[38;5;241;43m/\x1b[39;49m\x1b[38;5;241;43m0\x1b[39;49m\n\n\x1b[0;31mZeroDivisionError\x1b[0m: division by zero\n') - return traceback.format_exc(), False + outputs = traceback.format_exc() + success = False + return truncate(remove_escape_and_color_codes(outputs)), success else: # TODO: markdown raise NotImplementedError(f"Not support this code type : {language}, Only support code!") + + +def truncate(result: str, keep_len: int = 2000) -> str: + desc = f"Truncated to show only the last {keep_len} characters\n" + if result.startswith(desc): + result = result[-len(desc) :] + + if len(result) > keep_len: + result = result[-keep_len:] + + if not result.startswith(desc): + return desc + result + return desc + + +def remove_escape_and_color_codes(input_str): + # 使用正则表达式去除转义字符和颜色代码 + pattern = re.compile(r'\x1b\[[0-9;]*[mK]') + result = pattern.sub('', input_str) + return result diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py index a4537dad9..6be4b3040 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_da_action.py @@ -7,8 +7,8 @@ from metagpt.utils.common import CodeParser from metagpt.logs import logger -def truncate(result: str, keep_len: int = 1000) -> str: - desc = "Truncated to show only the last 1000 characters\n" +def truncate(result: str, keep_len: int = 2000) -> str: + desc = "Truncated to show only the last keep_len characters\n" if result.startswith(desc): result = result[-len(desc) :] @@ -70,7 +70,9 @@ class AskReview(Action): if rsp.lower() in ReviewConst.EXIT_WORD: exit() - confirmed = rsp.lower() in ReviewConst.CONTINUE_WORD + # Confirmation can be one of "confirm", "continue", "c", "yes", "y" exactly, or sentences containing "confirm". + # One could say "confirm this task, but change the next task to ..." + confirmed = rsp.lower() in ReviewConst.CONTINUE_WORD or ReviewConst.CONTINUE_WORD[0] in rsp.lower() return rsp, confirmed @@ -109,13 +111,13 @@ class Reflect(Action): ```json { "summary": str = "summarize each of your previous trial in a triple of (your methods, the corresponding result, potential improvement), list them out", - "takeaways": str = "carefully find key takeaways from your summarization in a step-by-step thinking process", - "reflection": "in one sentence, state executable actions for improving your future plan", + "takeaways": str = "carefully find key takeaways from your summarization", + "reflection": str = "give specific instruction to improve your next trial in a step-by-step thinking process", } ``` """ - REWRITE_PLAN_INSTRUCTION = """When taking this reflection for rewriting plan, modify the current plan in place, replace, add, or delete tasks in the plan, - only make necessary change to the current plan, keep reusable tasks unchanged, provide the complete new plan.""" + REWRITE_PLAN_INSTRUCTION = """Take this reflection for rewriting plan, modify the current plan in place, make reference to your specific instruction, think about you should + change which task, add or delete what tasks in the plan. Only make necessary changes, keep reusable tasks unchanged, output the COMPLETE new plan starting from the first task. Your plan should have no more than 5 tasks.""" async def run(self, context: str, user_requirement: str = "") -> str: user_requirement = user_requirement or "Score as high as possible in a data modeling competition" @@ -124,5 +126,4 @@ class Reflect(Action): rsp_json = await self._aask(prompt) rsp = CodeParser.parse_code(block=None, text=rsp_json) reflection = json.loads(rsp)["reflection"] - reflection += self.REWRITE_PLAN_INSTRUCTION return reflection diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index 71133bb4d..f7ca1ff4c 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -4,12 +4,14 @@ @Author : orange-crow @File : plan.py """ -from typing import List, Dict +from typing import List, Dict, Tuple import json +from copy import deepcopy +import traceback from metagpt.actions import Action from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_PROMPT, ASSIGN_TASK_TYPE -from metagpt.schema import Message, Task +from metagpt.schema import Message, Task, Plan from metagpt.utils.common import CodeParser, create_func_config @@ -67,8 +69,30 @@ class WritePlan(Action): rsp = await self.assign_task_type(json.loads(rsp)) return rsp - @staticmethod - def rsp_to_tasks(rsp: str) -> List[Task]: - rsp = json.loads(rsp) - tasks = [Task(**task_config) for task_config in rsp] - return tasks +def rsp_to_tasks(rsp: str) -> List[Task]: + rsp = json.loads(rsp) + tasks = [Task(**task_config) for task_config in rsp] + return tasks + +def update_plan_from_rsp(rsp: str, current_plan: Plan): + tasks = rsp_to_tasks(rsp) + if len(tasks) == 1: + # handle a single task + if current_plan.has_task_id(tasks[0].task_id): + # replace an existing task + current_plan.replace_task(tasks[0]) + else: + # append one task + current_plan.append_task(tasks[0]) + + else: + # add tasks in general + current_plan.add_tasks(tasks) + +def precheck_update_plan_from_rsp(rsp: str, current_plan: Plan) -> Tuple[bool, str]: + temp_plan = deepcopy(current_plan) + try: + update_plan_from_rsp(rsp, temp_plan) + return True, "" + except Exception as e: + return False, e diff --git a/metagpt/roles/kaggle_manager.py b/metagpt/roles/kaggle_manager.py index 354289975..18ac6733a 100644 --- a/metagpt/roles/kaggle_manager.py +++ b/metagpt/roles/kaggle_manager.py @@ -1,6 +1,7 @@ from typing import Dict, List, Union, Tuple import json import subprocess +import os import fire import pandas as pd @@ -14,7 +15,7 @@ from metagpt.schema import Message, Task, Plan from metagpt.logs import logger from metagpt.utils.common import CodeParser -import os + os.environ["KAGGLE_USERNAME"] = CONFIG.kaggle_username os.environ["KAGGLE_KEY"] = CONFIG.kaggle_key diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 4e818ca3c..6e7331281 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -10,7 +10,7 @@ from metagpt.actions import Action from metagpt.schema import Message, Task, Plan from metagpt.memory import Memory from metagpt.logs import logger -from metagpt.actions.write_plan import WritePlan +from metagpt.actions.write_plan import WritePlan, update_plan_from_rsp, precheck_update_plan_from_rsp from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis, Reflect, ReviewConst from metagpt.actions.execute_code import ExecutePyCode @@ -69,13 +69,24 @@ class MLEngineer(Role): # ask for acceptance, users can other refuse and change tasks in the plan review, task_result_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - if success and task_result_confirmed: + if task_result_confirmed: # tick off this task and record progress task.code = code task.result = result self.plan.finish_current_task() self.working_memory.clear() + confirmed_and_more = (ReviewConst.CONTINUE_WORD[0] in review.lower() + and review.lower() not in ReviewConst.CONTINUE_WORD[0]) # "confirm, ... (more content, such as changing downstream tasks)" + if confirmed_and_more: + self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) + await self._update_plan(review) + + elif "redo" in review: + # Ask the Role to redo this task with help of review feedback, + # useful when the code run is successful but the procedure or result is not what we want + continue + else: # update plan according to user's feedback and to take on changed tasks await self._update_plan(review) @@ -151,7 +162,7 @@ class MLEngineer(Role): return review, confirmed return "", True - async def _update_plan(self, review: str = "", max_tasks: int = 3): + async def _update_plan(self, review: str = "", max_tasks: int = 3, max_retries: int = 3): plan_confirmed = False while not plan_confirmed: context = self.get_useful_memories() @@ -162,15 +173,19 @@ class MLEngineer(Role): Message(content=rsp, role="assistant", cause_by=WritePlan) ) - # TODO: precheck plan before asking reviews + # precheck plan before asking reviews + is_plan_valid, error = precheck_update_plan_from_rsp(rsp, self.plan) + if not is_plan_valid and max_retries > 0: + error_msg = f"The generated plan is not valid with error: {error}, try regenerating, remember to generate either the whole plan or the single changed task only" + logger.warning(error_msg) + self.working_memory.add(Message(content=error_msg, role="assistant", cause_by=WritePlan)) + max_retries -= 1 + continue _, plan_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - tasks = WritePlan.rsp_to_tasks(rsp) - if len(tasks) == 1 and self.plan.has_task_id(tasks[0].task_id): - self.plan.replace_task(tasks[0]) - else: - self.plan.add_tasks(tasks) + update_plan_from_rsp(rsp, self.plan) + self.working_memory.clear() async def _reflect(self): @@ -181,6 +196,7 @@ class MLEngineer(Role): # print("*" * 10) reflection = await Reflect().run(context=context) self.working_memory.add(Message(content=reflection, role="assistant")) + self.working_memory.add(Message(content=Reflect.REWRITE_PLAN_INSTRUCTION, role="user")) def get_useful_memories(self) -> List[Message]: """find useful memories only to reduce context length and improve performance""" diff --git a/metagpt/schema.py b/metagpt/schema.py index 9b86a2448..4e5e083ec 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -149,10 +149,7 @@ class Plan(BaseModel): self.tasks = final_tasks # Update current_task_id to the first unfinished task in the merged list - for task in self.tasks: - if not task.is_finished: - self.current_task_id = task.task_id - break + self._update_current_task() # Update the task map for quick access to tasks by ID self.task_map = {task.task_id: task for task in self.tasks} @@ -196,8 +193,36 @@ class Plan(BaseModel): if new_task.task_id in task.dependent_task_ids: self.reset_task(task.task_id) + def append_task(self, new_task: Task): + """ + Append a new task to the end of existing task sequences + + Args: + new_task (Task): The new task to be appended to the existing task sequence + + Returns: + None + """ + assert not self.has_task_id(new_task.task_id), "Task already in current plan, use replace_task instead" + + assert all([self.has_task_id(dep_id) for dep_id in new_task.dependent_task_ids]), \ + "New task has unknown dependencies" + + # Existing tasks do not depend on the new task, it's fine to put it to the end of the sorted task sequence + self.tasks.append(new_task) + self.task_map[new_task.task_id] = new_task + self._update_current_task() + def has_task_id(self, task_id: str) -> bool: return task_id in self.task_map + + def _update_current_task(self): + current_task_id = "" + for task in self.tasks: + if not task.is_finished: + current_task_id = task.task_id + break + self.current_task_id = current_task_id # all tasks finished @property def current_task(self) -> Task: @@ -212,10 +237,8 @@ class Plan(BaseModel): """Finish current task, set Task.is_finished=True, set current task to next task """ if self.current_task_id: - current_task = self.current_task - current_task.is_finished = True - next_task_index = self.tasks.index(current_task) + 1 - self.current_task_id = self.tasks[next_task_index].task_id if next_task_index < len(self.tasks) else None + self.current_task.is_finished = True + self._update_current_task() # set to next task def get_finished_tasks(self) -> list[Task]: """return all finished tasks in correct linearized order diff --git a/tests/metagpt/actions/test_write_plan.py b/tests/metagpt/actions/test_write_plan.py index 2bf200ab3..7766e0d51 100644 --- a/tests/metagpt/actions/test_write_plan.py +++ b/tests/metagpt/actions/test_write_plan.py @@ -1,13 +1,15 @@ import pytest -from metagpt.actions.write_plan import WritePlan +from metagpt.actions.write_plan import WritePlan, precheck_update_plan_from_rsp, Plan, Task +def test_precheck_update_plan_from_rsp(): + plan = Plan(goal="") + plan.add_tasks([Task(task_id="1")]) + rsp = '[{"task_id": "2"}]' + success, _ = precheck_update_plan_from_rsp(rsp, plan) + assert success + assert len(plan.tasks) == 1 and plan.tasks[0].task_id == "1" # precheck should not change the original one -@pytest.mark.asyncio -async def test_plan(): - p = WritePlan() - task_desc = """Here’s some background information on Cyclistic, a bike-sharing company designing a marketing strategy aimed at converting casual riders into annual members: So far, Cyclistic’s marketing strategy has relied on building general awareness and engaging a wide range of consumers. group. One way to help achieve these goals is the flexibility of its pricing plans: one-way passes, full-day passes, and annual memberships. Customers who purchase a one-way or full-day pass are known as recreational riders. Customers purchasing an annual membership are Cyclistic members. I will provide you with a data sheet that records user behavior: '/Users/vicis/Downloads/202103-divvy-tripdata.csv""" - rsp = await p.run(task_desc, role="data analyst") - assert len(rsp.content) > 0 - assert rsp.sent_from == "WritePlan" - print(rsp) + invalid_rsp = 'wrong' + success, _ = precheck_update_plan_from_rsp(invalid_rsp, plan) + assert not success diff --git a/tests/metagpt/test_schema.py b/tests/metagpt/test_schema.py index 324a083ca..b5d49b7a1 100644 --- a/tests/metagpt/test_schema.py +++ b/tests/metagpt/test_schema.py @@ -5,6 +5,7 @@ @Author : alexanderwu @File : test_schema.py """ +import pytest from metagpt.schema import AIMessage, Message, SystemMessage, UserMessage from metagpt.schema import Task, Plan @@ -143,3 +144,43 @@ class TestPlan: plan.replace_task(new_task) # Task with ID 2 does not exist in plan assert "1" in plan.task_map assert "2" not in plan.task_map + + def test_append_task_with_valid_dependencies(self): + plan = Plan(goal="Test") + existing_task = [Task(task_id="1")] + plan.add_tasks(existing_task) + new_task = Task(task_id="2", dependent_task_ids=["1"]) + plan.append_task(new_task) + assert plan.tasks[-1].task_id == "2" + assert plan.task_map["2"] == new_task + + def test_append_task_with_invalid_dependencies(self): + new_task = Task(task_id="2", dependent_task_ids=["3"]) + plan = Plan(goal="Test") + with pytest.raises(AssertionError): + plan.append_task(new_task) + + def test_append_task_without_dependencies(self): + plan = Plan(goal="Test") + existing_task = [Task(task_id="1")] + plan.add_tasks(existing_task) + + new_task = Task(task_id="2") + plan.append_task(new_task) + + assert len(plan.tasks) == 2 + assert plan.current_task_id == "1" + + def test_append_task_updates_current_task(self): + finished_task = Task(task_id="1", is_finished=True) + new_task = Task(task_id="2") + plan = Plan(goal="Test", tasks=[finished_task]) + plan.append_task(new_task) + assert plan.current_task_id == "2" + + def test_update_current_task(self): + task1 = Task(task_id="1", is_finished=True) + task2 = Task(task_id="2") + plan = Plan(goal="Test", tasks=[task1, task2]) + plan._update_current_task() + assert plan.current_task_id == "2" From 1b4aac394d1a5095224a735a83e3034d447231c5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 19:28:57 +0800 Subject: [PATCH 095/637] chore: update DEFAULT_SYSTEM_MSG and self.workspace. --- metagpt/actions/make_tools.py | 10 ++++++---- metagpt/tools/functions/libs/udf/__init__.py | 0 2 files changed, 6 insertions(+), 4 deletions(-) create mode 100644 metagpt/tools/functions/libs/udf/__init__.py diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index aa2ebe501..7cad8ef7b 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -12,9 +12,10 @@ from metagpt.actions.write_analysis_code import WriteCodeByGenerate class MakeTools(WriteCodeByGenerate): DEFAULT_SYSTEM_MSG = """Please Create a very General Function Code startswith `def` from any codes you got.\n - **Notice:1. The import statement must be written after `def`, it is very important for you. - 2. Reflect on whether it meets the requirements of a general function. - 3. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. + **Notice: + 1. Reflect on whether it meets the requirements of a general function. + 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. + 3. Use Google style for function annotations. 4. Write example code by using old varibales in old code, and make sure it could be execute in the user's machine.** """ @@ -26,7 +27,7 @@ class MakeTools(WriteCodeByGenerate): :param str workspace: tools code saved file path dir, defaults to None """ super().__init__(name, context, llm) - self.workspace = workspace or "." + self.workspace = workspace or str(Path(__file__).parents[1].joinpath("./tools/functions/libs/udf")) self.file_suffix: str = '.py' def parse_function_name(self, function_code: str) -> str: @@ -47,6 +48,7 @@ class MakeTools(WriteCodeByGenerate): saved_path = Path(self.workspace).joinpath(func_name+self.file_suffix) logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") saved_path.write_text(tool_code, encoding='utf-8') + # TODO: 保存到udf中,供WriteCodeWithMakeTools使用 @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) async def run(self, code_message: List[Message | Dict], **kwargs) -> str: diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py new file mode 100644 index 000000000..e69de29bb From 51bf8863af9414069d7de54d780fc5f1d83bf51a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 19:47:23 +0800 Subject: [PATCH 096/637] add udf. --- .gitignore | 1 + 1 file changed, 1 insertion(+) diff --git a/.gitignore b/.gitignore index e03eab3d3..1a517d027 100644 --- a/.gitignore +++ b/.gitignore @@ -129,6 +129,7 @@ venv.bak/ .mypy_cache/ .dmypy.json dmypy.json +metagpt/tools/functions/libs/udf/*.py # Pyre type checker .pyre/ From 2b8dbec5d044c7e5c67a6cb4b3146e69a632bab8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 21:01:53 +0800 Subject: [PATCH 097/637] update DEFAULT_SYSTEM_MSG. --- metagpt/actions/make_tools.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index 7cad8ef7b..2b2ba1cd5 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -16,7 +16,8 @@ class MakeTools(WriteCodeByGenerate): 1. Reflect on whether it meets the requirements of a general function. 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. 3. Use Google style for function annotations. - 4. Write example code by using old varibales in old code, and make sure it could be execute in the user's machine.** + 4. Write example code after `if __name__ == '__main__':`by using old varibales in old code, + and make sure it could be execute in the user's machine.** """ def __init__(self, name: str = '', context: list[Message] = None, llm: LLM = None, workspace: str = None): From 3de10e76562c71d884c8c2a3dd93a1180eae15b0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 11 Dec 2023 21:12:36 +0800 Subject: [PATCH 098/637] add UDFS for make tools. --- metagpt/tools/functions/libs/udf/__init__.py | 50 ++++++++++++++++++++ tests/metagpt/tools/functions/test_udf.py | 9 ++++ 2 files changed, 59 insertions(+) create mode 100644 tests/metagpt/tools/functions/test_udf.py diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index e69de29bb..0bdf84d87 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -0,0 +1,50 @@ +import ast +import os +import inspect +import importlib +from pathlib import Path + + +def extract_function_signatures(file_path): + with open(file_path, 'r', encoding='utf-8') as file: + source_code = file.read() + + tree = ast.parse(source_code) + function_signatures = [] + for node in ast.walk(tree): + if isinstance(node, ast.FunctionDef): + # 只提取用户自定义函数,排除内置函数 + if not (node.name.startswith('__') and node.name.endswith('__')): + # 获取函数名 + function_name = node.name + # 获取参数列表 + args = [arg.arg for arg in node.args.args] + # 获取函数签名 + function_signature = f"{function_name}({', '.join(args)})" + # 导入函数 + module = Path(file_path).parts[-1][:-len(Path(file_path).suffix)] + module = importlib.import_module(f"metagpt.tools.functions.libs.udf.{module}") + # 获取函数注释 + function_schema = {'name': function_signature, 'doc': inspect.getdoc(getattr(module, function_name))} + function_signatures.append(function_schema) + + return function_signatures + + +def get_function_signatures_in_folder(folder_path): + python_files = [f for f in os.listdir(folder_path) if f.endswith('.py')] + all_function_signatures = [] + + for file_name in python_files: + file_path = os.path.join(folder_path, file_name) + function_signatures = extract_function_signatures(file_path) + all_function_signatures.extend(function_signatures) + + return all_function_signatures + + +folder_path = str(Path(__file__).parent.absolute()) +function_signatures = get_function_signatures_in_folder(folder_path) + +UDFS = [func for func in function_signatures + if not func['name'].startswith(('extract_function_signatures', 'get_function_signatures_in_folder'))] diff --git a/tests/metagpt/tools/functions/test_udf.py b/tests/metagpt/tools/functions/test_udf.py new file mode 100644 index 000000000..b0c921180 --- /dev/null +++ b/tests/metagpt/tools/functions/test_udf.py @@ -0,0 +1,9 @@ +from metagpt.tools.functions.libs.udf import UDFS +from metagpt.logs import logger + + +def test_udfs(): + assert len(UDFS) > 0 + assert 'name' in UDFS[0] + assert 'doc' in UDFS[0] + logger.info(UDFS) From ee1e8609a6523995ca002e4a6c4b1ea792cda1ca Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 12 Dec 2023 10:08:15 +0800 Subject: [PATCH 099/637] add function path for function_signatures. --- metagpt/tools/functions/libs/udf/__init__.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index 0bdf84d87..c90357b5c 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -25,7 +25,8 @@ def extract_function_signatures(file_path): module = Path(file_path).parts[-1][:-len(Path(file_path).suffix)] module = importlib.import_module(f"metagpt.tools.functions.libs.udf.{module}") # 获取函数注释 - function_schema = {'name': function_signature, 'doc': inspect.getdoc(getattr(module, function_name))} + function_schema = {'name': function_signature, 'doc': inspect.getdoc(getattr(module, function_name)), + 'path': f'from metagpt.tools.functions.libs.udf.{module} import function_name'} function_signatures.append(function_schema) return function_signatures From c3a06ad20365a89ce3209f33fa33c9ae7e98af67 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Tue, 12 Dec 2023 10:10:07 +0800 Subject: [PATCH 100/637] =?UTF-8?q?=E6=9B=B4=E6=96=B0reflection=EF=BC=8C?= =?UTF-8?q?=E5=88=86=E5=BC=80=E5=8E=86=E5=8F=B2code=E5=92=8C=E5=B7=B2?= =?UTF-8?q?=E6=9C=89=E8=BF=90=E8=A1=8C=E7=BB=93=E6=9E=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/actions/debug_code.py | 39 ++++++++++++++++++++++++----------- 1 file changed, 27 insertions(+), 12 deletions(-) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 3d460fa40..9efe93efc 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -41,6 +41,12 @@ REFLECTION_PROMPT = """ {debug_example} [requirement] {goal} + [finished code] + finished code are executable, and you should based on the code to continue your current code debug + {finished_code} + + try to reuse the code here to understand the coding task. + [previous impl] {code} [runtime Error] @@ -65,47 +71,56 @@ class DebugCode(BaseWriteAnalysisCode): name: str = "debugcode" context: Optional[str] = None llm: None - + def __init__(self, **kwargs: Any): super().__init__(**kwargs) - - async def run_reflection(self, plan, code, runtime_result) -> str: + + async def run_reflection(self, goal, finished_code, finished_code_result, code, runtime_result) -> str: info = [] + finished_code_and_result = finished_code + "\n [finished results]\n\n" + finished_code_result reflection_prompt = REFLECTION_PROMPT.format(debug_example=DEBUG_REFLECTION_EXAMPLE, - goal=plan.goal, + goal=goal, + finished_code=finished_code_and_result, code=code, runtime_result=runtime_result ) system_prompt = "You are an AI Python assistant. You will be given your previous implementation of a function, runtime error results, and a hint to change the implementation appropriately. Write your full implementation " info.append(Message(role="system", content=system_prompt)) info.append(Message(role="assistant", content=reflection_prompt)) - + msg = messages_to_str(info) resp = await self.llm.aask(msg=msg) logger.info(f"reflection is {resp}") return resp - - async def rewrite_code(self, reflection: str = "") -> str: + + async def rewrite_code(self, reflection: str = "", code_context: str = "") -> str: """ 根据reflection重写代码 """ info = [] - info.append(Message(role="assistant", content=f"[reflection]: \n {reflection}")) + info.append(Message(role="assistant", content=f"[code context]:{code_context}" + f"finished code are executable, and you should based on the code to continue your current code debug and improvement" + f"[reflection]: \n {reflection}")) info.append(Message(role="user", content=f"[improved impl]:\n Return in Python block")) msg = messages_to_str(info) resp = await self.llm.aask(msg=msg) logger.info(f"improve code is {resp}") improv_code = CodeParser.parse_code(block=None, text=resp) return improv_code - + async def run(self, - plan: Plan = None, + plan: str = "", + finished_code: str = "", + finished_code_result: str = "", code: str = "", runtime_result: str = "") -> str: """ 根据当前运行代码和报错信息进行reflection和纠错 """ - reflection = await self.run_reflection(plan, code, runtime_result) + reflection = await self.run_reflection(plan, finished_code=finished_code, + finished_code_result=finished_code_result, + code=code, + runtime_result=runtime_result) # 根据reflection结果重写代码 - improv_code = await self.rewrite_code(reflection) + improv_code = await self.rewrite_code(reflection, code_context=finished_code) return improv_code From 4f1aa0333ec9cae6bf69c711735794c8c6677693 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Tue, 12 Dec 2023 10:10:19 +0800 Subject: [PATCH 101/637] =?UTF-8?q?=E5=A2=9E=E5=8A=A0retry?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/provider/openai_api.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 34e5693f8..d8d2e9a4f 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -15,6 +15,7 @@ from tenacity import ( retry, retry_if_exception_type, stop_after_attempt, + wait_random_exponential, wait_fixed, ) @@ -259,7 +260,8 @@ class OpenAIGPTAPI(BaseGPTAPI, RateLimiter): rsp = self.llm.ChatCompletion.create(**self._func_configs(messages, **kwargs)) self._update_costs(rsp.get("usage")) return rsp - + + @retry(wait=wait_random_exponential(min=1, max=60), stop=stop_after_attempt(6)) async def _achat_completion_function(self, messages: list[dict], **chat_configs) -> dict: rsp = await self.llm.ChatCompletion.acreate(**self._func_configs(messages, **chat_configs)) self._update_costs(rsp.get("usage")) From 4634415e378dc3b02659721ff97cd9b852d53cd4 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Tue, 12 Dec 2023 10:15:21 +0800 Subject: [PATCH 102/637] =?UTF-8?q?=E5=8F=AA=E4=BD=BF=E7=94=A8=E5=BD=93?= =?UTF-8?q?=E5=89=8Dcode=E8=BF=90=E8=A1=8C=EF=BC=8C=E4=B8=8D=E8=BF=AD?= =?UTF-8?q?=E4=BB=A3=E5=8E=86=E5=8F=B2code=20=E5=88=86=E5=BC=80=E5=BD=93?= =?UTF-8?q?=E5=89=8Dcode=E5=92=8C=E5=8E=86=E5=8F=B2=E4=BB=A3=E7=A0=81?= =?UTF-8?q?=E7=BB=99reflection?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/roles/ml_engineer.py | 114 +++++++++++++++++++---------------- 1 file changed, 62 insertions(+), 52 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 1b191c8ba..45fe728dd 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -37,17 +37,14 @@ catboost """ - - - def truncate(result: str, keep_len: int = 1000) -> str: desc = "Truncated to show only the last 1000 characters\n" if result.startswith(desc): result = result[-len(desc):] - + if len(result) > keep_len: result = result[-keep_len:] - + if not result.startswith(desc): return desc + result return desc @@ -80,7 +77,7 @@ def get_column_info(df: pd.DataFrame) -> str: nan_freq = float("%.2g" % (df[i].isna().mean() * 100)) n_unique = df[i].nunique() data.append([i, df[i].dtype, nan_freq, n_unique]) - + samples = pd.DataFrame( data, columns=["Column_name", "Data_type", "NaN_Frequency(%)", "N_unique"], @@ -94,7 +91,7 @@ class AskReview(Action): logger.info( "\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks]) ) - + logger.info("most recent context:") latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" prompt = f"\nPlease review output from {latest_action}:\n" \ @@ -102,12 +99,12 @@ class AskReview(Action): "If you confirm the output and wish to continue with the current process, type CONFIRM\n" \ "If you want to terminate the process, type exit:\n" rsp = input(prompt) - + if rsp.lower() in ("exit"): exit() - + confirmed = rsp.lower() in ("confirm", "yes", "y") - + return rsp, confirmed @@ -141,24 +138,24 @@ class MLEngineer(Role): self.auto_run = auto_run self.data_path = data_path self.data_desc = {} - + async def _plan_and_act(self): if self.data_path: self.data_desc = await self._generate_data_desc() - + # create initial plan and update until confirmation await self._update_plan() - + while self.plan.current_task: task = self.plan.current_task logger.info(f"ready to take on task {task}") - + # take on current task code, result, success, code_steps = await self._write_and_exec_code() - + # ask for acceptance, users can other refuse and change tasks in the plan task_result_confirmed = await self._ask_review() - + if success and task_result_confirmed: # tick off this task and record progress task.code = code @@ -166,14 +163,13 @@ class MLEngineer(Role): task.code_steps = code_steps self.plan.finish_current_task() self.working_memory.clear() - + if "print(df_processed.info())" in code: self.data_desc["column_info"] = result else: # update plan according to user's feedback and to take on changed tasks await self._update_plan() - - + finished_tasks = self.plan.get_finished_tasks() if len(finished_tasks) == len(self.plan.tasks): code_context = [task.code for task in finished_tasks] @@ -181,46 +177,51 @@ class MLEngineer(Role): result, success = await self.execute_code.run(code_context) # truncated the result print(truncate(result)) - + async def _generate_data_desc(self): data_desc = await GenerateDataDesc().run(self.data_path) return data_desc - + async def _write_and_exec_code(self, max_retry: int = 3): code_steps = ( await WriteCodeSteps().run(self.plan) if self.use_code_steps else "" ) - + counter = 0 improve_code = "" success = False - + finished_tasks = self.plan.get_finished_tasks() code_context = [task.code for task in finished_tasks] + code_result = [task.result for task in finished_tasks] code_context = "\n\n".join(code_context) - + code_result = "\n\n".join(code_result) + while not success and counter < max_retry: if counter == 0: context = self.get_useful_memories() else: - improve_code = await DebugCode().run(plan=self.plan, - code= code_context + "\n\n" + code, + # context = self.get_useful_memories() + # logger.info(f"context {context}") + improve_code = await DebugCode().run(plan=self.plan.current_task.instruction, + finished_code=code_context, + finished_code_result=code_result, + code=code, runtime_result=self.working_memory.get()) - - + if not self.use_tools or self.plan.current_task.task_type == "other": logger.info("Write code with pure generation") - + code = await WriteCodeByGenerate().run( context=context, plan=self.plan, code_steps=code_steps, temperature=0.0 ) cause_by = WriteCodeByGenerate else: logger.info("Write code with tools") - - if improve_code!="": + + if improve_code != "": code = improve_code logger.info(f"new code {code}") cause_by = DebugCode @@ -228,15 +229,17 @@ class MLEngineer(Role): code = await WriteCodeWithTools().run( context=context, plan=self.plan, code_steps=code_steps, **{"column_names": {}} ) - + cause_by = WriteCodeWithTools - + self.working_memory.add( Message(content=code, role="assistant", cause_by=cause_by) ) - + # debug on code, run on runcode with finished code and new_df - runcode = code_context + "\n\n" + code + # runcode = code_context + "\n\n" + code + runcode = code + result, success = await self.execute_code.run(runcode) # truncated the result print(truncate(result)) @@ -244,16 +247,16 @@ class MLEngineer(Role): self.working_memory.add( Message(content=truncate(remove_escape_and_color_codes(result)), role="user", cause_by=ExecutePyCode) ) - + if "!pip" in code: - success = False + success = False # if not success: # await self._ask_review() - + counter += 1 - + return code, result, success, code_steps - + async def _ask_review(self): if not self.auto_run: context = self.get_useful_memories() @@ -262,9 +265,10 @@ class MLEngineer(Role): self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) return confirmed return True - + async def _update_plan(self, max_tasks: int = 3): plan_confirmed = False + while not plan_confirmed: context = self.get_useful_memories() rsp = await WritePlan().run( @@ -274,12 +278,17 @@ class MLEngineer(Role): Message(content=rsp, role="assistant", cause_by=WritePlan) ) plan_confirmed = await self._ask_review() - - tasks = WritePlan.rsp_to_tasks(rsp) + + new_tasks = WritePlan.rsp_to_tasks(rsp) + logger.debug(len(self.plan.tasks)) + logger.debug(len(new_tasks)) + ## fixme: 能重复执行多轮重新plan,但应该有更优处理逻辑 + ## fixme: do not overwrite original tasks + tasks = self.plan.tasks + new_tasks + self.plan.add_tasks(tasks) self.working_memory.clear() - - + def get_useful_memories(self) -> List[Message]: """find useful memories only to reduce context length and improve performance""" # TODO dataset description , code steps @@ -295,9 +304,9 @@ class MLEngineer(Role): current_task=current_task ) context_msg = [Message(content=context, role="user")] - + return context_msg + self.working_memory.get() - + @property def working_memory(self): return self._rc.memory @@ -309,15 +318,16 @@ if __name__ == "__main__": # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" - + # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." - + data_path = f"{DATA_PATH}/titanic" requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - + + async def main(requirement: str = requirement, auto_run: bool = True, data_path: str = ""): role = MLEngineer(goal=requirement, auto_run=auto_run, data_path=data_path) await role.run(requirement) - - + + fire.Fire(main) From 0278934131ff53d8a83fcb46a4b17c6c262ac28f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 12 Dec 2023 10:22:55 +0800 Subject: [PATCH 103/637] chore --- metagpt/actions/make_tools.py | 1 - 1 file changed, 1 deletion(-) diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index 2b2ba1cd5..74037e900 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -49,7 +49,6 @@ class MakeTools(WriteCodeByGenerate): saved_path = Path(self.workspace).joinpath(func_name+self.file_suffix) logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") saved_path.write_text(tool_code, encoding='utf-8') - # TODO: 保存到udf中,供WriteCodeWithMakeTools使用 @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) async def run(self, code_message: List[Message | Dict], **kwargs) -> str: From fd31cc065a74ce8b17765ab7b44ff51ce0adc833 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Tue, 12 Dec 2023 10:30:05 +0800 Subject: [PATCH 104/637] save jupyter file --- metagpt/actions/execute_code.py | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 981aa894c..6fd980494 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -156,6 +156,11 @@ class ExecutePyCode(ExecuteCode, Action): return code, language + def save_notebook(self, path: str): + path = Path(path) + path.parent.mkdir(parents=True, exist_ok=True) + nbformat.write(self.nb, path) + async def run(self, code: Union[str, Dict, Message], language: str = "python") -> Tuple[str, bool]: code, language = self._process_code(code, language) From db96644a0842f30545ed7de106ed01c3cdb75cd9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 12 Dec 2023 10:45:15 +0800 Subject: [PATCH 105/637] chore --- metagpt/tools/functions/libs/udf/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index c90357b5c..c581dd992 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -24,7 +24,7 @@ def extract_function_signatures(file_path): # 导入函数 module = Path(file_path).parts[-1][:-len(Path(file_path).suffix)] module = importlib.import_module(f"metagpt.tools.functions.libs.udf.{module}") - # 获取函数注释 + # 获取函数注释和函数路径 function_schema = {'name': function_signature, 'doc': inspect.getdoc(getattr(module, function_name)), 'path': f'from metagpt.tools.functions.libs.udf.{module} import function_name'} function_signatures.append(function_schema) From 4f0d55656e17c2247b84d748f8cb0cc0ebba5176 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Tue, 12 Dec 2023 10:56:05 +0800 Subject: [PATCH 106/637] update ml tool from Function to Class --- metagpt/tools/functions/libs/base.py | 16 + .../tools/functions/libs/data_preprocess.py | 248 ++++++---- .../functions/libs/feature_engineering.py | 427 +++++++++++------- 3 files changed, 445 insertions(+), 246 deletions(-) create mode 100644 metagpt/tools/functions/libs/base.py diff --git a/metagpt/tools/functions/libs/base.py b/metagpt/tools/functions/libs/base.py new file mode 100644 index 000000000..c39adc66b --- /dev/null +++ b/metagpt/tools/functions/libs/base.py @@ -0,0 +1,16 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Time : 2023/12/10 20:12 +# @Author : lidanyang +# @File : base +# @Desc : +class MLProcess(object): + def fit(self, df): + raise NotImplementedError + + def transform(self, df): + raise NotImplementedError + + def fit_transform(self, df): + self.fit(df) + return self.transform(df) diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index 5579c5bd8..39474b0fd 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -1,6 +1,6 @@ import numpy as np from sklearn.impute import SimpleImputer -from sklearn.preprocessing import KBinsDiscretizer +from sklearn.preprocessing import KBinsDiscretizer, LabelEncoder from sklearn.preprocessing import MaxAbsScaler from sklearn.preprocessing import MinMaxScaler from sklearn.preprocessing import OneHotEncoder @@ -9,31 +9,52 @@ from sklearn.preprocessing import RobustScaler from sklearn.preprocessing import StandardScaler from metagpt.tools.functions import registry +from metagpt.tools.functions.libs.base import MLProcess from metagpt.tools.functions.schemas.data_preprocess import * -@registry.register("data_preprocess", FillMissingValue) -def fill_missing_value(df: pd.DataFrame, features: list, strategy: str = 'mean', fill_value=None,): - df[features] = SimpleImputer(strategy=strategy, fill_value=fill_value).fit_transform(df[features]) - return df +class FillMissingValue(MLProcess): + def __init__(self, features: list, strategy: str = 'mean', fill_value=None,): + self.features = features + self.strategy = strategy + self.fill_value = fill_value + self.si = None + + def fit(self, df: pd.DataFrame): + self.si = SimpleImputer(strategy=self.strategy, fill_value=self.fill_value) + self.si.fit(df[self.features]) + + def transform(self, df: pd.DataFrame): + df[self.features] = self.si.transform(df[self.features]) + return df -@registry.register("data_preprocess", SplitBins) -def split_bins(df: pd.DataFrame, features: list, strategy: str = 'quantile',): - df[features] = KBinsDiscretizer(strategy=strategy, encode='ordinal').fit_transform(df[features]) - return df +class MinMaxScale(MLProcess): + def __init__(self, features: list,): + self.features = features + self.mms = None + + def fit(self, df: pd.DataFrame): + self.mms = MinMaxScaler() + self.mms.fit(df[self.features]) + + def transform(self, df: pd.DataFrame): + df[self.features] = self.mms.transform(df[self.features]) + return df -@registry.register("data_preprocess", MinMaxScale) -def min_max_scale(df: pd.DataFrame, features: list, ): - df[features] = MinMaxScaler().fit_transform(df[features]) - return df +class StandardScale(MLProcess): + def __init__(self, features: list,): + self.features = features + self.ss = None + def fit(self, df: pd.DataFrame): + self.ss = StandardScaler() + self.ss.fit(df[self.features]) -@registry.register("data_preprocess", StandardScale) -def standard_scale(df: pd.DataFrame, features: list, ): - df[features] = StandardScaler().fit_transform(df[features]) - return df + def transform(self, df: pd.DataFrame): + df[self.features] = self.ss.transform(df[self.features]) + return df @registry.register("data_preprocess", LogTransform) @@ -45,80 +66,145 @@ def log_transform(df: pd.DataFrame, features: list, ): return df -@registry.register("data_preprocess", MaxAbsScale) -def max_abs_scale(df: pd.DataFrame, features: list, ): - df[features] = MaxAbsScaler().fit_transform(df[features]) - return df +class MaxAbsScale(MLProcess): + def __init__(self, features: list,): + self.features = features + self.mas = None + + def fit(self, df: pd.DataFrame): + self.mas = MaxAbsScaler() + self.mas.fit(df[self.features]) + + def transform(self, df: pd.DataFrame): + df[self.features] = self.mas.transform(df[self.features]) + return df -@registry.register("data_preprocess", RobustScale) -def robust_scale(df: pd.DataFrame, features: list, ): - df[features] = RobustScaler().fit_transform(df[features]) - return df +class RobustScale(MLProcess): + def __init__(self, features: list,): + self.features = features + self.rs = None + + def fit(self, df: pd.DataFrame): + self.rs = RobustScaler() + self.rs.fit(df[self.features]) + + def transform(self, df: pd.DataFrame): + df[self.features] = self.rs.transform(df[self.features]) + return df -@registry.register("data_preprocess", OrdinalEncode) -def ordinal_encode(df: pd.DataFrame, features: list,): - df[features] = OrdinalEncoder().fit_transform(df[features]) - return df +class OrdinalEncode(MLProcess): + def __init__(self, features: list,): + self.features = features + self.oe = None + + def fit(self, df: pd.DataFrame): + self.oe = OrdinalEncoder() + self.oe.fit(df[self.features]) + + def transform(self, df: pd.DataFrame): + df[self.features] = self.oe.transform(df[self.features]) + return df -@registry.register("data_preprocess", OneHotEncoding) -def one_hot_encoding(df, cols): - enc = OneHotEncoder(handle_unknown="ignore", sparse=False) - ts_data = enc.fit_transform(df[cols]) - new_columns = enc.get_feature_names_out(cols) - ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) - df.drop(cols, axis=1, inplace=True) - df = pd.concat([df, ts_data], axis=1) - return df +class OneHotEncode(MLProcess): + def __init__(self, features: list,): + self.features = features + self.ohe = None + + def fit(self, df: pd.DataFrame): + self.ohe = OneHotEncoder(handle_unknown="ignore", sparse=False) + self.ohe.fit(df[self.features]) + + def transform(self, df: pd.DataFrame): + ts_data = self.ohe.transform(df[self.features]) + new_columns = self.ohe.get_feature_names_out(self.features) + ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) + df.drop(self.features, axis=1, inplace=True) + df = pd.concat([df, ts_data], axis=1) + return df -if __name__ == '__main__': - def run(): - V = { - 'a': [-1, 2, 3, 6, 5, 4], - 'b': [1.1, 2.2, 3.3, 6.6, 5.5, 4.4], - 'c': ['aa', 'bb', 'cc', 'dd', 'ee', 'ff'], - 'd': [1, None, 3, None, 5, 4], - 'e': [1.1, np.NAN, 3.3, None, 5.5, 4.4], - 'f': ['aa', np.NAN, 'cc', None, '', 'ff'], +class LabelEncode(MLProcess): + def __init__(self, features: list,): + self.features = features + self.le_encoders = [] - } + def fit(self, df: pd.DataFrame): + for col in self.features: + le = LabelEncoder().fit(df[col].astype(str).unique().tolist() + ['unknown']) + self.le_encoders.append(le) - df = pd.DataFrame(V) - print(df.dtypes) + def transform(self, df: pd.DataFrame): + for i in range(len(self.features)): + data_list = df[self.features[i]].astype(str).tolist() + for unique_item in np.unique(df[self.features[i]].astype(str)): + if unique_item not in self.le_encoders[i].classes_: + data_list = ['unknown' if x == unique_item else x for x in data_list] + df[self.features[i]] = self.le_encoders[i].transform(data_list) + return df - numeric_features = ['a', 'b', 'd', 'e'] - numeric_features_wo_miss = ['a', 'b', ] - categorial_features = ['c', 'f'] - df_ = fill_missing_value(df.copy(), numeric_features) - print(df_) - df_ = fill_missing_value(df.copy(), categorial_features, strategy='constant', fill_value='hehe') - print(df_) +def get_column_info(df: pd.DataFrame) -> str: + data = [] + for i in df.columns: + nan_freq = float("%.2g" % (df[i].isna().mean() * 100)) + n_unique = df[i].nunique() + data.append([i, df[i].dtype, nan_freq, n_unique]) - df_ = fill_missing_value(df.copy(), numeric_features, strategy='constant', fill_value=999) - print(df_) - - # df_ = label_encode(df.copy(), numeric_features + categorial_features, ) - # print(df_) - - df_ = split_bins(df.copy(), numeric_features_wo_miss, strategy='quantile') - print(df_) - - df_ = min_max_scale(df.copy(), numeric_features, ) - print(df_) - - df_ = standard_scale(df.copy(), numeric_features, ) - print(df_) - - df_ = log_transform(df.copy(), numeric_features, ) - print(df_) - - df_ = max_abs_scale(df.copy(), numeric_features, ) - print(df_) - - df_ = robust_scale(df.copy(), numeric_features, ) - print(df_) - run() \ No newline at end of file + samples = pd.DataFrame( + data, + columns=["Column_name", "Data_type", "NaN_Frequency(%)", "N_unique"], + ) + return samples.to_string(index=False) +# +# +# if __name__ == '__main__': +# def run(): +# V = { +# 'a': [-1, 2, 3, 6, 5, 4], +# 'b': [1.1, 2.2, 3.3, 6.6, 5.5, 4.4], +# 'c': ['aa', 'bb', 'cc', 'dd', 'ee', 'ff'], +# 'd': [1, None, 3, None, 5, 4], +# 'e': [1.1, np.NAN, 3.3, None, 5.5, 4.4], +# 'f': ['aa', np.NAN, 'cc', None, '', 'ff'], +# +# } +# +# df = pd.DataFrame(V) +# print(df.dtypes) +# +# numeric_features = ['a', 'b', 'd', 'e'] +# numeric_features_wo_miss = ['a', 'b', ] +# categorial_features = ['c', 'f'] +# +# df_ = fill_missing_value(df.copy(), numeric_features) +# print(df_) +# df_ = fill_missing_value(df.copy(), categorial_features, strategy='constant', fill_value='hehe') +# print(df_) +# +# df_ = fill_missing_value(df.copy(), numeric_features, strategy='constant', fill_value=999) +# print(df_) +# +# # df_ = label_encode(df.copy(), numeric_features + categorial_features, ) +# # print(df_) +# +# df_ = split_bins(df.copy(), numeric_features_wo_miss, strategy='quantile') +# print(df_) +# +# df_ = min_max_scale(df.copy(), numeric_features, ) +# print(df_) +# +# df_ = standard_scale(df.copy(), numeric_features, ) +# print(df_) +# +# df_ = log_transform(df.copy(), numeric_features, ) +# print(df_) +# +# df_ = max_abs_scale(df.copy(), numeric_features, ) +# print(df_) +# +# df_ = robust_scale(df.copy(), numeric_features, ) +# print(df_) +# run() \ No newline at end of file diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index 4780e4fa0..06a988d9a 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -3,188 +3,285 @@ # @Time : 2023/11/17 10:33 # @Author : lidanyang # @File : feature_engineering.py -# @Desc : Feature Engineering Functions +# @Desc : Feature Engineering Tools import itertools +import numpy as np from dateutil.relativedelta import relativedelta +from joblib import Parallel, delayed from pandas.api.types import is_numeric_dtype from sklearn.model_selection import KFold -from sklearn.preprocessing import PolynomialFeatures +from sklearn.preprocessing import PolynomialFeatures, KBinsDiscretizer -from metagpt.tools.functions import registry +from metagpt.tools.functions.libs.base import MLProcess from metagpt.tools.functions.schemas.feature_engineering import * -@registry.register("feature_engineering", PolynomialExpansion) -def polynomial_expansion(df, cols, degree=2): - for col in cols: - if not is_numeric_dtype(df[col]): - raise ValueError(f"Column '{col}' must be numeric.") +class PolynomialExpansion(MLProcess): + def __init__(self, cols: list, degree: int = 2): + self.cols = cols + self.degree = degree + self.poly = PolynomialFeatures(degree=degree, include_bias=False) - poly = PolynomialFeatures(degree=degree, include_bias=False) - ts_data = poly.fit_transform(df[cols].fillna(0)) - new_columns = poly.get_feature_names_out(cols) - ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) - ts_data = ts_data.drop(cols, axis=1) - df = pd.concat([df, ts_data], axis=1) - return df + def fit(self, df: pd.DataFrame): + self.poly.fit(df[self.cols].fillna(0)) + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + ts_data = self.poly.transform(df[self.cols].fillna(0)) + column_name = self.poly.get_feature_names_out(self.cols) + ts_data = pd.DataFrame(ts_data, index=df.index, columns=column_name) + df.drop(self.cols, axis=1, inplace=True) + df = pd.concat([df, ts_data], axis=1) + return df -@registry.register("feature_engineering", FrequencyEncoding) -def frequency_encoding(df, cols): - for col in cols: - encoder_dict = df[col].value_counts().to_dict() - df[f"{col}_cnt"] = df[col].map(encoder_dict) - return df +class CatCount(MLProcess): + def __init__(self, col: str): + self.col = col + self.encoder_dict = None + + def fit(self, df: pd.DataFrame): + self.encoder_dict = df[self.col].value_counts().to_dict() + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + df[f"{self.col}_cnt"] = df[self.col].map(self.encoder_dict) + return df -@registry.register("feature_engineering", TargetMeanEncoder) -def target_mean_encoder(df, col, label): - encoder_dict = df.groupby(col)[label].mean().to_dict() - df[f"{col}_target_mean"] = df[col].map(encoder_dict) - return df +class TargetMeanEncoder(MLProcess): + def __init__(self, col: str, label: str): + self.col = col + self.label = label + self.encoder_dict = None + + def fit(self, df: pd.DataFrame): + self.encoder_dict = df.groupby(self.col)[self.label].mean().to_dict() + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + df[f"{self.col}_target_mean"] = df[self.col].map(self.encoder_dict) + return df -@registry.register("feature_engineering", KFoldTargetMeanEncoder) -def k_fold_target_mean_encoder(df, col, label, n_splits=5, random_state=2021): - tmp = df.copy() - kf = KFold(n_splits=n_splits, shuffle=True, random_state=random_state) +class KFoldTargetMeanEncoder(MLProcess): + def __init__(self, col: str, label: str, n_splits: int = 5, random_state: int = 2021): + self.col = col + self.label = label + self.n_splits = n_splits + self.random_state = random_state + self.encoder_dict = None - global_mean = tmp[label].mean() - col_name = f"{col}_kf_target_mean" - for trn_idx, val_idx in kf.split(tmp, tmp[label]): - _trn, _val = tmp.iloc[trn_idx], tmp.iloc[val_idx] - tmp.loc[tmp.index[val_idx], col_name] = _val[col].map( - _trn.groupby(col)[label].mean() - ) - tmp[col_name].fillna(global_mean, inplace=True) - encoder_dict = tmp.groupby(col)[col_name].mean().to_dict() - df[f"{col}_kf_target_mean"] = df[col].map(encoder_dict) - return df + def fit(self, df: pd.DataFrame): + tmp = df.copy() + kf = KFold(n_splits=self.n_splits, shuffle=True, random_state=self.random_state) - -@registry.register("feature_engineering", CatCross) -def cat_cross(df, cols, max_cat_num=100): - for col in cols: - if df[col].nunique() > max_cat_num: - cols.remove(col) - - for col1, col2 in itertools.combinations(cols, 2): - cross_col = f"{col1}_cross_{col2}" - crossed = df[col1].astype(str) + "_" + df[col2].astype(str) - df[cross_col] = crossed.astype('category').cat.codes - return df - - -@registry.register("feature_engineering", GroupStat) -def group_stat(df, group_col, agg_col, agg_funcs): - group_df = df.groupby(group_col)[agg_col].agg(agg_funcs).reset_index() - group_df.columns = group_col + [ - f"{agg_col}_{agg_func}_by_{group_col}" for agg_func in agg_funcs - ] - df = df.merge(group_df, on=group_col, how="left") - return df - - -@registry.register("feature_engineering", ExtractTimeComps) -def extract_time_comps(df, time_col, time_comps): - time_s = pd.to_datetime(df[time_col], errors="coerce") - time_comps_df = pd.DataFrame() - - if "year" in time_comps: - time_comps_df["year"] = time_s.dt.year - if "month" in time_comps: - time_comps_df["month"] = time_s.dt.month - if "day" in time_comps: - time_comps_df["day"] = time_s.dt.day - if "hour" in time_comps: - time_comps_df["hour"] = time_s.dt.hour - if "dayofweek" in time_comps: - time_comps_df["dayofweek"] = time_s.dt.dayofweek + 1 - if "is_weekend" in time_comps: - time_comps_df["is_weekend"] = time_s.dt.dayofweek.isin([5, 6]).astype(int) - df = pd.concat([df, time_comps_df], axis=1) - return df - - -@registry.register("feature_engineering", FeShiftByTime) -def fe_shift_by_time(df, time_col, group_col, shift_col, periods, freq): - df[time_col] = pd.to_datetime(df[time_col]) - - def shift_datetime(date, offset, unit): - if unit in ["year", "y", "Y"]: - return date + relativedelta(years=offset) - elif unit in ["month", "m", "M"]: - return date + relativedelta(months=offset) - elif unit in ["day", "d", "D"]: - return date + relativedelta(days=offset) - elif unit in ["week", "w", "W"]: - return date + relativedelta(weeks=offset) - elif unit in ["hour", "h", "H"]: - return date + relativedelta(hours=offset) - else: - return date - - def shift_by_time_on_key( - inner_df, time_col, group_col, shift_col, offset, unit, col_name - ): - inner_df = inner_df.drop_duplicates() - inner_df[time_col] = inner_df[time_col].map( - lambda x: shift_datetime(x, offset, unit) - ) - inner_df = inner_df.groupby([time_col, group_col], as_index=False)[ - shift_col - ].mean() - inner_df.rename(columns={shift_col: col_name}, inplace=True) - return inner_df - - shift_df = df[[time_col, group_col, shift_col]].copy() - for period in periods: - new_col_name = f"{group_col}_{shift_col}_lag_{period}_{freq}" - tmp = shift_by_time_on_key( - shift_df, time_col, group_col, shift_col, period, freq, new_col_name - ) - df = df.merge(tmp, on=[time_col, group_col], how="left") - - return df - - -@registry.register("feature_engineering", FeRollingByTime) -def fe_rolling_by_time(df, time_col, group_col, rolling_col, periods, freq, agg_funcs): - df[time_col] = pd.to_datetime(df[time_col]) - - def rolling_by_time_on_key(inner_df, offset, unit, agg_func, col_name): - time_freq = { - "Y": [365 * offset, "D"], - "M": [30 * offset, "D"], - "D": [offset, "D"], - "W": [7 * offset, "D"], - "H": [offset, "h"], - } - - if agg_func not in ["mean", "std", "max", "min", "median", "sum", "count"]: - raise ValueError(f"Invalid agg function: {agg_func}") - - rolling_feat = inner_df.rolling( - f"{time_freq[unit][0]}{time_freq[unit][1]}", closed="left" - ) - rolling_feat = getattr(rolling_feat, agg_func)() - depth = df.columns.nlevels - rolling_feat = rolling_feat.stack(list(range(depth))) - rolling_feat.name = col_name - return rolling_feat - - rolling_df = df[[time_col, group_col, rolling_col]].copy() - for period in periods: - for func in agg_funcs: - new_col_name = f"{group_col}_{rolling_col}_rolling_{period}_{freq}_{func}" - tmp = pd.pivot_table( - rolling_df, - index=time_col, - values=rolling_col, - columns=group_col, + global_mean = tmp[self.label].mean() + col_name = f"{self.col}_kf_target_mean" + for trn_idx, val_idx in kf.split(tmp, tmp[self.label]): + _trn, _val = tmp.iloc[trn_idx], tmp.iloc[val_idx] + tmp.loc[tmp.index[val_idx], col_name] = _val[self.col].map( + _trn.groupby(self.col)[self.label].mean() ) - tmp = rolling_by_time_on_key(tmp, period, freq, func, new_col_name) - df = df.merge(tmp, on=[time_col, group_col], how="left") + tmp[col_name].fillna(global_mean, inplace=True) + self.encoder_dict = tmp.groupby(self.col)[col_name].mean().to_dict() - return df + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + df[f"{self.col}_kf_target_mean"] = df[self.col].map(self.encoder_dict) + return df + + +class CatCross(MLProcess): + def __init__(self, cols: list, max_cat_num: int = 100): + self.cols = cols + self.max_cat_num = max_cat_num + self.combs = [] + self.combs_map = {} + + @staticmethod + def cross_two(comb, df): + new_col = f'{comb[0]}_{comb[1]}' + new_col_combs = list(itertools.product(df[comb[0]].unique(), df[comb[1]].unique())) + ll = list(range(len(new_col_combs))) + comb_map = dict(zip(new_col_combs, ll)) + return new_col, comb_map + + def fit(self, df: pd.DataFrame): + for col in self.cols: + if df[col].nunique() > self.max_cat_num: + self.cols.remove(col) + self.combs = list(itertools.combinations(self.cols, 2)) + res = Parallel(n_jobs=4, require='sharedmem')( + delayed(self.cross_two)(comb, df) for comb in self.combs) + self.combs_map = dict(res) + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + for comb in self.combs: + new_col = f'{comb[0]}_{comb[1]}' + _map = self.combs_map[new_col] + df[new_col] = pd.Series(zip(df[comb[0]], df[comb[1]])).map(_map) + # set the unknown value to a new number + df[new_col].fillna(max(_map.values()) + 1, inplace=True) + df[new_col] = df[new_col].astype(int) + return df + + +class GroupStat(MLProcess): + def __init__(self, group_col: str, agg_col: str, agg_funcs: list): + self.group_col = group_col + self.agg_col = agg_col + self.agg_funcs = agg_funcs + self.group_df = None + + def fit(self, df: pd.DataFrame): + group_df = df.groupby(self.group_col)[self.agg_col].agg(self.agg_funcs).reset_index() + group_df.columns = [self.group_col] + [ + f"{self.agg_col}_{agg_func}_by_{self.group_col}" for agg_func in self.agg_funcs + ] + self.group_df = group_df + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + df = df.merge(self.group_df, on=self.group_col, how="left") + return df + + +class SplitBins(MLProcess): + def __init__(self, cols: str, strategy: str = 'quantile'): + self.cols = cols + self.strategy = strategy + self.encoder = None + + def fit(self, df: pd.DataFrame): + self.encoder = KBinsDiscretizer(strategy=self.strategy, encode='ordinal') + self.encoder.fit(df[self.cols].fillna(0)) + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + df[self.cols] = self.encoder.transform(df[self.cols].fillna(0)) + return df + +# @registry.register("feature_engineering", ExtractTimeComps) +# def extract_time_comps(df, time_col, time_comps): +# time_s = pd.to_datetime(df[time_col], errors="coerce") +# time_comps_df = pd.DataFrame() +# +# if "year" in time_comps: +# time_comps_df["year"] = time_s.dt.year +# if "month" in time_comps: +# time_comps_df["month"] = time_s.dt.month +# if "day" in time_comps: +# time_comps_df["day"] = time_s.dt.day +# if "hour" in time_comps: +# time_comps_df["hour"] = time_s.dt.hour +# if "dayofweek" in time_comps: +# time_comps_df["dayofweek"] = time_s.dt.dayofweek + 1 +# if "is_weekend" in time_comps: +# time_comps_df["is_weekend"] = time_s.dt.dayofweek.isin([5, 6]).astype(int) +# df = pd.concat([df, time_comps_df], axis=1) +# return df +# +# +# @registry.register("feature_engineering", FeShiftByTime) +# def fe_shift_by_time(df, time_col, group_col, shift_col, periods, freq): +# df[time_col] = pd.to_datetime(df[time_col]) +# +# def shift_datetime(date, offset, unit): +# if unit in ["year", "y", "Y"]: +# return date + relativedelta(years=offset) +# elif unit in ["month", "m", "M"]: +# return date + relativedelta(months=offset) +# elif unit in ["day", "d", "D"]: +# return date + relativedelta(days=offset) +# elif unit in ["week", "w", "W"]: +# return date + relativedelta(weeks=offset) +# elif unit in ["hour", "h", "H"]: +# return date + relativedelta(hours=offset) +# else: +# return date +# +# def shift_by_time_on_key( +# inner_df, time_col, group_col, shift_col, offset, unit, col_name +# ): +# inner_df = inner_df.drop_duplicates() +# inner_df[time_col] = inner_df[time_col].map( +# lambda x: shift_datetime(x, offset, unit) +# ) +# inner_df = inner_df.groupby([time_col, group_col], as_index=False)[ +# shift_col +# ].mean() +# inner_df.rename(columns={shift_col: col_name}, inplace=True) +# return inner_df +# +# shift_df = df[[time_col, group_col, shift_col]].copy() +# for period in periods: +# new_col_name = f"{group_col}_{shift_col}_lag_{period}_{freq}" +# tmp = shift_by_time_on_key( +# shift_df, time_col, group_col, shift_col, period, freq, new_col_name +# ) +# df = df.merge(tmp, on=[time_col, group_col], how="left") +# +# return df +# +# +# @registry.register("feature_engineering", FeRollingByTime) +# def fe_rolling_by_time(df, time_col, group_col, rolling_col, periods, freq, agg_funcs): +# df[time_col] = pd.to_datetime(df[time_col]) +# +# def rolling_by_time_on_key(inner_df, offset, unit, agg_func, col_name): +# time_freq = { +# "Y": [365 * offset, "D"], +# "M": [30 * offset, "D"], +# "D": [offset, "D"], +# "W": [7 * offset, "D"], +# "H": [offset, "h"], +# } +# +# if agg_func not in ["mean", "std", "max", "min", "median", "sum", "count"]: +# raise ValueError(f"Invalid agg function: {agg_func}") +# +# rolling_feat = inner_df.rolling( +# f"{time_freq[unit][0]}{time_freq[unit][1]}", closed="left" +# ) +# rolling_feat = getattr(rolling_feat, agg_func)() +# depth = df.columns.nlevels +# rolling_feat = rolling_feat.stack(list(range(depth))) +# rolling_feat.name = col_name +# return rolling_feat +# +# rolling_df = df[[time_col, group_col, rolling_col]].copy() +# for period in periods: +# for func in agg_funcs: +# new_col_name = f"{group_col}_{rolling_col}_rolling_{period}_{freq}_{func}" +# tmp = pd.pivot_table( +# rolling_df, +# index=time_col, +# values=rolling_col, +# columns=group_col, +# ) +# tmp = rolling_by_time_on_key(tmp, period, freq, func, new_col_name) +# df = df.merge(tmp, on=[time_col, group_col], how="left") +# +# return df + + +class GeneralSelection(MLProcess): + def __init__(self, label_col: str): + self.label_col = label_col + self.feats = [] + + def fit(self, df: pd.DataFrame): + feats = [f for f in df.columns if f != self.label_col] + for col in df.columns: + if df[col].isnull().sum() / df.shape[0] == 1: + feats.remove(col) + + if df[col].nunique() == 1: + feats.remove(col) + + if ( + df.loc[df[col] == np.inf].shape[0] != 0 + or df.loc[df[col] == np.inf].shape[0] != 0 + ): + feats.remove(col) + self.feats = feats + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + df = df[self.feats] + return df From 07771a769955f900b305334920d2ef5c70eae5bc Mon Sep 17 00:00:00 2001 From: lidanyang Date: Tue, 12 Dec 2023 10:56:51 +0800 Subject: [PATCH 107/637] add ml Class tool schema --- .../functions/schemas/data_preprocess.yml | 306 +++++++++++++ .../functions/schemas/feature_engineering.yml | 429 ++++++++++++++++++ 2 files changed, 735 insertions(+) create mode 100644 metagpt/tools/functions/schemas/data_preprocess.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering.yml diff --git a/metagpt/tools/functions/schemas/data_preprocess.yml b/metagpt/tools/functions/schemas/data_preprocess.yml new file mode 100644 index 000000000..95b0124cc --- /dev/null +++ b/metagpt/tools/functions/schemas/data_preprocess.yml @@ -0,0 +1,306 @@ +FillMissingValue: + type: class + description: "Completing missing values with simple strategies" + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "columns to be processed" + strategy: + type: str + description: "the imputation strategy" + default: mean + enum: + - mean + - median + - most_frequent + - constant + fill_value: + type: int + description: "fill_value is used to replace all occurrences of missing_values" + default: null + required: + - features + fit: + description: "Fit the FillMissingValue model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +MinMaxScale: + type: class + description: "Transform features by scaling each feature to a range, witch is (0, 1)" + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "columns to be processed" + required: + - features + fit: + description: "Fit the MinMaxScale model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +StandardScale: + type: class + description: "Standardize features by removing the mean and scaling to unit variance" + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "columns to be processed" + required: + - features + fit: + description: "Fit the StandardScale model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +MaxAbsScale: + type: class + description: "cale each feature by its maximum absolute value" + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "columns to be processed" + required: + - features + fit: + description: "Fit the MaxAbsScale model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +LabelEncode: + type: class + description: "Apply label encoding to specified categorical columns in-place." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "Categorical columns to be label encoded" + required: + - features + fit: + description: "Fit the LabelEncode model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +OneHotEncode: + type: class + description: "Apply one-hot encoding to specified categorical columns, the original columns will be dropped." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "Categorical columns to be one-hot encoded and dropped" + required: + - features + fit: + description: "Fit the OneHotEncoding model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering.yml b/metagpt/tools/functions/schemas/feature_engineering.yml new file mode 100644 index 000000000..2cc4ec2fa --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering.yml @@ -0,0 +1,429 @@ +PolynomialExpansion: + type: class + description: "Add polynomial and interaction features from selected numeric columns, excluding the bias column." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + cols: + type: list + description: "Columns for polynomial expansion." + degree: + type: int + description: "The degree of the polynomial features." + default: 2 + required: + - cols + fit: + description: "Fit the PolynomialExpansion model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +CatCount: + type: class + description: "Add value counts of categorical columns as new features." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + cols: + type: list + description: "Columns for value counts." + required: + - cols + fit: + description: "Fit the CatCount model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +TargetMeanEncoder: + type: class + description: "Encodes a categorical column by the mean of the label column, and adds the result as a new feature." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + col: + type: str + description: "Column to be mean encoded." + label: + type: str + description: "Predicted label column." + required: + - col + - label + fit: + description: "Fit the TargetMeanEncoder model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +KFoldTargetMeanEncoder: + type: class + description: "Adds a new feature to the DataFrame by k-fold mean encoding of a categorical column using the label column." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + col: + type: str + description: "Column to be k-fold mean encoded." + label: + type: str + description: "Predicted label column." + n_splits: + type: int + description: "Number of splits for K-fold." + default: 5 + random_state: + type: int + description: "Random seed." + default: 2021 + required: + - col + - label + fit: + description: "Fit the KFoldTargetMeanEncoder model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +CatCross: + type: class + description: "Add pairwise crossed features and convert them to numerical features." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + cols: + type: list + description: "Columns to be pairwise crossed." + max_cat_num: + type: int + description: "Maximum unique categories per crossed feature." + default: 100 + required: + - cols + fit: + description: "Fit the CatCross model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +GroupStat: + type: class + description: "Aggregate specified column in a DataFrame grouped by another column, adding new features named '__by_'." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + group_col: + type: str + description: "Column used for grouping." + agg_col: + type: str + description: "Column on which aggregation is performed." + agg_funcs: + type: list + description: >- + List of aggregation functions to apply, such as ['mean', 'std']. + Each function must be supported by pandas. + required: + - group_col + - agg_col + - agg_funcs + fit: + description: "Fit the GroupStat model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +SplitBins: + type: class + description: "Bin continuous data into intervals and return the bin identifier encoded as an integer value" + methods: + __init__: + description: "Initialize self." + parameters: + properties: + cols: + type: list + description: "Columns to be binned." + strategy: + type: str + description: "Strategy used to define the widths of the bins." + default: quantile + required: + - cols + fit: + description: "Fit the SplitBins model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +GeneralSelection: + type: class + description: "Drop all nan feats and feats with only one unique value." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + label_col: + type: str + description: "Label column name." + required: + - label_col + fit: + description: "Fit the GeneralSelection model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file From 10d488c49a2aaf93f93e5ff43daf58811b5cd195 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 12 Dec 2023 11:12:17 +0800 Subject: [PATCH 108/637] fix: path in function_signatures. --- metagpt/tools/functions/libs/udf/__init__.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index c581dd992..e44e97c41 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -22,11 +22,11 @@ def extract_function_signatures(file_path): # 获取函数签名 function_signature = f"{function_name}({', '.join(args)})" # 导入函数 - module = Path(file_path).parts[-1][:-len(Path(file_path).suffix)] - module = importlib.import_module(f"metagpt.tools.functions.libs.udf.{module}") + module_name = Path(file_path).parts[-1][:-len(Path(file_path).suffix)] + module = importlib.import_module(f"metagpt.tools.functions.libs.udf.{module_name}") # 获取函数注释和函数路径 function_schema = {'name': function_signature, 'doc': inspect.getdoc(getattr(module, function_name)), - 'path': f'from metagpt.tools.functions.libs.udf.{module} import function_name'} + 'path': f'from metagpt.tools.functions.libs.udf.{module_name} import {function_name}'} function_signatures.append(function_schema) return function_signatures From 988c7072ef6084b0a4cf46d55cc6023f36c0b8b8 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Tue, 12 Dec 2023 13:36:54 +0800 Subject: [PATCH 109/637] give history code for current code steps --- metagpt/actions/write_code_steps.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py index 0bfb9c225..a19549b71 100644 --- a/metagpt/actions/write_code_steps.py +++ b/metagpt/actions/write_code_steps.py @@ -63,7 +63,7 @@ class WriteCodeSteps(Action): def get_context(self, plan: Plan): user_requirement = plan.goal - select_task_keys = ['task_id', 'instruction', 'is_finished', 'code_steps'] + select_task_keys = ['task_id', 'instruction', 'is_finished', 'code'] def process_task(task): task_dict = task.dict() From 9c426f73dc7ba876fbe076a4d5f71996424fcfcf Mon Sep 17 00:00:00 2001 From: lidanyang Date: Tue, 12 Dec 2023 13:39:41 +0800 Subject: [PATCH 110/637] fix bug --- metagpt/tools/functions/libs/feature_engineering.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index 06a988d9a..67247d0d1 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -283,5 +283,5 @@ class GeneralSelection(MLProcess): self.feats = feats def transform(self, df: pd.DataFrame) -> pd.DataFrame: - df = df[self.feats] + df = df[self.feats + [self.label_col]] return df From 3847e672b1ad8ad4f6ca5c8a149f570c445b2e09 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 12 Dec 2023 14:20:15 +0800 Subject: [PATCH 111/637] rm redundant --- metagpt/actions/execute_code.py | 2 -- metagpt/actions/ml_da_action.py | 13 ------------- 2 files changed, 15 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 9c2b8d96c..1d20bf3f6 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -175,8 +175,6 @@ class ExecutePyCode(ExecuteCode, Action): outputs = self.parse_outputs(self.nb.cells[-1].outputs) success = True except Exception as e: - # FIXME: CellExecutionError is hard to read. for example `1\0` raise ZeroDivisionError: - # CellExecutionError('An error occurred while executing the following cell:\n------------------\nz=1/0\n------------------\n\n\n\x1b[0;31m---------------------------------------------------------------------------\x1b[0m\n\x1b[0;31mZeroDivisionError\x1b[0m Traceback (most recent call last)\nCell \x1b[0;32mIn[1], line 1\x1b[0m\n\x1b[0;32m----> 1\x1b[0m z\x1b[38;5;241m=\x1b[39m\x1b[38;5;241;43m1\x1b[39;49m\x1b[38;5;241;43m/\x1b[39;49m\x1b[38;5;241;43m0\x1b[39;49m\n\n\x1b[0;31mZeroDivisionError\x1b[0m: division by zero\n') outputs = traceback.format_exc() success = False return truncate(remove_escape_and_color_codes(outputs)), success diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py index 6be4b3040..5e4580b17 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_da_action.py @@ -7,19 +7,6 @@ from metagpt.utils.common import CodeParser from metagpt.logs import logger -def truncate(result: str, keep_len: int = 2000) -> str: - desc = "Truncated to show only the last keep_len characters\n" - if result.startswith(desc): - result = result[-len(desc) :] - - if len(result) > keep_len: - result = result[-keep_len:] - - if not result.startswith(desc): - return desc + result - return desc - - class ReviewConst: TASK_REVIEW_TRIGGER = "task" CODE_REVIEW_TRIGGER = "code" From b7624d7298536135e84c1af1f08ad3e51bf09093 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 12 Dec 2023 14:41:43 +0800 Subject: [PATCH 112/637] feat: add WriteCodeWithUDFs. --- metagpt/actions/write_analysis_code.py | 24 +++++++++++++++++++++++- 1 file changed, 23 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 1127dc78b..725c4aa2a 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -7,6 +7,7 @@ from typing import Dict, List, Union, Tuple from metagpt.actions import Action +from metagpt.llm import LLM from metagpt.logs import logger from metagpt.prompts.ml_engineer import ( TOOL_RECOMMENDATION_PROMPT, @@ -19,7 +20,7 @@ from metagpt.prompts.ml_engineer import ( ) from metagpt.schema import Message, Plan from metagpt.tools.functions import registry -from metagpt.utils.common import create_func_config +from metagpt.utils.common import create_func_config, CodeParser class BaseWriteAnalysisCode(Action): @@ -203,3 +204,24 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) return rsp["code"] + + +class WriteCodeWithUDFs(WriteCodeByGenerate): + """Write code with user defined function.""" + from metagpt.tools.functions.libs.udf import UDFS + + DEFAULT_SYSTEM_MSG = f"""Please remember these functions, you will use these functions to write code:\n + {UDFS} + """ + + async def aask_code_and_text(self, context: List[Dict], **kwargs) -> Tuple[str]: + rsp = await self.llm.acompletion(context, **kwargs) + rsp_content = self.llm.get_choice_text(rsp) + code = CodeParser.parse_code(None, rsp_content) + return code, rsp_content + + async def run(self, context: List[Message], plan: Plan = None, task_guide: str = "", **kwargs) -> str: + prompt = self.process_msg(context) + logger.info(prompt[-1]) + code, _ = await self.aask_code_and_text(prompt, **kwargs) + return code From 116e7718babf53904d0fb3a76b168d23fc1b46d7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 12 Dec 2023 14:42:29 +0800 Subject: [PATCH 113/637] add test_write_code_with_udfs. --- tests/metagpt/actions/test_write_analysis_code.py | 14 +++++++++++++- 1 file changed, 13 insertions(+), 1 deletion(-) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index 661202115..c3e7adc1b 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -1,7 +1,7 @@ import asyncio import pytest -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools, WriteCodeWithUDFs from metagpt.actions.execute_code import ExecutePyCode from metagpt.schema import Message, Plan, Task from metagpt.logs import logger @@ -311,3 +311,15 @@ async def test_write_code_reuse_code_long_for_wine(): success_rate = sum(success) / trials_num logger.info(f"success rate: {success_rate :.2f}") assert success_rate >= 0.8 + + +@pytest.mark.asyncio +async def test_write_code_with_udfs(): + wudf = WriteCodeWithUDFs() + ep = ExecutePyCode() + rsp = await wudf.run("Get Apple stock data for the past 90 days.") + logger.info(rsp) + assert 'metagpt' in rsp + output, output_type = await ep.run(rsp) + assert output_type is True + logger.info(output) From 9651cdd735bf82928f3ada3d299d0c442edbfd73 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 12 Dec 2023 14:45:06 +0800 Subject: [PATCH 114/637] update function_schema. --- metagpt/tools/functions/libs/udf/__init__.py | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index e44e97c41..c9c818a96 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -25,8 +25,9 @@ def extract_function_signatures(file_path): module_name = Path(file_path).parts[-1][:-len(Path(file_path).suffix)] module = importlib.import_module(f"metagpt.tools.functions.libs.udf.{module_name}") # 获取函数注释和函数路径 - function_schema = {'name': function_signature, 'doc': inspect.getdoc(getattr(module, function_name)), - 'path': f'from metagpt.tools.functions.libs.udf.{module_name} import {function_name}'} + function_schema = {'udf_name': function_signature, + 'udf_path': f'from metagpt.tools.functions.libs.udf.{module_name} import {function_name}', + 'udf_doc': inspect.getdoc(getattr(module, function_name))} function_signatures.append(function_schema) return function_signatures @@ -48,4 +49,4 @@ folder_path = str(Path(__file__).parent.absolute()) function_signatures = get_function_signatures_in_folder(folder_path) UDFS = [func for func in function_signatures - if not func['name'].startswith(('extract_function_signatures', 'get_function_signatures_in_folder'))] + if not func['udf_name'].startswith(('extract_function_signatures', 'get_function_signatures_in_folder'))] From 86e320be1187ef4738a8000e270cc69cdbf31030 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 12 Dec 2023 14:57:22 +0800 Subject: [PATCH 115/637] update for no_udf_found. --- metagpt/actions/write_analysis_code.py | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 725c4aa2a..663f76b7b 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -211,13 +211,16 @@ class WriteCodeWithUDFs(WriteCodeByGenerate): from metagpt.tools.functions.libs.udf import UDFS DEFAULT_SYSTEM_MSG = f"""Please remember these functions, you will use these functions to write code:\n - {UDFS} + {UDFS}, **Notice: 1. if no right udf for user requirement, please send `No udf found`** """ async def aask_code_and_text(self, context: List[Dict], **kwargs) -> Tuple[str]: rsp = await self.llm.acompletion(context, **kwargs) rsp_content = self.llm.get_choice_text(rsp) code = CodeParser.parse_code(None, rsp_content) + if code.startswith('No udf found') or rsp_content.startswith('No udf found'): + rsp_content = 'No udf found' + code = 'No udf found' return code, rsp_content async def run(self, context: List[Message], plan: Plan = None, task_guide: str = "", **kwargs) -> str: From 3fc5080b811f44e7ac6ff90458cd48a424c2ca50 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 12 Dec 2023 14:58:10 +0800 Subject: [PATCH 116/637] add test_write_code_with_udfs_no_udf_found. --- tests/metagpt/actions/test_write_analysis_code.py | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index c3e7adc1b..71628d439 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -323,3 +323,11 @@ async def test_write_code_with_udfs(): output, output_type = await ep.run(rsp) assert output_type is True logger.info(output) + + +@pytest.mark.asyncio +async def test_write_code_with_udfs_no_udf_found(): + wudf = WriteCodeWithUDFs() + rsp = await wudf.run("Identify if there is a dog in the picture.") + logger.info(rsp) + assert 'No udf found' in rsp From 1da4409475579705e5c7a44e1a873e337d02eb83 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Tue, 12 Dec 2023 16:10:05 +0800 Subject: [PATCH 117/637] add step plan --- metagpt/roles/ml_engineer.py | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 45fe728dd..8ad75b399 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -156,6 +156,11 @@ class MLEngineer(Role): # ask for acceptance, users can other refuse and change tasks in the plan task_result_confirmed = await self._ask_review() + # 针对当前task进行单独plan + if not success or not task_result_confirmed: + # fixme: 增加对应plan + self.state.plan() + if success and task_result_confirmed: # tick off this task and record progress task.code = code @@ -203,8 +208,6 @@ class MLEngineer(Role): if counter == 0: context = self.get_useful_memories() else: - # context = self.get_useful_memories() - # logger.info(f"context {context}") improve_code = await DebugCode().run(plan=self.plan.current_task.instruction, finished_code=code_context, finished_code_result=code_result, @@ -255,6 +258,8 @@ class MLEngineer(Role): counter += 1 + success = False + return code, result, success, code_steps async def _ask_review(self): From 0231cfdcc750f3366c3eee16fc776581f67cbaf6 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Tue, 12 Dec 2023 16:23:56 +0800 Subject: [PATCH 118/637] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E4=BB=A3=E7=A0=81?= =?UTF-8?q?=E6=96=87=E4=BB=B6=E8=BE=93=E5=87=BA=E4=BF=9D=E5=AD=98=EF=BC=8C?= =?UTF-8?q?=E5=88=9B=E5=BB=BA=E9=A1=B9=E7=9B=AE=E6=96=87=E4=BB=B6=E5=A4=B9?= =?UTF-8?q?=EF=BC=8C=E4=BD=BF=E7=94=A8=E9=A1=B9=E7=9B=AE=E6=96=87=E4=BB=B6?= =?UTF-8?q?=E5=A4=B9=E9=9A=94=E7=A6=BB=20=E5=AE=8C=E6=95=B4=E4=BB=A3?= =?UTF-8?q?=E7=A0=81=E4=BF=9D=E5=AD=98=E5=89=8D=EF=BC=8C=E5=8F=AF=E8=80=83?= =?UTF-8?q?=E8=99=91=E6=8B=BC=E6=8E=A5=E5=85=A8=E9=87=8F=E4=BB=A3=E7=A0=81?= =?UTF-8?q?=E5=86=8D=E8=BE=93=E5=87=BA?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/utils/save_code.py | 40 +++++++++++++++++++++++++++ tests/metagpt/utils/test_save_code.py | 30 ++++++++++++++++++++ 2 files changed, 70 insertions(+) create mode 100644 metagpt/utils/save_code.py create mode 100644 tests/metagpt/utils/test_save_code.py diff --git a/metagpt/utils/save_code.py b/metagpt/utils/save_code.py new file mode 100644 index 000000000..b0720a5cf --- /dev/null +++ b/metagpt/utils/save_code.py @@ -0,0 +1,40 @@ +# -*- coding: utf-8 -*- +# @Date : 12/12/2023 4:14 PM +# @Author : stellahong (stellahong@fuzhi.ai) +# @Desc : +import os +import json + +from metagpt.const import DATA_PATH + +def save_code_file(name: str, code_context: str, file_format: str = "py") -> None: + """ + Save code files to a specified path. + + Args: + - name (str): The name of the folder to save the files. + - code_context (str): The code content. + - file_format (str, optional): The file format, supports 'py' (Python file) and 'json' (JSON file). Default is 'py'. + + Returns: + - None + """ + # Create the folder path if it doesn't exist + os.makedirs(name=DATA_PATH / "output" / f"{name}", exist_ok=True) + + # Choose to save as a Python file or a JSON file based on the file format + file_path = DATA_PATH / "output" / f"{name}/code.{file_format}" + if file_format == "py": + with open(file_path, "w", encoding="utf-8") as fp: + fp.write(code_context + "\n\n") + elif file_format == "json": + # Parse the code content as JSON and save + data = {"code": code_context} + with open(file_path, "w", encoding="utf-8") as fp: + json.dump(data, fp, indent=2) + else: + raise ValueError("Unsupported file format. Please choose 'py' or 'json'.") + + + + diff --git a/tests/metagpt/utils/test_save_code.py b/tests/metagpt/utils/test_save_code.py new file mode 100644 index 000000000..33addb2bf --- /dev/null +++ b/tests/metagpt/utils/test_save_code.py @@ -0,0 +1,30 @@ +# -*- coding: utf-8 -*- +# @Date : 12/12/2023 4:17 PM +# @Author : stellahong (stellahong@fuzhi.ai) +# @Desc : +import os +import json + +from metagpt.utils.save_code import save_code_file, DATA_PATH + + +def test_save_code_file_python(): + save_code_file("example", "print('Hello, World!')") + file_path = DATA_PATH / "output" / "example" / "code.py" + assert os.path.exists(file_path), f"File does not exist: {file_path}" + + +def test_save_code_file_python(): + save_code_file("example", "print('Hello, World!')") + file_path = DATA_PATH / "output" / "example" / "code.py" + with open(file_path, "r", encoding="utf-8") as fp: + content = fp.read() + assert "print('Hello, World!')" in content, "File content does not match" + +def test_save_code_file_json(): + save_code_file("example_json", "print('Hello, JSON!')", file_format="json") + file_path = DATA_PATH / "output" / "example_json" / "code.json" + with open(file_path, "r", encoding="utf-8") as fp: + data = json.load(fp) + assert "code" in data, "JSON key 'code' is missing" + assert data["code"] == "print('Hello, JSON!')", "JSON content does not match" From 35c9d744a46b8f0ad75512ebf6bf51537de089a9 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Tue, 12 Dec 2023 16:29:35 +0800 Subject: [PATCH 119/637] update gitignore --- .gitignore | 1 + 1 file changed, 1 insertion(+) diff --git a/.gitignore b/.gitignore index e03eab3d3..d36fbb856 100644 --- a/.gitignore +++ b/.gitignore @@ -164,3 +164,4 @@ tmp output.wav metagpt/roles/idea_agent.py .aider* +/config/config.yaml From a4cef261e07b380bd55856bef752e380c82f238b Mon Sep 17 00:00:00 2001 From: stellahsr Date: Tue, 12 Dec 2023 17:17:40 +0800 Subject: [PATCH 120/637] =?UTF-8?q?update:=20=E6=B7=BB=E5=8A=A0nb=E6=94=AF?= =?UTF-8?q?=E6=8C=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 1 + metagpt/roles/ml_engineer.py | 2 +- metagpt/utils/save_code.py | 4 ++++ tests/metagpt/utils/test_save_code.py | 26 ++++++++++++++++++++++++++ 4 files changed, 32 insertions(+), 1 deletion(-) diff --git a/.gitignore b/.gitignore index d36fbb856..5f8e400e3 100644 --- a/.gitignore +++ b/.gitignore @@ -165,3 +165,4 @@ output.wav metagpt/roles/idea_agent.py .aider* /config/config.yaml +/tests/metagpt/actions/check_data.py diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index fe6f81841..08451ec89 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -93,7 +93,7 @@ class MLEngineer(Role): summary = await SummarizeAnalysis().run(self.plan) rsp = Message(content=summary, cause_by=SummarizeAnalysis) self._rc.memory.add(rsp) - + return rsp async def _write_and_exec_code(self, max_retry: int = 3): diff --git a/metagpt/utils/save_code.py b/metagpt/utils/save_code.py index b0720a5cf..f1fdf0403 100644 --- a/metagpt/utils/save_code.py +++ b/metagpt/utils/save_code.py @@ -5,6 +5,8 @@ import os import json +import nbformat + from metagpt.const import DATA_PATH def save_code_file(name: str, code_context: str, file_format: str = "py") -> None: @@ -32,6 +34,8 @@ def save_code_file(name: str, code_context: str, file_format: str = "py") -> Non data = {"code": code_context} with open(file_path, "w", encoding="utf-8") as fp: json.dump(data, fp, indent=2) + elif file_format == "ipynb": + nbformat.write(code_context, file_path) else: raise ValueError("Unsupported file format. Please choose 'py' or 'json'.") diff --git a/tests/metagpt/utils/test_save_code.py b/tests/metagpt/utils/test_save_code.py index 33addb2bf..60a9e1ff4 100644 --- a/tests/metagpt/utils/test_save_code.py +++ b/tests/metagpt/utils/test_save_code.py @@ -2,8 +2,13 @@ # @Date : 12/12/2023 4:17 PM # @Author : stellahong (stellahong@fuzhi.ai) # @Desc : +import pytest import os import json +import nbformat + +from metagpt.actions.write_analysis_code import WriteCodeByGenerate +from metagpt.actions.execute_code import ExecutePyCode from metagpt.utils.save_code import save_code_file, DATA_PATH @@ -21,6 +26,7 @@ def test_save_code_file_python(): content = fp.read() assert "print('Hello, World!')" in content, "File content does not match" + def test_save_code_file_json(): save_code_file("example_json", "print('Hello, JSON!')", file_format="json") file_path = DATA_PATH / "output" / "example_json" / "code.json" @@ -28,3 +34,23 @@ def test_save_code_file_json(): data = json.load(fp) assert "code" in data, "JSON key 'code' is missing" assert data["code"] == "print('Hello, JSON!')", "JSON content does not match" + + + +@pytest.mark.asyncio +async def test_save_code_file_notebook(): + code = await WriteCodeByGenerate().run( + context="basic python, hello world", plan="", code_steps="", temperature=0.0 + ) + executor = ExecutePyCode() + await executor.run(code) + # Save as a Notebook file + save_code_file("example_nb", executor.nb, file_format="ipynb") + file_path = DATA_PATH / "output" / "example_nb" / "code.ipynb" + assert os.path.exists(file_path), f"Notebook file does not exist: {file_path}" + + # Additional checks specific to notebook format + notebook = nbformat.read(file_path, as_version=4) + assert len(notebook.cells) > 0, "Notebook should have at least one cell" + first_cell_source = notebook.cells[0].source + assert "print('Hello, World!')" in first_cell_source, "Notebook cell content does not match" From 8db5f22105b344eeebbe7df2281f9f062fd8fa0a Mon Sep 17 00:00:00 2001 From: stellahsr Date: Tue, 12 Dec 2023 17:26:15 +0800 Subject: [PATCH 121/637] =?UTF-8?q?=E6=9B=B4=E6=96=B0=E6=B3=A8=E9=87=8A?= =?UTF-8?q?=E5=92=8C=E8=AF=B4=E6=98=8E?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/utils/save_code.py | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/metagpt/utils/save_code.py b/metagpt/utils/save_code.py index f1fdf0403..96c310336 100644 --- a/metagpt/utils/save_code.py +++ b/metagpt/utils/save_code.py @@ -16,7 +16,8 @@ def save_code_file(name: str, code_context: str, file_format: str = "py") -> Non Args: - name (str): The name of the folder to save the files. - code_context (str): The code content. - - file_format (str, optional): The file format, supports 'py' (Python file) and 'json' (JSON file). Default is 'py'. + - file_format (str, optional): The file format. Supports 'py' (Python file), 'json' (JSON file), and 'ipynb' (Jupyter Notebook file). Default is 'py'. + Returns: - None @@ -37,7 +38,7 @@ def save_code_file(name: str, code_context: str, file_format: str = "py") -> Non elif file_format == "ipynb": nbformat.write(code_context, file_path) else: - raise ValueError("Unsupported file format. Please choose 'py' or 'json'.") + raise ValueError("Unsupported file format. Please choose 'py', 'json', or 'ipynb'.") From 7c1809af1ef39f5cc134870d03b2e5603d885789 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 12 Dec 2023 22:35:06 +0800 Subject: [PATCH 122/637] support more forms of task generation --- metagpt/actions/write_plan.py | 10 +++++++++- metagpt/roles/ml_engineer.py | 21 +++++++++++++++------ 2 files changed, 24 insertions(+), 7 deletions(-) diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index f7ca1ff4c..11a3f3e1e 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -13,6 +13,7 @@ from metagpt.actions import Action from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_PROMPT, ASSIGN_TASK_TYPE from metagpt.schema import Message, Task, Plan from metagpt.utils.common import CodeParser, create_func_config +from metagpt.logs import logger class WritePlan(Action): @@ -22,6 +23,7 @@ class WritePlan(Action): # Task: Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to __max_tasks__ tasks. If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan. + If you encounter errors on the current task, revise and output the current single task only. Output a list of jsons following the format: ```json [ @@ -76,7 +78,13 @@ def rsp_to_tasks(rsp: str) -> List[Task]: def update_plan_from_rsp(rsp: str, current_plan: Plan): tasks = rsp_to_tasks(rsp) - if len(tasks) == 1: + if len(tasks) == 1 or tasks[0].dependent_task_ids: + if tasks[0].dependent_task_ids and len(tasks) > 1: + # tasks[0].dependent_task_ids means the generated tasks are not a complete plan + # for they depend on tasks in the current plan, in this case, we only support updating one task each time + logger.warning( + "Current plan will take only the first generated task if the generated tasks are not a complete plan" + ) # handle a single task if current_plan.has_task_id(tasks[0].task_id): # replace an existing task diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index fe6f81841..de649e857 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -28,7 +28,7 @@ class MLEngineer(Role): self.plan = Plan(goal=goal) self.use_tools = False - self.use_code_steps = True + self.use_code_steps = False self.execute_code = ExecutePyCode() self.auto_run = auto_run @@ -64,6 +64,11 @@ class MLEngineer(Role): # ask for acceptance, users can other refuse and change tasks in the plan review, task_result_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) + if self.auto_run: + # if human confirms the task result, then we deem the task completed, regardless of whether the code run succeeds; + # if auto mode, then the code run has to succeed for the task to be considered completed + task_result_confirmed = success + if task_result_confirmed: # tick off this task and record progress task.code = code @@ -143,7 +148,7 @@ class MLEngineer(Role): if not success and counter >= max_retry: logger.info("coding failed!") review, _ = await self._ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) - if ReviewConst.CHANGE_WORD in review: + if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions return code, result, success, code_steps @@ -199,9 +204,12 @@ class MLEngineer(Role): # TODO dataset description , code steps user_requirement = self.plan.goal data_desc = self.plan.context - tasks = json.dumps( - [task.dict() for task in self.plan.tasks], indent=4, ensure_ascii=False - ) + tasks = [task.dict() for task in self.plan.tasks] + for task in tasks: + # Shorten the context as we don't need code steps after we get the codes. + # This doesn't affect current_task below, which should hold the code steps + task.pop("code_steps") + tasks = json.dumps(tasks, indent=4, ensure_ascii=False) current_task = self.plan.current_task.json() if self.plan.current_task else {} context = STRUCTURAL_CONTEXT.format( user_requirement=user_requirement, data_desc=data_desc, tasks=tasks, current_task=current_task @@ -219,7 +227,8 @@ if __name__ == "__main__": # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" - requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" + # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" + requirement = "This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: 'workspace/house-prices-advanced-regression-techniques/split_train.csv', eval data path: 'workspace/house-prices-advanced-regression-techniques/split_eval.csv'." async def main(requirement: str = requirement, auto_run: bool = False): role = MLEngineer(goal=requirement, auto_run=auto_run) From 0147e0bb534ab487dcbdbc52cce938c62893f4be Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 13 Dec 2023 10:29:50 +0800 Subject: [PATCH 123/637] add ignore --- .gitignore | 1 + config/config.yaml | 100 --------------------------------------------- 2 files changed, 1 insertion(+), 100 deletions(-) delete mode 100644 config/config.yaml diff --git a/.gitignore b/.gitignore index 5f8e400e3..f79581de4 100644 --- a/.gitignore +++ b/.gitignore @@ -166,3 +166,4 @@ metagpt/roles/idea_agent.py .aider* /config/config.yaml /tests/metagpt/actions/check_data.py +/config/config.yaml diff --git a/config/config.yaml b/config/config.yaml deleted file mode 100644 index bf998def7..000000000 --- a/config/config.yaml +++ /dev/null @@ -1,100 +0,0 @@ -# DO NOT MODIFY THIS FILE, create a new key.yaml, define OPENAI_API_KEY. -# The configuration of key.yaml has a higher priority and will not enter git - -#### if OpenAI -## The official OPENAI_API_BASE is https://api.openai.com/v1 -## If the official OPENAI_API_BASE is not available, we recommend using the [openai-forward](https://github.com/beidongjiedeguang/openai-forward). -## Or, you can configure OPENAI_PROXY to access official OPENAI_API_BASE. -OPENAI_API_BASE: "https://api.openai.com/v1" -#OPENAI_PROXY: "http://127.0.0.1:8118" -#OPENAI_API_KEY: "YOUR_API_KEY" # set the value to sk-xxx if you host the openai interface for open llm model -OPENAI_API_MODEL: "gpt-4" -MAX_TOKENS: 1500 -RPM: 10 - -#### if Spark -#SPARK_APPID : "YOUR_APPID" -#SPARK_API_SECRET : "YOUR_APISecret" -#SPARK_API_KEY : "YOUR_APIKey" -#DOMAIN : "generalv2" -#SPARK_URL : "ws://spark-api.xf-yun.com/v2.1/chat" - -#### if Anthropic -#Anthropic_API_KEY: "YOUR_API_KEY" - -#### if AZURE, check https://github.com/openai/openai-cookbook/blob/main/examples/azure/chat.ipynb -#### You can use ENGINE or DEPLOYMENT mode -#OPENAI_API_TYPE: "azure" -#OPENAI_API_BASE: "YOUR_AZURE_ENDPOINT" -#OPENAI_API_KEY: "YOUR_AZURE_API_KEY" -#OPENAI_API_VERSION: "YOUR_AZURE_API_VERSION" -#DEPLOYMENT_NAME: "YOUR_DEPLOYMENT_NAME" -#DEPLOYMENT_ID: "YOUR_DEPLOYMENT_ID" - -#### if zhipuai from `https://open.bigmodel.cn`. You can set here or export API_KEY="YOUR_API_KEY" -# ZHIPUAI_API_KEY: "YOUR_API_KEY" - -#### for Search - -## Supported values: serpapi/google/serper/ddg -#SEARCH_ENGINE: serpapi - -## Visit https://serpapi.com/ to get key. -#SERPAPI_API_KEY: "YOUR_API_KEY" - -## Visit https://console.cloud.google.com/apis/credentials to get key. -#GOOGLE_API_KEY: "YOUR_API_KEY" -## Visit https://programmablesearchengine.google.com/controlpanel/create to get id. -#GOOGLE_CSE_ID: "YOUR_CSE_ID" - -## Visit https://serper.dev/ to get key. -#SERPER_API_KEY: "YOUR_API_KEY" - -#### for web access - -## Supported values: playwright/selenium -#WEB_BROWSER_ENGINE: playwright - -## Supported values: chromium/firefox/webkit, visit https://playwright.dev/python/docs/api/class-browsertype -##PLAYWRIGHT_BROWSER_TYPE: chromium - -## Supported values: chrome/firefox/edge/ie, visit https://www.selenium.dev/documentation/webdriver/browsers/ -# SELENIUM_BROWSER_TYPE: chrome - -#### for TTS - -#AZURE_TTS_SUBSCRIPTION_KEY: "YOUR_API_KEY" -#AZURE_TTS_REGION: "eastus" - -#### for Stable Diffusion -## Use SD service, based on https://github.com/AUTOMATIC1111/stable-diffusion-webui -SD_URL: "YOUR_SD_URL" -SD_T2I_API: "/sdapi/v1/txt2img" - -#### for Execution -#LONG_TERM_MEMORY: false - -#### for Mermaid CLI -## If you installed mmdc (Mermaid CLI) only for metagpt then enable the following configuration. -#PUPPETEER_CONFIG: "./config/puppeteer-config.json" -#MMDC: "./node_modules/.bin/mmdc" - - -### for calc_usage -# CALC_USAGE: false - -### for Research -MODEL_FOR_RESEARCHER_SUMMARY: gpt-3.5-turbo -MODEL_FOR_RESEARCHER_REPORT: gpt-3.5-turbo-16k - -### choose the engine for mermaid conversion, -# default is nodejs, you can change it to playwright,pyppeteer or ink -# MERMAID_ENGINE: nodejs - -### browser path for pyppeteer engine, support Chrome, Chromium,MS Edge -#PYPPETEER_EXECUTABLE_PATH: "/usr/bin/google-chrome-stable" - -PROMPT_FORMAT: json #json or markdown - -# KAGGLE_USERNAME: "" -# KAGGLE_KEY: "" \ No newline at end of file From 32c4a557556a6e23afa18ea1a316169cd858e7dd Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 13 Dec 2023 12:54:50 +0800 Subject: [PATCH 124/637] add save code --- metagpt/roles/ml_engineer.py | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 08451ec89..d679b2e01 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,13 +1,11 @@ -from typing import Dict, List, Union +from typing import List import json -import subprocess +from datetime import datetime import fire -import re from metagpt.roles import Role -from metagpt.actions import Action -from metagpt.schema import Message, Task, Plan +from metagpt.schema import Message, Plan from metagpt.memory import Memory from metagpt.logs import logger from metagpt.actions.write_plan import WritePlan, update_plan_from_rsp, precheck_update_plan_from_rsp @@ -17,6 +15,7 @@ from metagpt.actions.execute_code import ExecutePyCode from metagpt.roles.kaggle_manager import DownloadData, SubmitResult from metagpt.prompts.ml_engineer import STRUCTURAL_CONTEXT from metagpt.actions.write_code_steps import WriteCodeSteps +from metagpt.utils.save_code import save_code_file class MLEngineer(Role): def __init__( @@ -93,7 +92,10 @@ class MLEngineer(Role): summary = await SummarizeAnalysis().run(self.plan) rsp = Message(content=summary, cause_by=SummarizeAnalysis) self._rc.memory.add(rsp) - + + # save code using datetime.now or keywords related to the goal of your project (plan.goal). + project_record = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") + save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") return rsp async def _write_and_exec_code(self, max_retry: int = 3): From 2e4094c7a798f15f42ec3d85fc87395e4260d352 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 13 Dec 2023 12:56:54 +0800 Subject: [PATCH 125/637] test auto mode --- .gitignore | 1 - metagpt/roles/ml_engineer.py | 2 +- 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/.gitignore b/.gitignore index f79581de4..5f8e400e3 100644 --- a/.gitignore +++ b/.gitignore @@ -166,4 +166,3 @@ metagpt/roles/idea_agent.py .aider* /config/config.yaml /tests/metagpt/actions/check_data.py -/config/config.yaml diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index d679b2e01..8b7b72517 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -223,7 +223,7 @@ if __name__ == "__main__": # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" - async def main(requirement: str = requirement, auto_run: bool = False): + async def main(requirement: str = requirement, auto_run: bool = True): role = MLEngineer(goal=requirement, auto_run=auto_run) await role.run(requirement) From f81f355ff24378701c17de6d0c7260ad649fbf54 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 13 Dec 2023 13:01:32 +0800 Subject: [PATCH 126/637] add default config.yaml --- .gitignore | 1 - config/config.yaml | 97 ++++++++++++++++++++++++++++++++++++ metagpt/roles/ml_engineer.py | 4 +- 3 files changed, 99 insertions(+), 3 deletions(-) create mode 100644 config/config.yaml diff --git a/.gitignore b/.gitignore index 5f8e400e3..9b679d48a 100644 --- a/.gitignore +++ b/.gitignore @@ -164,5 +164,4 @@ tmp output.wav metagpt/roles/idea_agent.py .aider* -/config/config.yaml /tests/metagpt/actions/check_data.py diff --git a/config/config.yaml b/config/config.yaml new file mode 100644 index 000000000..bed67083c --- /dev/null +++ b/config/config.yaml @@ -0,0 +1,97 @@ +# DO NOT MODIFY THIS FILE, create a new key.yaml, define OPENAI_API_KEY. +# The configuration of key.yaml has a higher priority and will not enter git + +#### if OpenAI +## The official OPENAI_API_BASE is https://api.openai.com/v1 +## If the official OPENAI_API_BASE is not available, we recommend using the [openai-forward](https://github.com/beidongjiedeguang/openai-forward). +## Or, you can configure OPENAI_PROXY to access official OPENAI_API_BASE. +OPENAI_API_BASE: "https://api.openai.com/v1" +#OPENAI_PROXY: "http://127.0.0.1:8118" +#OPENAI_API_KEY: "YOUR_API_KEY" # set the value to sk-xxx if you host the openai interface for open llm model +OPENAI_API_MODEL: "gpt-4" +MAX_TOKENS: 1500 +RPM: 10 + +#### if Spark +#SPARK_APPID : "YOUR_APPID" +#SPARK_API_SECRET : "YOUR_APISecret" +#SPARK_API_KEY : "YOUR_APIKey" +#DOMAIN : "generalv2" +#SPARK_URL : "ws://spark-api.xf-yun.com/v2.1/chat" + +#### if Anthropic +#Anthropic_API_KEY: "YOUR_API_KEY" + +#### if AZURE, check https://github.com/openai/openai-cookbook/blob/main/examples/azure/chat.ipynb +#### You can use ENGINE or DEPLOYMENT mode +#OPENAI_API_TYPE: "azure" +#OPENAI_API_BASE: "YOUR_AZURE_ENDPOINT" +#OPENAI_API_KEY: "YOUR_AZURE_API_KEY" +#OPENAI_API_VERSION: "YOUR_AZURE_API_VERSION" +#DEPLOYMENT_NAME: "YOUR_DEPLOYMENT_NAME" +#DEPLOYMENT_ID: "YOUR_DEPLOYMENT_ID" + +#### if zhipuai from `https://open.bigmodel.cn`. You can set here or export API_KEY="YOUR_API_KEY" +# ZHIPUAI_API_KEY: "YOUR_API_KEY" + +#### for Search + +## Supported values: serpapi/google/serper/ddg +#SEARCH_ENGINE: serpapi + +## Visit https://serpapi.com/ to get key. +#SERPAPI_API_KEY: "YOUR_API_KEY" + +## Visit https://console.cloud.google.com/apis/credentials to get key. +#GOOGLE_API_KEY: "YOUR_API_KEY" +## Visit https://programmablesearchengine.google.com/controlpanel/create to get id. +#GOOGLE_CSE_ID: "YOUR_CSE_ID" + +## Visit https://serper.dev/ to get key. +#SERPER_API_KEY: "YOUR_API_KEY" + +#### for web access + +## Supported values: playwright/selenium +#WEB_BROWSER_ENGINE: playwright + +## Supported values: chromium/firefox/webkit, visit https://playwright.dev/python/docs/api/class-browsertype +##PLAYWRIGHT_BROWSER_TYPE: chromium + +## Supported values: chrome/firefox/edge/ie, visit https://www.selenium.dev/documentation/webdriver/browsers/ +# SELENIUM_BROWSER_TYPE: chrome + +#### for TTS + +#AZURE_TTS_SUBSCRIPTION_KEY: "YOUR_API_KEY" +#AZURE_TTS_REGION: "eastus" + +#### for Stable Diffusion +## Use SD service, based on https://github.com/AUTOMATIC1111/stable-diffusion-webui +SD_URL: "YOUR_SD_URL" +SD_T2I_API: "/sdapi/v1/txt2img" + +#### for Execution +#LONG_TERM_MEMORY: false + +#### for Mermaid CLI +## If you installed mmdc (Mermaid CLI) only for metagpt then enable the following configuration. +#PUPPETEER_CONFIG: "./config/puppeteer-config.json" +#MMDC: "./node_modules/.bin/mmdc" + + +### for calc_usage +# CALC_USAGE: false + +### for Research +MODEL_FOR_RESEARCHER_SUMMARY: gpt-3.5-turbo +MODEL_FOR_RESEARCHER_REPORT: gpt-3.5-turbo-16k + +### choose the engine for mermaid conversion, +# default is nodejs, you can change it to playwright,pyppeteer or ink +# MERMAID_ENGINE: nodejs + +### browser path for pyppeteer engine, support Chrome, Chromium,MS Edge +#PYPPETEER_EXECUTABLE_PATH: "/usr/bin/google-chrome-stable" + +PROMPT_FORMAT: json #json or markdown \ No newline at end of file diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 8b7b72517..c3f1bd669 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -217,11 +217,11 @@ class MLEngineer(Role): if __name__ == "__main__": - # requirement = "Run data analysis on sklearn Iris dataset, include a plot" + requirement = "Run data analysis on sklearn Iris dataset, include a plot" # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" - requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" + # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" async def main(requirement: str = requirement, auto_run: bool = True): role = MLEngineer(goal=requirement, auto_run=auto_run) From 49779d8615e4b05b759b549b6d7ceb9b5258ec0a Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 13:35:22 +0800 Subject: [PATCH 127/637] refine schema desc --- metagpt/tools/functions/schemas/feature_engineering.yml | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/metagpt/tools/functions/schemas/feature_engineering.yml b/metagpt/tools/functions/schemas/feature_engineering.yml index 2cc4ec2fa..4f2a7100d 100644 --- a/metagpt/tools/functions/schemas/feature_engineering.yml +++ b/metagpt/tools/functions/schemas/feature_engineering.yml @@ -328,7 +328,7 @@ GroupStat: SplitBins: type: class - description: "Bin continuous data into intervals and return the bin identifier encoded as an integer value" + description: "Inplace binning of continuous data into intervals, returning integer-encoded bin identifiers directly." methods: __init__: description: "Initialize self." @@ -336,11 +336,15 @@ SplitBins: properties: cols: type: list - description: "Columns to be binned." + description: "Columns to be binned inplace." strategy: type: str description: "Strategy used to define the widths of the bins." default: quantile + enum: + - quantile + - uniform + - kmeans required: - cols fit: From f614fbfa7c3e22e968bc4229271df092c3be9575 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 13:37:40 +0800 Subject: [PATCH 128/637] update ml tools --- metagpt/tools/functions/libs/data_preprocess.py | 16 +++------------- .../tools/functions/libs/feature_engineering.py | 5 +++++ 2 files changed, 8 insertions(+), 13 deletions(-) diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index 39474b0fd..fa70bf8fc 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -1,6 +1,6 @@ import numpy as np from sklearn.impute import SimpleImputer -from sklearn.preprocessing import KBinsDiscretizer, LabelEncoder +from sklearn.preprocessing import LabelEncoder from sklearn.preprocessing import MaxAbsScaler from sklearn.preprocessing import MinMaxScaler from sklearn.preprocessing import OneHotEncoder @@ -8,7 +8,6 @@ from sklearn.preprocessing import OrdinalEncoder from sklearn.preprocessing import RobustScaler from sklearn.preprocessing import StandardScaler -from metagpt.tools.functions import registry from metagpt.tools.functions.libs.base import MLProcess from metagpt.tools.functions.schemas.data_preprocess import * @@ -57,15 +56,6 @@ class StandardScale(MLProcess): return df -@registry.register("data_preprocess", LogTransform) -def log_transform(df: pd.DataFrame, features: list, ): - for col in features: - if df[col].min() <= 0: - df[col] = df[col] - df[col].min() + 2 - df[col] = np.log(df[col]) - return df - - class MaxAbsScale(MLProcess): def __init__(self, features: list,): self.features = features @@ -146,7 +136,7 @@ class LabelEncode(MLProcess): return df -def get_column_info(df: pd.DataFrame) -> str: +def get_column_info(df: pd.DataFrame) -> dict: data = [] for i in df.columns: nan_freq = float("%.2g" % (df[i].isna().mean() * 100)) @@ -157,7 +147,7 @@ def get_column_info(df: pd.DataFrame) -> str: data, columns=["Column_name", "Data_type", "NaN_Frequency(%)", "N_unique"], ) - return samples.to_string(index=False) + return samples.to_dict(orient='list') # # # if __name__ == '__main__': diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index 67247d0d1..de54e4db0 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -10,6 +10,7 @@ import numpy as np from dateutil.relativedelta import relativedelta from joblib import Parallel, delayed from pandas.api.types import is_numeric_dtype +from pandas.core.dtypes.common import is_object_dtype from sklearn.model_selection import KFold from sklearn.preprocessing import PolynomialFeatures, KBinsDiscretizer @@ -280,6 +281,10 @@ class GeneralSelection(MLProcess): or df.loc[df[col] == np.inf].shape[0] != 0 ): feats.remove(col) + + if is_object_dtype(df[col]) and df[col].nunique() == df.shape[0]: + feats.remove(col) + self.feats = feats def transform(self, df: pd.DataFrame) -> pd.DataFrame: From 92d59ea31bb7bcb563d2fdd94cd6b6af64963aa7 Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 13 Dec 2023 13:48:18 +0800 Subject: [PATCH 129/637] save code steps early --- metagpt/actions/write_analysis_code.py | 7 ++----- metagpt/roles/ml_engineer.py | 11 +++++------ metagpt/schema.py | 2 +- 3 files changed, 8 insertions(+), 12 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 1127dc78b..7e6483371 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -23,9 +23,7 @@ from metagpt.utils.common import create_func_config class BaseWriteAnalysisCode(Action): - async def run( - self, context: List[Message], plan: Plan = None, task_guide: str = "" - ) -> str: + async def run(self, context: List[Message], plan: Plan = None) -> str: """Run of a code writing action, used in data analysis or modeling Args: @@ -85,7 +83,6 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): self, context: [List[Message]], plan: Plan = None, - code_steps: str = "", system_msg: str = None, **kwargs, ) -> str: @@ -155,11 +152,11 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): self, context: List[Message], plan: Plan = None, - code_steps: str = "", data_desc: str = "", ) -> str: task_type = plan.current_task.task_type task = plan.current_task.instruction + code_steps = plan.current_task.code_steps available_tools = registry.get_all_schema_by_module(task_type) available_tools = [ {k: tool[k] for k in ["name", "description"] if k in tool} diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index de649e857..3260dd43f 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -59,7 +59,7 @@ class MLEngineer(Role): logger.info(f"ready to take on task {task}") # take on current task - code, result, success, code_steps = await self._write_and_exec_code() + code, result, success = await self._write_and_exec_code() # ask for acceptance, users can other refuse and change tasks in the plan review, task_result_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) @@ -73,7 +73,6 @@ class MLEngineer(Role): # tick off this task and record progress task.code = code task.result = result - task.code_steps = code_steps self.plan.finish_current_task() self.working_memory.clear() @@ -102,7 +101,7 @@ class MLEngineer(Role): return rsp async def _write_and_exec_code(self, max_retry: int = 3): - code_steps = ( + self.plan.current_task.code_steps = ( await WriteCodeSteps().run(self.plan) if self.use_code_steps else "" @@ -121,12 +120,12 @@ class MLEngineer(Role): if not self.use_tools or self.plan.current_task.task_type == "other": # code = "print('abc')" code = await WriteCodeByGenerate().run( - context=context, plan=self.plan, code_steps=code_steps, temperature=0.0 + context=context, plan=self.plan, temperature=0.0 ) cause_by = WriteCodeByGenerate else: code = await WriteCodeWithTools().run( - context=context, plan=self.plan, code_steps=code_steps, data_desc="" + context=context, plan=self.plan, data_desc="" ) cause_by = WriteCodeWithTools @@ -151,7 +150,7 @@ class MLEngineer(Role): if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions - return code, result, success, code_steps + return code, result, success async def _ask_review(self, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): auto_run = auto_run or self.auto_run diff --git a/metagpt/schema.py b/metagpt/schema.py index f91922535..8eb7e31ca 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -78,10 +78,10 @@ class Task(BaseModel): dependent_task_ids: list[str] = [] # Tasks prerequisite to this Task instruction: str = "" task_type: str = "" + code_steps: str = "" code: str = "" result: str = "" is_finished: bool = False - code_steps: str = "" class Plan(BaseModel): From 33810829a072467a8f61f2f7dc14ffd1792e793a Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 14:31:32 +0800 Subject: [PATCH 130/637] support tool in debug --- metagpt/actions/debug_code.py | 106 ++++++++++++++++++++++------------ 1 file changed, 68 insertions(+), 38 deletions(-) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 9efe93efc..53ca2f54d 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -3,7 +3,7 @@ from typing import Dict, List, Union, Tuple, Optional, Any from metagpt.actions import Action from metagpt.logs import logger from metagpt.schema import Message, Plan -from metagpt.utils.common import CodeParser +from metagpt.utils.common import CodeParser, create_func_config from metagpt.actions.write_analysis_code import BaseWriteAnalysisCode DEBUG_REFLECTION_EXAMPLE = '''Example 1: @@ -39,25 +39,39 @@ DEBUG_REFLECTION_EXAMPLE = '''Example 1: REFLECTION_PROMPT = """ Here is an example for you. {debug_example} - [requirement] - {goal} - [finished code] - finished code are executable, and you should based on the code to continue your current code debug - {finished_code} - - try to reuse the code here to understand the coding task. + [context] + {context} [previous impl] {code} [runtime Error] {runtime_result} - Analysis the error step by step, provide me improve method. Do not repeat [previous impl] + Analysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. [reflection on previous impl]: xxx """ +CODE_REFLECTION = { + "name": "execute_reflection_code", + "description": "Execute reflection code.", + "parameters": { + "type": "object", + "properties": { + "reflection": { + "type": "string", + "description": "Reflection on previous impl.", + }, + "improved_impl": { + "type": "string", + "description": "Refined code after reflection.", + }, + }, + "required": ["reflection", "improved_impl"], + }, +} + def message_to_str(message: Message) -> str: return f"{message.role}: {message.content}" @@ -75,52 +89,68 @@ class DebugCode(BaseWriteAnalysisCode): def __init__(self, **kwargs: Any): super().__init__(**kwargs) - async def run_reflection(self, goal, finished_code, finished_code_result, code, runtime_result) -> str: + async def run_reflection( + self, + # goal, + # finished_code, + # finished_code_result, + context: List[Message], + code, + runtime_result, + ) -> dict: info = [] - finished_code_and_result = finished_code + "\n [finished results]\n\n" + finished_code_result + # finished_code_and_result = finished_code + "\n [finished results]\n\n" + finished_code_result reflection_prompt = REFLECTION_PROMPT.format(debug_example=DEBUG_REFLECTION_EXAMPLE, - goal=goal, - finished_code=finished_code_and_result, + context=context, + # goal=goal, + # finished_code=finished_code_and_result, code=code, runtime_result=runtime_result ) - system_prompt = "You are an AI Python assistant. You will be given your previous implementation of a function, runtime error results, and a hint to change the implementation appropriately. Write your full implementation " + system_prompt = "You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation " info.append(Message(role="system", content=system_prompt)) - info.append(Message(role="assistant", content=reflection_prompt)) + info.append(Message(role="user", content=reflection_prompt)) - msg = messages_to_str(info) - resp = await self.llm.aask(msg=msg) + # msg = messages_to_str(info) + # resp = await self.llm.aask(msg=msg) + resp = await self.llm.aask_code(messages=info, **create_func_config(CODE_REFLECTION)) logger.info(f"reflection is {resp}") return resp - async def rewrite_code(self, reflection: str = "", code_context: str = "") -> str: - """ - 根据reflection重写代码 - """ - info = [] - info.append(Message(role="assistant", content=f"[code context]:{code_context}" - f"finished code are executable, and you should based on the code to continue your current code debug and improvement" - f"[reflection]: \n {reflection}")) - info.append(Message(role="user", content=f"[improved impl]:\n Return in Python block")) - msg = messages_to_str(info) - resp = await self.llm.aask(msg=msg) - logger.info(f"improve code is {resp}") - improv_code = CodeParser.parse_code(block=None, text=resp) - return improv_code + # async def rewrite_code(self, reflection: str = "", context: List[Message] = None) -> str: + # """ + # 根据reflection重写代码 + # """ + # info = context + # # info.append(Message(role="assistant", content=f"[code context]:{code_context}" + # # f"finished code are executable, and you should based on the code to continue your current code debug and improvement" + # # f"[reflection]: \n {reflection}")) + # info.append(Message(role="assistant", content=f"[reflection]: \n {reflection}")) + # info.append(Message(role="user", content=f"[improved impl]:\n Return in Python block")) + # msg = messages_to_str(info) + # resp = await self.llm.aask(msg=msg) + # improv_code = CodeParser.parse_code(block=None, text=resp) + # return improv_code async def run(self, + context: List[Message] = None, plan: str = "", - finished_code: str = "", - finished_code_result: str = "", + # finished_code: str = "", + # finished_code_result: str = "", code: str = "", runtime_result: str = "") -> str: """ 根据当前运行代码和报错信息进行reflection和纠错 """ - reflection = await self.run_reflection(plan, finished_code=finished_code, - finished_code_result=finished_code_result, - code=code, - runtime_result=runtime_result) + reflection = await self.run_reflection( + # plan, + # finished_code=finished_code, + # finished_code_result=finished_code_result, + code=code, + context=context, + runtime_result=runtime_result, + ) # 根据reflection结果重写代码 - improv_code = await self.rewrite_code(reflection, code_context=finished_code) + # improv_code = await self.rewrite_code(reflection, context=context) + improv_code = reflection['improved_impl'] return improv_code From ab7af7768c00acc0c3f900430b402c64637f7b0f Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 14:31:50 +0800 Subject: [PATCH 131/637] refine prompt --- metagpt/prompts/ml_engineer.py | 298 +++++++++++++-------------------- 1 file changed, 117 insertions(+), 181 deletions(-) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 5c7b9f82e..d11cbf453 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -4,6 +4,31 @@ # @Author : lidanyang # @File : ml_engineer # @Desc : +UPDATE_DATA_COLUMNS = """ +# Background +Keep dataset column information updated to reflect changes in training or testing datasets, aiding in informed decision-making during data analysis. +## Done Tasks +```python +{history_code} +```end + +# Task +Update and print the dataset's column information only if the train or test data has changed. Use the following code: +```python +from metagpt.tools.functions.libs.data_preprocess import get_column_info + +column_info = get_column_info(df) +print("df_column_info") +print(column_info) +```end + +# Constraints: +- Use the DataFrame variable from 'Done Tasks' in place of df. +- Import `get_column_info` only if it's not already imported. +- Skip update if no changes in training/testing data, except for initial data load. +- No need to update info if only model evaluation is performed. +""" + GEN_DATA_DESC_PROMPT = """ Here is the head 5 rows of the dataset: {data_head} @@ -34,7 +59,8 @@ Please assign a task type to each task in the list below from the given categori - **feature_engineering**: Only for creating new columns for input data. - **data_preprocess**: Only for changing value inplace. - **model_train**: Only for training model. -- **other**: Any tasks that do not fit into the previous categories, such as visualization, summarizing findings, build model, etc. +- **model_evaluate**: Only for evaluating model. +- **other**: Any tasks that do not fit into the previous categories, such as visualization, summarizing findings, etc. """ ASSIGN_TASK_TYPE = { @@ -107,206 +133,122 @@ CODE_GENERATOR_WITH_TOOLS = { }, } -TOOL_USAGE_PROMPT = """ -## Target -{goal} -Specifically, {special_prompt} - -## History Info -{context} - -## Code Steps for Current Task: -Follow steps below when you writing code if it's convenient. -{code_steps} - -## Available Tools: -Each function is described in JSON format, including the function name and parameters. {output_desc} -{function_catalog} - -When you call a function above, you should import the function from `{module_name}` first, e.g.: -```python -from metagpt.tools.functions.libs.data_preprocess import fill_missing_value -```end - -## Your Output Format: -Generate the complete code for this task: -```python -# Tools used: [function names or 'none'] - -```end - -## Attention: -Make sure use the columns from the dataset columns: {column_names} -Finish your coding tasks as a helpful programmer based on the tools. - -""" +PRINT_DATA_COLUMNS = { + "name": "print_column_info", + "description": "Print the latest column information after 'Done Tasks' code if first read or data changed.", + "parameters": { + "type": "object", + "properties": { + "is_update": { + "type": "boolean", + "description": "Whether need to update the column info.", + }, + "code": { + "type": "string", + "description": "The code to be added to a new cell in jupyter.", + }, + }, + "required": ["is_update", "code"], + }, +} GENERATE_CODE_PROMPT = """ -## Target -{goal} - -Specifically, {special_prompt} - - -## Finished Task and Code -{context} - -## Code Steps for Current Task: -Follow steps below when you writing code if it's convenient. -{code_steps} - -## Instruction -Finished task and code are executable, and you should based on the code to continue your current task -Do not repeat functions and code, try to reuse the code in [Finished Task and Code] - -## Your Output Format: -Generate the complete code for this task: -```python -import pandas as pd - -``` - -## Attention: -Make sure use the columns from the dataset columns -Finish your coding tasks as a helpful programmer based on the code. - -""" - -TOOL_USAGE_PROMPT = """ -## Target -{goal} - -## History Info -{context} - -## Available Tools: -Each function is described in JSON format, including the function name and parameters. {output_desc} -{function_catalog} - -When you call a function above, you should import the function from `{module_name}` first, e.g.: -```python -from metagpt.tools.functions.libs.data_preprocess import fill_missing_value -```end - -## Your Output Format: -Generate the complete code for this task: -```python -# Tools used: [function names or 'none'] - -```end - -## Attention: -Make sure use the columns from the dataset columns -Finish your coding tasks as a helpful programmer based on the tools. -""" - -TOOL_ORGANIZATION_PROMPT = """ -The previous conversation has provided all tasks step-by-step for the use goal and their statuses. -Now, begin writing code for the current task. This code should writen strictly on the basis of all previous completed tasks code, not a standalone code. And avoid writing duplicate code that has already been written in previous tasks, such as repeated import of packages, reading data, etc. -Specifically, {special_prompt} -You can utilize pre-defined tools in 'Available Tools' if the tools are sufficient. And you should combine the use of other public packages if necessary, like sklearn, numpy, pandas, etc.. - -## Code Steps for Current Task: -Follow steps below when you writing code if it's convenient. -{code_steps} - -## Available Tools: -Each function is described in JSON format, including the function name and parameters. {output_desc} -{function_catalog} - -When you call a function above, you should import the function from `{module_name}` first, e.g.: -```python -from metagpt.tools.functions.libs.data_preprocess import fill_missing_value -```end - -## Your Output Format: -Generate the complete code for this task: -```python -# Tools used: [function names or 'none'] - -```end - -*** Important Rules *** -- If you use tool not in the list, you should implement it by yourself. -- Ensure the output new code is executable in the same Jupyter notebook environment with previous tasks code have been executed. -- When write code for current task, remember the code should be coherent with previous tasks code. -- Remember that don't process the columns have been processed in previous tasks and don't mock data yourself. -- Prioritize using tools for the same functionality. -""" - -DATA_PREPROCESS_PROMPT = """ -The current task is about data preprocessing, closely monitor each column's data type. Apply suitable methods for various types (numerical, categorical, datetime, textual, etc.) to ensure the pandas.DataFrame is correctly formatted. -Additionally, ensure that the columns being processed must be the ones that actually exist in the dataset. -Don't write processed data to files. -""" - -FEATURE_ENGINEERING_PROMPT = """ -The current task is about feature engineering. when performing it, please adhere to the following principles: -- Ensure that the feature you're working with is indeed present in the dataset and consider the data type (numerical, categorical, etc.) and application scenario (classification, regression tasks, etc.). -- When generate new features, you should combine real world knowledge and decide what features are useful for the task. -- Generate as diverse features as possible to improve the model's performance. -- Before generating a new feature, ensure the used features are already processed and ready to use. -""" - -DATA_PROCESS_PROMPT = """ # Background -As a data scientist, you need to help user to achieve the goal [{user_requirement}] step-by-step in an continuous Jupyter notebook. +Assist in completing [{user_requirement}] in a Jupyter notebook. -## Done Tasks +## Task Progress +### Done Tasks ```python {history_code} ```end -## Current Task +### Current Task {current_task} -# Latest Data Info -Latest data info after previous tasks: +## Latest Data Info {column_info} # Task -Write a Python function for 'Current Task'. Start by copying the input DataFrame. Avoid duplicating code from 'Done Tasks'. -Specifically, {special_prompt} +Fully implement 'Current Task', ensuring all necessary steps are covered without repeating code from 'Done Tasks'. Specifically, {special_prompt} + +# Code Steps: +Follow steps below when you writing code if it's convenient. +{code_steps} +""" + +TOOL_USAGE_PROMPT = """ +# Background +Assist in completing [{user_requirement}] in a Jupyter notebook. + +## Task Progress +### Done Tasks +```python +{history_code} +```end + +### Current Task +{current_task} + +## Latest Data Info +{column_info} + +# Task +Fully implement 'Current Task', ensuring all necessary steps are covered without repeating code from 'Done Tasks'. Specifically, {special_prompt} # Code Steps: Follow steps below when you writing code if it's convenient. {code_steps} # Capabilities -- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of python functions. +- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class. - You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. -- You can do anything about data preprocessing, feature engineering, model training, etc.. # Available Tools: -Each function tool is described in JSON format. {output_desc} -When you call a function below, import the function from `{module_name}` first. -{function_catalog} +Each Class tool is described in JSON format. When you call it, import the tool from `{module_name}` first. +{tool_catalog} # Output Example: -when current task is "fill missing value and handle outliers", the output code be like: +For "fill missing value and handle outliers", the output code be like when there are training data and test data: ```python -from metagpt.tools.functions.libs.data_preprocess import fill_missing_value +# Tools used: ['FillMissingValue'] +from metagpt.tools.functions.libs.data_preprocess import FillMissingValue -def function_name(df): - df_processed = df.copy() - num_cols = df_processed.select_dtypes(include='number').columns.tolist() - df_processed = fill_missing_value(df_processed, num_cols, 'mean') - - for col in num_cols: - low, high = df_processed[col].quantile([0.01, 0.99]) - df_processed[col] = df_processed[col].clip(low, high) - return df_processed +train_processed = train.copy() +test_processed = test.copy() +num_cols = train_processed.select_dtypes(include='number').columns.tolist() +fill_missing_value = FillMissingValue(features=num_cols, strategy='mean') +fill_missing_value.fit(train_processed) +train_processed = fill_missing_value.transform(train_processed) +test_processed = fill_missing_value.transform(test_processed) -df_processed = function_name(df) -print(df_processed.info()) +for col in num_cols: + low, high = train_processed[col].quantile([0.01, 0.99]) + train_processed[col] = train_processed[col].clip(low, high) + test_processed[col] = test_processed[col].clip(low, high) ```end # Constraints: -- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. - Prioritize using pre-defined tools for the same functionality. -- Return DataFrame should always be named `df_processed`, while the input DataFrame should based on the done tasks' output DataFrame. -- Limit to one print statement for the output DataFrame's info. +- Copy DataFrame before processing if needed. +- If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. +""" + +DATA_PREPROCESS_PROMPT = """ +The current task is about data preprocessing, please note the following: +- Monitor data types per column, applying appropriate methods. +- Ensure operations are on existing dataset columns. +- Avoid writing processed data to files. +- Prefer alternatives to one-hot encoding for categorical data. +- Only encode necessary categorical columns to allow for potential feature-specific engineering tasks later. +""" + +FEATURE_ENGINEERING_PROMPT = """ +The current task is about feature engineering. when performing it, please adhere to the following principles: +- Ensure operations are on existing dataset columns and consider the data type (numerical, categorical, etc.) and application scenario (classification, regression tasks, etc.). +- Create impactful features based on real-world knowledge and column info. +- Generate as diverse features as possible to improve the model's performance. +- If potential impactful features are not included in 'Code Steps', add new steps to generate them. """ MODEL_TRAIN_PROMPT = """ @@ -316,23 +258,17 @@ The current task is about training a model, please ensure high performance: - Use the data from previous task result directly, do not mock or reload data yourself. """ -DATA_PREPROCESS_OUTPUT_DESC = "Please note that all functions output a updated pandas.DataFrame after data preprocessing." - -FEATURE_ENGINEERING_OUTPUT_DESC = "Please note that all functions output a updated pandas.DataFrame with new features added or existing features modified." - -CLASSIFICATION_MODEL_OUTPUT_DESC = "" - -REGRESSION_MODEL_OUTPUT_DESC = "" +MODEL_EVALUATE_PROMPT = """ +The current task is about evaluating a model, please note the following: +- Ensure that the evaluated data is same processed as the training data. +- Use trained model from previous task result directly, do not mock or reload model yourself. +""" ML_SPECIFIC_PROMPT = { "data_preprocess": DATA_PREPROCESS_PROMPT, "feature_engineering": FEATURE_ENGINEERING_PROMPT, "model_train": MODEL_TRAIN_PROMPT, -} - -TOOL_OUTPUT_DESC = { - "data_preprocess": DATA_PREPROCESS_OUTPUT_DESC, - "feature_engineering": FEATURE_ENGINEERING_OUTPUT_DESC, + "model_evaluate": MODEL_EVALUATE_PROMPT, } ML_MODULE_MAP = { From 537d51c26e29a1774269825e3611667b2436e80d Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 14:32:25 +0800 Subject: [PATCH 132/637] write code with class tool --- metagpt/actions/write_analysis_code.py | 130 +++++++++---------- metagpt/roles/ml_engineer.py | 170 ++++++++++--------------- 2 files changed, 131 insertions(+), 169 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 58cab9c6a..aceebbfeb 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -4,7 +4,9 @@ @Author : orange-crow @File : write_code_v2.py """ -from typing import Dict, List, Union, Tuple, Optional, Any +from typing import Dict, List, Union, Tuple + +import yaml from metagpt.actions import Action from metagpt.logs import logger @@ -15,11 +17,9 @@ from metagpt.prompts.ml_engineer import ( TOOL_USAGE_PROMPT, ML_SPECIFIC_PROMPT, ML_MODULE_MAP, - TOOL_OUTPUT_DESC, DATA_PROCESS_PROMPT, - GENERATE_CODE_PROMPT + GENERATE_CODE_PROMPT, ) from metagpt.schema import Message, Plan -from metagpt.tools.functions import registry from metagpt.utils.common import create_func_config, remove_comments @@ -100,40 +100,55 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - @staticmethod - def _parse_recommend_tools(module: str, recommend_tools: list) -> List[Dict]: + def __init__(self, name: str = "", context=None, llm=None, schema_path=None): + super().__init__(name, context, llm) + self.schema_path = schema_path + self.available_tools = {} + + if self.schema_path is not None: + self._load_tools(schema_path) + + def _load_tools(self, schema_path): + """Load tools from yaml file""" + yml_files = schema_path.glob("*.yml") + for yml_file in yml_files: + module = yml_file.stem + with open(yml_file, "r", encoding="utf-8") as f: + self.available_tools[module] = yaml.safe_load(f) + + def _parse_recommend_tools(self, module: str, recommend_tools: list) -> dict: """ Parses and validates a list of recommended tools, and retrieves their schema from registry. Args: module (str): The module name for querying tools in the registry. - recommend_tools (list): A list of lists of recommended tools for each step. + recommend_tools (list): A list of recommended tools. Returns: - List[Dict]: A list of dicts of valid tool schemas. + dict: A dict of valid tool schemas. """ valid_tools = [] - available_tools = registry.get_all_by_module(module).keys() + available_tools = self.available_tools[module].keys() for tool in recommend_tools: if tool in available_tools: valid_tools.append(tool) - tool_catalog = registry.get_schemas(module, valid_tools) + tool_catalog = {tool: self.available_tools[module][tool] for tool in valid_tools} return tool_catalog async def _tool_recommendation( - self, - task: str, - code_steps: str, - available_tools: list + self, + task: str, + code_steps: str, + available_tools: dict, ) -> list: """ Recommend tools for the specified task. Args: - context (List[Message]): Action output history, source action denoted by Message.cause_by + task (str): the task to recommend tools for code_steps (str): the code steps to generate the full code for the task - available_tools (list): the available tools for the task + available_tools (dict): the available tools description Returns: list: recommended tools for the specified task @@ -149,27 +164,23 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): return recommend_tools async def run( - self, - context: List[Message], - plan: Plan = None, - code_steps: str = "", - column_info: str = "", - **kwargs, - ) -> str: + self, + context: List[Message], + plan: Plan = None, + code_steps: str = "", + column_info: str = "", + **kwargs, + ) -> Tuple[List[Message], str]: task_type = plan.current_task.task_type - available_tools = registry.get_all_schema_by_module(task_type) + available_tools = self.available_tools.get(task_type, {}) special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") - column_names = kwargs.get("column_names", {}) finished_tasks = plan.get_finished_tasks() code_context = [remove_comments(task.code) for task in finished_tasks] code_context = "\n\n".join(code_context) if len(available_tools) > 0: - available_tools = [ - {k: tool[k] for k in ["name", "description"] if k in tool} - for tool in available_tools - ] + available_tools = {k: v["description"] for k, v in available_tools.items()} recommend_tools = await self._tool_recommendation( plan.current_task.instruction, @@ -180,46 +191,27 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): logger.info(f"Recommended tools: \n{recommend_tools}") module_name = ML_MODULE_MAP[task_type] - output_desc = TOOL_OUTPUT_DESC.get(task_type, "") - new_code = "" - - for idx, tool in enumerate(recommend_tools): - hist_info = f"Previous finished code is \n\n ```Python {code_context} ``` \n\n " - - prompt = TOOL_USAGE_PROMPT.format( - goal=plan.current_task.instruction, - context=hist_info, - code_steps=code_steps, - column_names=column_names, - special_prompt=special_prompt, - module_name=module_name, - output_desc=output_desc, - function_catalog=tool_catalog[idx], - ) - - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - - rsp = await self.llm.aask_code(prompt, **tool_config) - logger.info(f"rsp is: {rsp}") - # final_code = final_code + "\n\n" + rsp["code"] - # final_code[key] = rsp["code"] - new_code = new_code + "\n\n" + rsp["code"] - code_context = code_context + "\n\n" + rsp["code"] - return new_code - - else: - hist_info = f"Previous finished code is \n\n ```Python {code_context} ``` \n\n " - - prompt = GENERATE_CODE_PROMPT.format( - goal=plan.current_task.instruction, - context=hist_info, - code_steps=code_steps, + prompt = TOOL_USAGE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, special_prompt=special_prompt, - # column_names=column_names + code_steps=code_steps, + module_name=module_name, + tool_catalog=tool_catalog, + ) + else: + prompt = GENERATE_CODE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + special_prompt=special_prompt, + code_steps=code_steps, ) - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - logger.info(f"prompt is: {prompt}") - rsp = await self.llm.aask_code(prompt, **tool_config) - logger.info(f"rsp is: {rsp}") - return rsp["code"] + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + rsp = await self.llm.aask_code(prompt, **tool_config) + context = [Message(content=prompt, role="user")] + return context, rsp["code"] diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 45fe728dd..20589079d 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,5 +1,6 @@ import json import re +from datetime import datetime from typing import List import fire @@ -10,12 +11,16 @@ from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.actions.write_plan import WritePlan -from metagpt.const import DATA_PATH +from metagpt.const import DATA_PATH, PROJECT_ROOT from metagpt.logs import logger -from metagpt.prompts.ml_engineer import GEN_DATA_DESC_PROMPT +from metagpt.prompts.ml_engineer import ( + GEN_DATA_DESC_PROMPT, + UPDATE_DATA_COLUMNS, + PRINT_DATA_COLUMNS +) from metagpt.roles import Role from metagpt.schema import Message, Plan -from metagpt.utils.common import CodeParser +from metagpt.utils.common import CodeParser, remove_comments, create_func_config from metagpt.actions.debug_code import DebugCode STRUCTURAL_CONTEXT = """ @@ -57,34 +62,6 @@ def remove_escape_and_color_codes(input_str): return result -def read_data(file: str) -> pd.DataFrame: - if file.endswith(".csv"): - df = pd.read_csv(file, sep=",") - sep_list = [";", "\t", ":", " ", "|"] - for sep in sep_list: - if df.shape[1] == 1: - df = pd.read_csv(file, sep=sep) - else: - break - else: - raise ValueError(f"Unsupported file type: {file}") - return df - - -def get_column_info(df: pd.DataFrame) -> str: - data = [] - for i in df.columns: - nan_freq = float("%.2g" % (df[i].isna().mean() * 100)) - n_unique = df[i].nunique() - data.append([i, df[i].dtype, nan_freq, n_unique]) - - samples = pd.DataFrame( - data, - columns=["Column_name", "Data_type", "NaN_Frequency(%)", "N_unique"], - ) - return samples.to_string(index=False) - - class AskReview(Action): async def run(self, context: List[Message], plan: Plan = None): logger.info("Current overall plan:") @@ -108,26 +85,20 @@ class AskReview(Action): return rsp, confirmed -class GenerateDataDesc(Action): - async def run(self, file: str) -> dict: - data_desc = {} - df = read_data(file) - data_head = df.head().to_dict(orient="list") - data_head = json.dumps(data_head, indent=4, ensure_ascii=False) - prompt = GEN_DATA_DESC_PROMPT.replace("{data_head}", data_head) - rsp = await self._aask(prompt) - rsp = CodeParser.parse_code(block=None, text=rsp) - rsp = json.loads(rsp) - data_desc["path"] = file - data_desc["data_desc"] = rsp["data_desc"] - data_desc["column_desc"] = rsp["column_desc"] - data_desc["column_info"] = get_column_info(df) - return data_desc +class UpdateDataColumns(Action): + async def run(self, plan: Plan = None) -> dict: + finished_tasks = plan.get_finished_tasks() + code_context = [remove_comments(task.code) for task in finished_tasks] + code_context = "\n\n".join(code_context) + prompt = UPDATE_DATA_COLUMNS.format(history_code=code_context) + tool_config = create_func_config(PRINT_DATA_COLUMNS) + rsp = await self.llm.aask_code(prompt, **tool_config) + return rsp class MLEngineer(Role): def __init__( - self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False, data_path: str = None + self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False, ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") @@ -136,13 +107,9 @@ class MLEngineer(Role): self.use_code_steps = True self.execute_code = ExecutePyCode() self.auto_run = auto_run - self.data_path = data_path self.data_desc = {} async def _plan_and_act(self): - if self.data_path: - self.data_desc = await self._generate_data_desc() - # create initial plan and update until confirmation await self._update_plan() @@ -163,25 +130,27 @@ class MLEngineer(Role): task.code_steps = code_steps self.plan.finish_current_task() self.working_memory.clear() - - if "print(df_processed.info())" in code: - self.data_desc["column_info"] = result + + success, new_code = await self._update_data_columns() + if success: + task.code = task.code + "\n\n" + new_code else: # update plan according to user's feedback and to take on changed tasks await self._update_plan() - - finished_tasks = self.plan.get_finished_tasks() - if len(finished_tasks) == len(self.plan.tasks): - code_context = [task.code for task in finished_tasks] - code_context = "\n\n".join(code_context) - result, success = await self.execute_code.run(code_context) - # truncated the result - print(truncate(result)) - - async def _generate_data_desc(self): - data_desc = await GenerateDataDesc().run(self.data_path) - return data_desc - + + time = datetime.now().strftime('%Y-%m-%d_%H-%M-%S') + self.execute_code.save_notebook(f"{DATA_PATH}/notebooks/ml_{time}.ipynb") + + async def _update_data_columns(self): + rsp = await UpdateDataColumns().run(self.plan) + is_update, code = rsp["is_update"], rsp["code"] + success = False + if is_update: + result, success = await self.execute_code.run(code) + if success: + self.data_desc["column_info"] = result + return success, code + async def _write_and_exec_code(self, max_retry: int = 3): code_steps = ( await WriteCodeSteps().run(self.plan) @@ -192,6 +161,7 @@ class MLEngineer(Role): counter = 0 improve_code = "" success = False + debug_context = [] finished_tasks = self.plan.get_finished_tasks() code_context = [task.code for task in finished_tasks] @@ -200,37 +170,38 @@ class MLEngineer(Role): code_result = "\n\n".join(code_result) while not success and counter < max_retry: - if counter == 0: - context = self.get_useful_memories() - else: - # context = self.get_useful_memories() - # logger.info(f"context {context}") + context = self.get_useful_memories() + + if counter > 0: improve_code = await DebugCode().run(plan=self.plan.current_task.instruction, - finished_code=code_context, - finished_code_result=code_result, + # finished_code=code_context, + # finished_code_result=code_result, code=code, - runtime_result=self.working_memory.get()) - - if not self.use_tools or self.plan.current_task.task_type == "other": + runtime_result=self.working_memory.get(), + context=debug_context) + + if improve_code != "": + code = improve_code + logger.info(f"new code \n{improve_code}") + cause_by = DebugCode + elif not self.use_tools or self.plan.current_task.task_type == "other": logger.info("Write code with pure generation") - code = await WriteCodeByGenerate().run( context=context, plan=self.plan, code_steps=code_steps, temperature=0.0 ) + debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] cause_by = WriteCodeByGenerate else: logger.info("Write code with tools") - - if improve_code != "": - code = improve_code - logger.info(f"new code {code}") - cause_by = DebugCode - else: - code = await WriteCodeWithTools().run( - context=context, plan=self.plan, code_steps=code_steps, **{"column_names": {}} - ) - - cause_by = WriteCodeWithTools + schema_path = PROJECT_ROOT / "metagpt/tools/functions/schemas" + tool_context, code = await WriteCodeWithTools(schema_path=schema_path).run( + context=context, + plan=self.plan, + code_steps=code_steps, + column_info=self.data_desc.get("column_info", ""), + ) + debug_context = tool_context + cause_by = WriteCodeWithTools self.working_memory.add( Message(content=code, role="assistant", cause_by=cause_by) @@ -238,9 +209,7 @@ class MLEngineer(Role): # debug on code, run on runcode with finished code and new_df # runcode = code_context + "\n\n" + code - runcode = code - - result, success = await self.execute_code.run(runcode) + result, success = await self.execute_code.run(code) # truncated the result print(truncate(result)) @@ -289,12 +258,12 @@ class MLEngineer(Role): self.plan.add_tasks(tasks) self.working_memory.clear() - def get_useful_memories(self) -> List[Message]: + def get_useful_memories(self, task_exclude_field: set = None) -> List[Message]: """find useful memories only to reduce context length and improve performance""" # TODO dataset description , code steps user_requirement = self.plan.goal tasks = json.dumps( - [task.dict() for task in self.plan.tasks], indent=4, ensure_ascii=False + [task.dict(exclude=task_exclude_field) for task in self.plan.tasks], indent=4, ensure_ascii=False ) current_task = self.plan.current_task.json() if self.plan.current_task else {} context = STRUCTURAL_CONTEXT.format( @@ -321,12 +290,13 @@ if __name__ == "__main__": # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." - data_path = f"{DATA_PATH}/titanic" - requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - - - async def main(requirement: str = requirement, auto_run: bool = True, data_path: str = ""): - role = MLEngineer(goal=requirement, auto_run=auto_run, data_path=data_path) + # data_path = f"{DATA_PATH}/titanic" + # requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + # requirement = f"Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" + data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" + requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." + async def main(requirement: str = requirement, auto_run: bool = True): + role = MLEngineer(goal=requirement, auto_run=auto_run) await role.run(requirement) From ea0b93d2b94997db94f470dbd3141f3f9dd435a6 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 13 Dec 2023 14:33:50 +0800 Subject: [PATCH 133/637] update code locally --- metagpt/actions/write_code_steps.py | 12 ++++++++---- metagpt/roles/ml_engineer.py | 26 +++++++++++++------------- 2 files changed, 21 insertions(+), 17 deletions(-) diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py index a19549b71..6bf223701 100644 --- a/metagpt/actions/write_code_steps.py +++ b/metagpt/actions/write_code_steps.py @@ -63,18 +63,22 @@ class WriteCodeSteps(Action): def get_context(self, plan: Plan): user_requirement = plan.goal - select_task_keys = ['task_id', 'instruction', 'is_finished', 'code'] - + # select_task_keys = ['task_id', 'instruction', 'is_finished', 'code'] + select_task_keys = ['task_id','code'] + def process_task(task): task_dict = task.dict() - ptask = {k: task_dict[k] for k in task_dict if k in select_task_keys} + ptask = {k: task_dict[k] for k in task_dict if k in select_task_keys } return ptask + tasks = json.dumps( - [process_task(task) for task in plan.tasks], indent=4, ensure_ascii=False + [process_task(task) for task in plan.tasks if task.is_finished==True], indent=4, ensure_ascii=False ) + current_task = json.dumps(process_task(plan.current_task)) if plan.current_task else {} context = STRUCTURAL_CONTEXT.format( user_requirement=user_requirement, tasks=tasks, current_task=current_task ) + print(context) # print(context) return context diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 8ad75b399..f50b6d494 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -148,7 +148,7 @@ class MLEngineer(Role): while self.plan.current_task: task = self.plan.current_task - logger.info(f"ready to take on task {task}") + logger.info(f"ready to take on task: {task}") # take on current task code, result, success, code_steps = await self._write_and_exec_code() @@ -157,9 +157,11 @@ class MLEngineer(Role): task_result_confirmed = await self._ask_review() # 针对当前task进行单独plan - if not success or not task_result_confirmed: - # fixme: 增加对应plan - self.state.plan() + # if not success or not task_result_confirmed: + # # fixme: 增加对应plan + # logger.info(task.result) + # # import pdb;pdb.set_trace() + # # self.state.plan() if success and task_result_confirmed: # tick off this task and record progress @@ -175,13 +177,13 @@ class MLEngineer(Role): # update plan according to user's feedback and to take on changed tasks await self._update_plan() - finished_tasks = self.plan.get_finished_tasks() - if len(finished_tasks) == len(self.plan.tasks): - code_context = [task.code for task in finished_tasks] - code_context = "\n\n".join(code_context) - result, success = await self.execute_code.run(code_context) - # truncated the result - print(truncate(result)) + # finished_tasks = self.plan.get_finished_tasks() + # if len(finished_tasks) == len(self.plan.tasks): + # code_context = [task.code for task in finished_tasks] + # code_context = "\n\n".join(code_context) + # result, success = await self.execute_code.run(code_context) + # # truncated the result + # print(truncate(result)) async def _generate_data_desc(self): data_desc = await GenerateDataDesc().run(self.data_path) @@ -258,8 +260,6 @@ class MLEngineer(Role): counter += 1 - success = False - return code, result, success, code_steps async def _ask_review(self): From 0d61e897002242f03f07d124bcc2b922cbd49cf0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 13 Dec 2023 14:59:04 +0800 Subject: [PATCH 134/637] add todo. --- metagpt/tools/functions/libs/udf/__init__.py | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index c9c818a96..5bad9a3a4 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -3,6 +3,7 @@ import os import inspect import importlib from pathlib import Path +from typing import Dict, List def extract_function_signatures(file_path): @@ -50,3 +51,8 @@ function_signatures = get_function_signatures_in_folder(folder_path) UDFS = [func for func in function_signatures if not func['udf_name'].startswith(('extract_function_signatures', 'get_function_signatures_in_folder'))] + + +# TODO: Create Yaml style UDFS Schema +def udfs2yaml(udfs: List[Dict]) -> Dict: + pass From abad52da85773c3f762eea7f7c956140e0c0cd3f Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 13 Dec 2023 15:38:19 +0800 Subject: [PATCH 135/637] update locally --- metagpt/actions/write_code_steps.py | 48 ++++++++++++++++++++++++++--- metagpt/roles/ml_engineer.py | 8 ++--- 2 files changed, 48 insertions(+), 8 deletions(-) diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py index 889c06679..efee96749 100644 --- a/metagpt/actions/write_code_steps.py +++ b/metagpt/actions/write_code_steps.py @@ -6,6 +6,31 @@ from metagpt.actions import Action from metagpt.schema import Message, Task, Plan from metagpt.utils.common import CodeParser +# CODE_STEPS_PROMPT_TEMPLATE = """ +# # Context +# {context} +# +# ----- +# Tasks are all code development tasks. +# You are a professional engineer, the main goal is to plan out concise solution steps for Current Task before coding. +# A planning process can reduce the difficulty and improve the quality of coding. +# You may be given some code plans for the tasks ahead, but you don't have to follow the existing plan when planning the current task. +# The output plan should following the subsequent principles: +# 1.The plan is a rough checklist of steps outlining the entire program's structure.Try to keep the number of steps fewer than 5. +# 2.The steps should be written concisely and at a high level, avoiding overly detailed implementation specifics. +# 3.The execution of the plan happens sequentially, but the plan can incorporate conditional (if) and looping(loop) keywords for more complex structures. +# +# Output the code steps in a JSON format, as shown in this example: +# ```json +# { +# "Step 1": "", +# "Step 2": "", +# "Step 3": "", +# ... +# } +# ``` +# """ + CODE_STEPS_PROMPT_TEMPLATE = """ # Context {context} @@ -19,6 +44,7 @@ The output plan should following the subsequent principles: 1.The plan is a rough checklist of steps outlining the entire program's structure.Try to keep the number of steps fewer than 5. 2.The steps should be written concisely and at a high level, avoiding overly detailed implementation specifics. 3.The execution of the plan happens sequentially, but the plan can incorporate conditional (if) and looping(loop) keywords for more complex structures. +4.Follow the code logic to design and provide the code steps. You can analysis it step by step Output the code steps in a JSON format, as shown in this example: ```json @@ -31,11 +57,22 @@ Output the code steps in a JSON format, as shown in this example: ``` """ +# STRUCTURAL_CONTEXT = """ +# ## User Requirement +# {user_requirement} +# ## Current Plan +# {tasks} +# ## Current Task +# {current_task} +# """ + STRUCTURAL_CONTEXT = """ ## User Requirement {user_requirement} -## Current Plan +## Plan {tasks} +## Codes +{codes} ## Current Task {current_task} """ @@ -63,21 +100,24 @@ class WriteCodeSteps(Action): def get_context(self, plan: Plan): user_requirement = plan.goal - select_task_keys = ['task_id', 'instruction', 'is_finished', 'code'] - # select_task_keys = ['task_id','code'] + # select_task_keys = ['task_id', 'instruction', 'is_finished', 'code'] + select_task_keys = ['task_id','instruction'] def process_task(task): task_dict = task.dict() ptask = {k: task_dict[k] for k in task_dict if k in select_task_keys } return ptask + tasks = json.dumps( [process_task(task) for task in plan.tasks], indent=4, ensure_ascii=False ) + code_lists = [task.code for task in plan.tasks if task.is_finished==True] + codes = "\n\n".join(code_lists) current_task = json.dumps(process_task(plan.current_task)) if plan.current_task else {} context = STRUCTURAL_CONTEXT.format( - user_requirement=user_requirement, tasks=tasks, current_task=current_task + user_requirement=user_requirement, tasks=tasks, codes=codes, current_task=current_task ) print(context) # print(context) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 20589079d..c735eb983 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -290,11 +290,11 @@ if __name__ == "__main__": # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." - # data_path = f"{DATA_PATH}/titanic" - # requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + data_path = f"{DATA_PATH}/titanic" + requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." # requirement = f"Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" - requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." + # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" + # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." async def main(requirement: str = requirement, auto_run: bool = True): role = MLEngineer(goal=requirement, auto_run=auto_run) await role.run(requirement) From 05ae935d8cfaef957c539ce1c3a6ebcb21d40ad8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 13 Dec 2023 15:55:04 +0800 Subject: [PATCH 136/637] fix truncate. --- metagpt/actions/execute_code.py | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 1d20bf3f6..36e01ed0e 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -186,14 +186,13 @@ class ExecutePyCode(ExecuteCode, Action): def truncate(result: str, keep_len: int = 2000) -> str: desc = f"Truncated to show only the last {keep_len} characters\n" if result.startswith(desc): - result = result[-len(desc) :] + result = result[len(desc) :] if len(result) > keep_len: result = result[-keep_len:] - - if not result.startswith(desc): return desc + result - return desc + + return result def remove_escape_and_color_codes(input_str): From cfbf1630841e05d07d6b537e736dbcf28e349622 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 13 Dec 2023 15:55:30 +0800 Subject: [PATCH 137/637] add test for truncate. --- tests/metagpt/actions/test_execute_code.py | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/tests/metagpt/actions/test_execute_code.py b/tests/metagpt/actions/test_execute_code.py index 73b5886dc..95f883e12 100644 --- a/tests/metagpt/actions/test_execute_code.py +++ b/tests/metagpt/actions/test_execute_code.py @@ -1,6 +1,6 @@ import pytest -from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.execute_code import ExecutePyCode, truncate from metagpt.schema import Message @@ -81,3 +81,10 @@ async def test_plotting_bug(): pi = ExecutePyCode() output = await pi.run(code) assert output[1] is True + + +def test_truncate(): + output = "hello world" + assert truncate(output) == output + output = "hello world" + assert truncate(output, 5) == "Truncated to show only the last 5 characters\nworld" From 8d694d47d9f2372011d39f759042b48cc54c8c27 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 13 Dec 2023 16:26:24 +0800 Subject: [PATCH 138/637] update code step prompts --- metagpt/actions/write_analysis_code.py | 57 ++++++++++++++------------ metagpt/actions/write_code_steps.py | 7 ++-- metagpt/prompts/ml_engineer.py | 2 +- 3 files changed, 35 insertions(+), 31 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index aceebbfeb..3e91f4b14 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -26,7 +26,7 @@ from metagpt.utils.common import create_func_config, remove_comments class BaseWriteAnalysisCode(Action): DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" - + def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG # 全部转成list @@ -45,7 +45,7 @@ class BaseWriteAnalysisCode(Action): messages.append(p.to_dict()) elif isinstance(p.content, dict) and "code" in p.content: messages.append(p.content["code"]) - + # 添加默认的提示词 if ( default_system_msg not in messages[0]["content"] @@ -61,7 +61,7 @@ class BaseWriteAnalysisCode(Action): "content": messages[0]["content"] + default_system_msg, } return messages - + async def run( self, context: List[Message], plan: Plan = None, code_steps: str = "" ) -> str: @@ -79,10 +79,10 @@ class BaseWriteAnalysisCode(Action): class WriteCodeByGenerate(BaseWriteAnalysisCode): """Write code fully by generation""" - + def __init__(self, name: str = "", context=None, llm=None) -> str: super().__init__(name, context, llm) - + async def run( self, context: [List[Message]], @@ -99,15 +99,15 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - + def __init__(self, name: str = "", context=None, llm=None, schema_path=None): super().__init__(name, context, llm) self.schema_path = schema_path self.available_tools = {} - + if self.schema_path is not None: self._load_tools(schema_path) - + def _load_tools(self, schema_path): """Load tools from yaml file""" yml_files = schema_path.glob("*.yml") @@ -115,7 +115,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module = yml_file.stem with open(yml_file, "r", encoding="utf-8") as f: self.available_tools[module] = yaml.safe_load(f) - + def _parse_recommend_tools(self, module: str, recommend_tools: list) -> dict: """ Parses and validates a list of recommended tools, and retrieves their schema from registry. @@ -132,15 +132,15 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): for tool in recommend_tools: if tool in available_tools: valid_tools.append(tool) - + tool_catalog = {tool: self.available_tools[module][tool] for tool in valid_tools} return tool_catalog - + async def _tool_recommendation( - self, - task: str, - code_steps: str, - available_tools: dict, + self, + task: str, + code_steps: str, + available_tools: dict, ) -> list: """ Recommend tools for the specified task. @@ -162,26 +162,26 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): rsp = await self.llm.aask_code(prompt, **tool_config) recommend_tools = rsp["recommend_tools"] return recommend_tools - + async def run( - self, - context: List[Message], - plan: Plan = None, - code_steps: str = "", - column_info: str = "", - **kwargs, + self, + context: List[Message], + plan: Plan = None, + code_steps: str = "", + column_info: str = "", + **kwargs, ) -> Tuple[List[Message], str]: task_type = plan.current_task.task_type available_tools = self.available_tools.get(task_type, {}) special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") - + finished_tasks = plan.get_finished_tasks() code_context = [remove_comments(task.code) for task in finished_tasks] code_context = "\n\n".join(code_context) - + if len(available_tools) > 0: available_tools = {k: v["description"] for k, v in available_tools.items()} - + recommend_tools = await self._tool_recommendation( plan.current_task.instruction, code_steps, @@ -189,8 +189,9 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): ) tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) logger.info(f"Recommended tools: \n{recommend_tools}") - + module_name = ML_MODULE_MAP[task_type] + prompt = TOOL_USAGE_PROMPT.format( user_requirement=plan.goal, history_code=code_context, @@ -201,6 +202,8 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name=module_name, tool_catalog=tool_catalog, ) + + else: prompt = GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, @@ -210,7 +213,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): special_prompt=special_prompt, code_steps=code_steps, ) - + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) context = [Message(content=prompt, role="user")] diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py index efee96749..9e06bc91e 100644 --- a/metagpt/actions/write_code_steps.py +++ b/metagpt/actions/write_code_steps.py @@ -44,7 +44,7 @@ The output plan should following the subsequent principles: 1.The plan is a rough checklist of steps outlining the entire program's structure.Try to keep the number of steps fewer than 5. 2.The steps should be written concisely and at a high level, avoiding overly detailed implementation specifics. 3.The execution of the plan happens sequentially, but the plan can incorporate conditional (if) and looping(loop) keywords for more complex structures. -4.Follow the code logic to design and provide the code steps. You can analysis it step by step +4.Design and provide code steps by following the code logic. Analyze the provided code step by step and reuse the imported library. Output the code steps in a JSON format, as shown in this example: ```json @@ -101,11 +101,12 @@ class WriteCodeSteps(Action): def get_context(self, plan: Plan): user_requirement = plan.goal # select_task_keys = ['task_id', 'instruction', 'is_finished', 'code'] - select_task_keys = ['task_id','instruction'] + # select_task_keys = ['task_id','instruction'] def process_task(task): task_dict = task.dict() - ptask = {k: task_dict[k] for k in task_dict if k in select_task_keys } + # ptask = {k: task_dict[k] for k in task_dict if k in select_task_keys } + ptask = f"task_id_{task_dict['task_id']}:{task_dict['instruction']}\n" return ptask diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index d11cbf453..2d2d3315a 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -231,8 +231,8 @@ for col in num_cols: # Constraints: - Prioritize using pre-defined tools for the same functionality. - Copy DataFrame before processing if needed. -- If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. """ +#- If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. DATA_PREPROCESS_PROMPT = """ The current task is about data preprocessing, please note the following: From abbaa6afa95e7fcada42df8a299f1dd3a7cc97c5 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 17:03:56 +0800 Subject: [PATCH 139/637] refine prompt --- metagpt/prompts/ml_engineer.py | 36 ++++++++++++++++++++-------------- 1 file changed, 21 insertions(+), 15 deletions(-) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 2d2d3315a..f2412c35b 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -155,46 +155,51 @@ PRINT_DATA_COLUMNS = { GENERATE_CODE_PROMPT = """ # Background -Assist in completing [{user_requirement}] in a Jupyter notebook. +As a data scientist, you need to help user to achieve their goal [{user_requirement}] step-by-step in an continuous Jupyter notebook. -## Task Progress -### Done Tasks +## Done Tasks ```python {history_code} ```end -### Current Task +## Current Task {current_task} -## Latest Data Info +# Latest Data Info +Latest data info after previous tasks: {column_info} # Task -Fully implement 'Current Task', ensuring all necessary steps are covered without repeating code from 'Done Tasks'. Specifically, {special_prompt} +Write complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc. +Specifically, {special_prompt} # Code Steps: Follow steps below when you writing code if it's convenient. {code_steps} + +# Constraints: +- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. """ TOOL_USAGE_PROMPT = """ # Background -Assist in completing [{user_requirement}] in a Jupyter notebook. +As a data scientist, you need to help user to achieve their goal [{user_requirement}] step-by-step in an continuous Jupyter notebook. -## Task Progress -### Done Tasks +## Done Tasks ```python {history_code} ```end -### Current Task +## Current Task {current_task} -## Latest Data Info +# Latest Data Info +Latest data info after previous tasks: {column_info} # Task -Fully implement 'Current Task', ensuring all necessary steps are covered without repeating code from 'Done Tasks'. Specifically, {special_prompt} +Write complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc. +Specifically, {special_prompt} # Code Steps: Follow steps below when you writing code if it's convenient. @@ -205,11 +210,11 @@ Follow steps below when you writing code if it's convenient. - You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. # Available Tools: -Each Class tool is described in JSON format. When you call it, import the tool from `{module_name}` first. +Each Class tool is described in JSON format. When you call a tool, import the tool from `{module_name}` first. {tool_catalog} # Output Example: -For "fill missing value and handle outliers", the output code be like when there are training data and test data: +when current task is "fill missing value and handle outliers", and their are training data and test data, the output code be like: ```python # Tools used: ['FillMissingValue'] from metagpt.tools.functions.libs.data_preprocess import FillMissingValue @@ -229,8 +234,9 @@ for col in num_cols: ```end # Constraints: +- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. - Prioritize using pre-defined tools for the same functionality. -- Copy DataFrame before processing if needed. +- Always copy the DataFrame before processing it and use the copy to process. """ #- If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. From 4423524734b15fdb9ca8aafb5eefa823d70ba671 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 18:11:54 +0800 Subject: [PATCH 140/637] fix schema --- .../tools/functions/schemas/feature_engineering.yml | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/metagpt/tools/functions/schemas/feature_engineering.yml b/metagpt/tools/functions/schemas/feature_engineering.yml index 4f2a7100d..3ba9e863b 100644 --- a/metagpt/tools/functions/schemas/feature_engineering.yml +++ b/metagpt/tools/functions/schemas/feature_engineering.yml @@ -53,17 +53,17 @@ PolynomialExpansion: CatCount: type: class - description: "Add value counts of categorical columns as new features." + description: "Add value counts of a categorical column as new feature." methods: __init__: description: "Initialize self." parameters: properties: - cols: - type: list - description: "Columns for value counts." + col: + type: str + description: "Column for value counts." required: - - cols + - col fit: description: "Fit the CatCount model." parameters: From e59bab73b06985fd02cc955002372909a0c571aa Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 19:36:02 +0800 Subject: [PATCH 141/637] refine prompt --- metagpt/prompts/ml_engineer.py | 31 ++++++++++++++++++++++++++----- metagpt/roles/ml_engineer.py | 9 +-------- 2 files changed, 27 insertions(+), 13 deletions(-) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index f2412c35b..05d8db8e9 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -174,11 +174,29 @@ Write complete code for 'Current Task'. And avoid duplicating code from 'Done Ta Specifically, {special_prompt} # Code Steps: -Follow steps below when you writing code if it's convenient. +Strictly follow steps below when you writing code if it's convenient. {code_steps} +# Output Example: +when current task is "train a lightgbm model on training data", and their are two steps in 'Code Steps', the code be like: +```python +# Step 1: check data type and convert to numeric +ojb_cols = train.select_dtypes(include='object').columns.tolist() + +for col in obj_cols: + encoder = LabelEncoder() + train[col] = encoder.fit_transform(train[col]) + test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown') + test[col] = encoder.transform(test[col]) + +# Step 2: train lightgbm model +model = LGBMClassifier() +model.fit(train, y_train) +```end + # Constraints: - Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. +- The output code should contain all steps implemented in 'Code Steps'. """ TOOL_USAGE_PROMPT = """ @@ -202,7 +220,7 @@ Write complete code for 'Current Task'. And avoid duplicating code from 'Done Ta Specifically, {special_prompt} # Code Steps: -Follow steps below when you writing code if it's convenient. +Strictly follow steps below when you writing code if it's convenient. {code_steps} # Capabilities @@ -214,8 +232,9 @@ Each Class tool is described in JSON format. When you call a tool, import the to {tool_catalog} # Output Example: -when current task is "fill missing value and handle outliers", and their are training data and test data, the output code be like: +when current task is "do data preprocess, like fill missing value, handle outliers, etc.", and their are two steps in 'Code Steps', the code be like: ```python +# Step 1: fill missing value # Tools used: ['FillMissingValue'] from metagpt.tools.functions.libs.data_preprocess import FillMissingValue @@ -227,6 +246,7 @@ fill_missing_value.fit(train_processed) train_processed = fill_missing_value.transform(train_processed) test_processed = fill_missing_value.transform(test_processed) +# Step 2: handle outliers for col in num_cols: low, high = train_processed[col].quantile([0.01, 0.99]) train_processed[col] = train_processed[col].clip(low, high) @@ -235,8 +255,9 @@ for col in num_cols: # Constraints: - Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. -- Prioritize using pre-defined tools for the same functionality. +- Always prioritize using pre-defined tools for the same functionality. - Always copy the DataFrame before processing it and use the copy to process. +- The output code should contain all steps implemented correctly in 'Code Steps'. """ #- If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. @@ -266,7 +287,7 @@ The current task is about training a model, please ensure high performance: MODEL_EVALUATE_PROMPT = """ The current task is about evaluating a model, please note the following: -- Ensure that the evaluated data is same processed as the training data. +- Ensure that the evaluated data is same processed as the training data. If not, remember use object in 'Done Tasks' to transform the data. - Use trained model from previous task result directly, do not mock or reload model yourself. """ diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index c735eb983..6a2a9e2b0 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -32,13 +32,6 @@ STRUCTURAL_CONTEXT = """ {tasks} ## Current Task {current_task} -## Packages Installed -scikit-learn -pandas -numpy -lightgbm -xgboost -catboost """ @@ -212,7 +205,7 @@ class MLEngineer(Role): result, success = await self.execute_code.run(code) # truncated the result print(truncate(result)) - + self.working_memory.add( Message(content=truncate(remove_escape_and_color_codes(result)), role="user", cause_by=ExecutePyCode) ) From 7e6e493499c41c91c56a19a2ebc7ecb329ab6f5f Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 19:36:31 +0800 Subject: [PATCH 142/637] refine prompt --- metagpt/actions/debug_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 53ca2f54d..58d006a08 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -47,7 +47,7 @@ REFLECTION_PROMPT = """ [runtime Error] {runtime_result} - Analysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. + Analysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step. [reflection on previous impl]: xxx From cfb577d6747ba7dca7cea92b7199494a66eb3dfb Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 20:10:17 +0800 Subject: [PATCH 143/637] rollback config --- config/config.yaml | 17 ++++++++--------- 1 file changed, 8 insertions(+), 9 deletions(-) diff --git a/config/config.yaml b/config/config.yaml index 694251f17..17605307a 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -5,7 +5,7 @@ ## The official OPENAI_API_BASE is https://api.openai.com/v1 ## If the official OPENAI_API_BASE is not available, we recommend using the [openai-forward](https://github.com/beidongjiedeguang/openai-forward). ## Or, you can configure OPENAI_PROXY to access official OPENAI_API_BASE. -#OPENAI_API_BASE: "https://api.openai.com/v1" +OPENAI_API_BASE: "https://api.openai.com/v1" #OPENAI_PROXY: "http://127.0.0.1:8118" #OPENAI_API_KEY: "YOUR_API_KEY" # set the value to sk-xxx if you host the openai interface for open llm model OPENAI_API_MODEL: "gpt-4" @@ -24,13 +24,12 @@ RPM: 10 #### if AZURE, check https://github.com/openai/openai-cookbook/blob/main/examples/azure/chat.ipynb #### You can use ENGINE or DEPLOYMENT mode -OPENAI_API_TYPE: "azure" -OPENAI_API_BASE: "https://deepwisdom.openai.azure.com/" -OPENAI_API_KEY: "02ae6058d09849c691176befeae2107c" -#OPENAI_API_VERSION: "2023-05-15" -OPENAI_API_VERSION: "2023-07-01-preview" -DEPLOYMENT_ID: "GPT-4" -OPENAI_API_ENGINE: "gpt-4" +#OPENAI_API_TYPE: "azure" +#OPENAI_API_BASE: "YOUR_AZURE_ENDPOINT" +#OPENAI_API_KEY: "YOUR_AZURE_API_KEY" +#OPENAI_API_VERSION: "YOUR_AZURE_API_VERSION" +#DEPLOYMENT_NAME: "YOUR_DEPLOYMENT_NAME" +#DEPLOYMENT_ID: "YOUR_DEPLOYMENT_ID" #### if zhipuai from `https://open.bigmodel.cn`. You can set here or export API_KEY="YOUR_API_KEY" # ZHIPUAI_API_KEY: "YOUR_API_KEY" @@ -88,7 +87,7 @@ SD_T2I_API: "/sdapi/v1/txt2img" MODEL_FOR_RESEARCHER_SUMMARY: gpt-3.5-turbo MODEL_FOR_RESEARCHER_REPORT: gpt-3.5-turbo-16k -### choose the engine for mermaid conversion, +### choose the engine for mermaid conversion, # default is nodejs, you can change it to playwright,pyppeteer or ink # MERMAID_ENGINE: nodejs From 8b0b5eeb804402f6a5329b92cdcb6da9e387d59d Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 20:14:10 +0800 Subject: [PATCH 144/637] fix conflict --- metagpt/actions/write_code_steps.py | 1 - metagpt/roles/ml_engineer.py | 48 ++++++++++++----------------- 2 files changed, 19 insertions(+), 30 deletions(-) diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py index 9e06bc91e..3c08adc19 100644 --- a/metagpt/actions/write_code_steps.py +++ b/metagpt/actions/write_code_steps.py @@ -120,6 +120,5 @@ class WriteCodeSteps(Action): context = STRUCTURAL_CONTEXT.format( user_requirement=user_requirement, tasks=tasks, codes=codes, current_task=current_task ) - print(context) # print(context) return context diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 26dfdbc67..8ab3ac981 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -4,30 +4,26 @@ from datetime import datetime import fire -from metagpt.roles import Role -from metagpt.schema import Message, Plan -from metagpt.memory import Memory -from metagpt.logs import logger from metagpt.actions import Action -from metagpt.actions.write_plan import WritePlan, update_plan_from_rsp, precheck_update_plan_from_rsp -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools -from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis, Reflect, ReviewConst +from metagpt.actions.debug_code import DebugCode from metagpt.actions.execute_code import ExecutePyCode -from metagpt.roles.kaggle_manager import DownloadData, SubmitResult -from metagpt.prompts.ml_engineer import STRUCTURAL_CONTEXT +from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis, Reflect, ReviewConst +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.actions.write_plan import WritePlan +from metagpt.actions.write_plan import update_plan_from_rsp, precheck_update_plan_from_rsp from metagpt.const import DATA_PATH, PROJECT_ROOT from metagpt.logs import logger +from metagpt.memory import Memory +from metagpt.prompts.ml_engineer import STRUCTURAL_CONTEXT from metagpt.prompts.ml_engineer import ( - GEN_DATA_DESC_PROMPT, UPDATE_DATA_COLUMNS, PRINT_DATA_COLUMNS ) from metagpt.roles import Role +from metagpt.roles.kaggle_manager import DownloadData, SubmitResult from metagpt.schema import Message, Plan -from metagpt.utils.common import CodeParser, remove_comments, create_func_config -from metagpt.actions.debug_code import DebugCode +from metagpt.utils.common import remove_comments, create_func_config from metagpt.utils.save_code import save_code_file @@ -103,9 +99,10 @@ class MLEngineer(Role): self.plan.finish_current_task() self.working_memory.clear() - success, new_code = await self._update_data_columns() - if success: - task.code = task.code + "\n\n" + new_code + if self.use_tools: + success, new_code = await self._update_data_columns() + if success: + task.code = task.code + "\n\n" + new_code confirmed_and_more = (ReviewConst.CONTINUE_WORD[0] in review.lower() and review.lower() not in ReviewConst.CONTINUE_WORD[0]) # "confirm, ... (more content, such as changing downstream tasks)" @@ -134,9 +131,6 @@ class MLEngineer(Role): save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") return rsp - time = datetime.now().strftime('%Y-%m-%d_%H-%M-%S') - self.execute_code.save_notebook(f"{DATA_PATH}/notebooks/ml_{time}.ipynb") - async def _update_data_columns(self): rsp = await UpdateDataColumns().run(self.plan) is_update, code = rsp["is_update"], rsp["code"] @@ -159,12 +153,6 @@ class MLEngineer(Role): success = False debug_context = [] - finished_tasks = self.plan.get_finished_tasks() - code_context = [task.code for task in finished_tasks] - code_result = [task.result for task in finished_tasks] - code_context = "\n\n".join(code_context) - code_result = "\n\n".join(code_result) - while not success and counter < max_retry: context = self.get_useful_memories() @@ -272,16 +260,18 @@ class MLEngineer(Role): self.working_memory.add(Message(content=reflection, role="assistant")) self.working_memory.add(Message(content=Reflect.REWRITE_PLAN_INSTRUCTION, role="user")) - def get_useful_memories(self, task_exclude_field: set = None) -> List[Message]: + def get_useful_memories(self, task_exclude_field=None) -> List[Message]: """find useful memories only to reduce context length and improve performance""" # TODO dataset description , code steps + if task_exclude_field is None: + task_exclude_field = {'code_steps'} user_requirement = self.plan.goal data_desc = self.plan.context tasks = [task.dict(exclude=task_exclude_field) for task in self.plan.tasks] - for task in tasks: - # Shorten the context as we don't need code steps after we get the codes. - # This doesn't affect current_task below, which should hold the code steps - task.pop("code_steps") + # for task in tasks: + # # Shorten the context as we don't need code steps after we get the codes. + # # This doesn't affect current_task below, which should hold the code steps + # task.pop("code_steps") tasks = json.dumps(tasks, indent=4, ensure_ascii=False) current_task = self.plan.current_task.json() if self.plan.current_task else {} context = STRUCTURAL_CONTEXT.format( From 7744815c5ff8f61eb90ccee07555c9f7207182bd Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 20:32:49 +0800 Subject: [PATCH 145/637] fix conflict --- requirements.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/requirements.txt b/requirements.txt index 9b75fd200..2328de2a1 100644 --- a/requirements.txt +++ b/requirements.txt @@ -45,6 +45,7 @@ wrapt==1.15.0 websocket-client==0.58.0 zhipuai==1.0.7 rich==13.6.0 +nbclient==0.9.0 nbformat==5.9.2 ipython==8.17.2 ipykernel==6.27.0 From edd6987a1c4738f27fb1936fa701441145b96869 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 13 Dec 2023 20:41:32 +0800 Subject: [PATCH 146/637] drop old tool definition --- metagpt/tools/functions/__init__.py | 3 - metagpt/tools/functions/libs/ml_model.py | 196 ------------------ metagpt/tools/functions/register/__init__.py | 6 - metagpt/tools/functions/register/register.py | 78 ------- metagpt/tools/functions/schemas/base.py | 100 --------- .../functions/schemas/data_preprocess.py | 67 ------ .../functions/schemas/feature_engineering.py | 110 ---------- metagpt/tools/functions/schemas/ml_model.py | 55 ----- 8 files changed, 615 deletions(-) delete mode 100644 metagpt/tools/functions/libs/ml_model.py delete mode 100644 metagpt/tools/functions/register/__init__.py delete mode 100644 metagpt/tools/functions/register/register.py delete mode 100644 metagpt/tools/functions/schemas/base.py delete mode 100644 metagpt/tools/functions/schemas/data_preprocess.py delete mode 100644 metagpt/tools/functions/schemas/feature_engineering.py delete mode 100644 metagpt/tools/functions/schemas/ml_model.py diff --git a/metagpt/tools/functions/__init__.py b/metagpt/tools/functions/__init__.py index 30ee10827..a0a43f507 100644 --- a/metagpt/tools/functions/__init__.py +++ b/metagpt/tools/functions/__init__.py @@ -4,6 +4,3 @@ # @Author : lidanyang # @File : __init__.py # @Desc : -from metagpt.tools.functions.register.register import registry -import metagpt.tools.functions.libs.feature_engineering -import metagpt.tools.functions.libs.data_preprocess diff --git a/metagpt/tools/functions/libs/ml_model.py b/metagpt/tools/functions/libs/ml_model.py deleted file mode 100644 index b669de2c1..000000000 --- a/metagpt/tools/functions/libs/ml_model.py +++ /dev/null @@ -1,196 +0,0 @@ -from sklearn.model_selection import train_test_split -from sklearn.preprocessing import LabelEncoder - -from sklearn.linear_model import LogisticRegression -from sklearn.ensemble import RandomForestClassifier -from sklearn.ensemble import GradientBoostingClassifier - - -from sklearn.linear_model import LinearRegression -from sklearn.ensemble import RandomForestRegressor -from sklearn.ensemble import GradientBoostingRegressor - -from metagpt.tools.functions import registry -from metagpt.tools.functions.schemas.ml_model import * - - -######### -## 分类 ## -######### - - -@registry.register("classification_model", LogisticRegressionClassification) -def logistic_regression_classification(df, label, test_size=0.2, penalty='l2', dual=False): - nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] - for col in nonnumeric_columns: - df[col] = LabelEncoder().fit_transform(df[col]) - df = df.fillna(0) - - features = [col for col in df if col != label] - x, y = df[features], df[label] - tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) - - model = LogisticRegression(penalty=penalty, dual=dual) - model.fit(tr_x, tr_y, ) - te_pred_prob = model.predict_proba(te_x) - - res = { - 'te_pred_prob': te_pred_prob - } - return res - - -@registry.register("classification_model", RandomForestClassification) -def random_forest_classification(df, label, test_size=0.2, n_estimators=100, criterion='gini'): - nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] - for col in nonnumeric_columns: - df[col] = LabelEncoder().fit_transform(df[col]) - df = df.fillna(0) - - features = [col for col in df if col != label] - x, y = df[features], df[label] - tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) - model = RandomForestClassifier(n_estimators=n_estimators, criterion=criterion) - model.fit(tr_x, tr_y, ) - te_pred_prob = model.predict_proba(te_x) - - res = { - 'te_pred_prob': te_pred_prob - } - return res - - -@registry.register("classification_model", GradientBoostingClassification) -def gradient_boosting_classification(df, label, test_size=0.2, n_estimators=100, learning_rate=0.1): - nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] - for col in nonnumeric_columns: - df[col] = LabelEncoder().fit_transform(df[col]) - df = df.fillna(0) - - features = [col for col in df if col != label] - x, y = df[features], df[label] - tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) - model = GradientBoostingClassifier(n_estimators=n_estimators, learning_rate=learning_rate) - model.fit(tr_x, tr_y, ) - te_pred_prob = model.predict_proba(te_x) - - res = { - 'te_pred_prob': te_pred_prob - } - return res - - - -######### -## 回归 ## -######### - - -@registry.register("regression_model", LinearRegressionRegression) -def linear_regression(df, label, test_size=0.2, ): - nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] - for col in nonnumeric_columns: - df[col] = LabelEncoder().fit_transform(df[col]) - df = df.fillna(0) - - features = [col for col in df if col != label] - x, y = df[features], df[label] - tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) - - model = LinearRegression() - model.fit(tr_x, tr_y, ) - te_pred_prob = model.predict(te_x) - - res = { - 'te_pred_prob': te_pred_prob - } - return res - - -@registry.register("regression_model", RandomForestRegression) -def random_forest_regression(df, label, test_size=0.2, n_estimators=100, criterion='squared_error'): - nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] - for col in nonnumeric_columns: - df[col] = LabelEncoder().fit_transform(df[col]) - df = df.fillna(0) - - features = [col for col in df if col != label] - x, y = df[features], df[label] - tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) - model = RandomForestRegressor(n_estimators=n_estimators, criterion=criterion) - model.fit(tr_x, tr_y, ) - te_pred_prob = model.predict(te_x) - - res = { - 'te_pred_prob': te_pred_prob - } - return res - - -@registry.register("regression_model", GradientBoostingRegression) -def gradient_boosting_regression(df, label, test_size=0.2, n_estimators=100, learning_rate=0.1): - nonnumeric_columns = [col for col in df if df[col].dtype == 'object'] - for col in nonnumeric_columns: - df[col] = LabelEncoder().fit_transform(df[col]) - df = df.fillna(0) - - features = [col for col in df if col != label] - x, y = df[features], df[label] - tr_x, te_x, tr_y, te_y = train_test_split(x, y, test_size=test_size, random_state=1) - model = GradientBoostingRegressor(n_estimators=n_estimators, learning_rate=learning_rate) - model.fit(tr_x, tr_y, ) - te_pred_prob = model.predict(te_x) - - res = { - 'te_pred_prob': te_pred_prob - } - return res - - -if __name__ == '__main__': - def run(): - from sklearn.datasets import load_iris - loader = load_iris(as_frame=True) - df = loader['data'] - df['target'] = loader['target'] - - df[df.columns[0]] = df[df.columns[0]].astype(str) - df[df.columns[1]] = df[df.columns[1]].astype(int) - df['target'] = df['target'].astype(str) - - print(df) - print('####'*5) - res = logistic_regression_classification(df, 'target', test_size=0.25, penalty='l2', dual=False) - print(res['te_pred_prob']) - - print('####'*5) - res = random_forest_classification(df, 'target', test_size=0.25, n_estimators=100, criterion='gini') - print(res['te_pred_prob']) - - print('####'*5) - res = gradient_boosting_classification(df, 'target', test_size=0.25, n_estimators=100, learning_rate=0.1) - print(res['te_pred_prob']) - - from sklearn.datasets import make_regression - import pandas as pd - loader = make_regression() - df = pd.DataFrame(loader[0]) - df['target'] = loader[1] - - df[df.columns[0]] = df[df.columns[0]].astype(str) - df[df.columns[1]] = df[df.columns[1]].astype(int) - # df['target'] = df['target'].astype(str) - - print(df) - print('####' * 5) - res = linear_regression(df, 'target', test_size=0.25, ) - print(res['te_pred_prob']) - - print('####' * 5) - res = random_forest_regression(df, 'target', test_size=0.25, n_estimators=100, criterion='squared_error') - print(res['te_pred_prob']) - - print('####' * 5) - res = gradient_boosting_regression(df, 'target', test_size=0.25, n_estimators=100, learning_rate=0.1) - print(res['te_pred_prob']) - run() \ No newline at end of file diff --git a/metagpt/tools/functions/register/__init__.py b/metagpt/tools/functions/register/__init__.py deleted file mode 100644 index c80872750..000000000 --- a/metagpt/tools/functions/register/__init__.py +++ /dev/null @@ -1,6 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/16 16:37 -# @Author : lidanyang -# @File : __init__.py -# @Desc : diff --git a/metagpt/tools/functions/register/register.py b/metagpt/tools/functions/register/register.py deleted file mode 100644 index 0731e31c0..000000000 --- a/metagpt/tools/functions/register/register.py +++ /dev/null @@ -1,78 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/16 16:38 -# @Author : lidanyang -# @File : register.py -# @Desc : -import inspect -from typing import Type, Optional, Callable, Dict, Union, List - -from metagpt.tools.functions.schemas.base import ToolSchema - - -class FunctionRegistry: - def __init__(self): - self.functions: Dict[str, Dict[str, Dict]] = {} - - @staticmethod - def _check_param_consistency(func_params, schema): - param_names = set(func_params.keys()) - schema_names = set(schema["parameters"]["properties"].keys()) - - if param_names != schema_names: - raise ValueError("Function parameters do not match schema properties") - - def register(self, module: str, tool_schema: Type[ToolSchema]) -> Callable: - def wrapper(func: Callable) -> Callable: - module_registry = self.functions.setdefault(module, {}) - - if func.__name__ in module_registry: - raise ValueError(f"Function {func.__name__} is already registered in {module}") - - func_params = inspect.signature(func).parameters - - schema = tool_schema.schema() - schema["name"] = func.__name__ - - self._check_param_consistency(func_params, schema) - - module_registry[func.__name__] = { - "func": func, - "schema": schema, - } - return func - - return wrapper - - def get(self, module: str, name: str) -> Optional[Union[Callable, Dict]]: - """Get function by module and name""" - module_registry = self.functions.get(module, {}) - return module_registry.get(name) - - def get_by_name(self, name: str) -> Optional[Dict]: - """Get function by name""" - for module_registry in self.functions.values(): - if name in module_registry: - return module_registry.get(name, {}) - - def get_all_by_module(self, module: str) -> Optional[Dict]: - """Get all functions by module""" - return self.functions.get(module, {}) - - def get_schema(self, module: str, name: str) -> Optional[Dict]: - """Get schema by module and name""" - module_registry = self.functions.get(module, {}) - return module_registry.get(name, {}).get("schema") - - def get_schemas(self, module: str, names: List[str]) -> List[Dict]: - """Get schemas by module and names""" - module_registry = self.functions.get(module, {}) - return [module_registry.get(name, {}).get("schema") for name in names] - - def get_all_schema_by_module(self, module: str) -> List[Dict]: - """Get all schemas by module""" - module_registry = self.functions.get(module, {}) - return [v.get("schema") for v in module_registry.values()] - - -registry = FunctionRegistry() diff --git a/metagpt/tools/functions/schemas/base.py b/metagpt/tools/functions/schemas/base.py deleted file mode 100644 index aef604c8d..000000000 --- a/metagpt/tools/functions/schemas/base.py +++ /dev/null @@ -1,100 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/16 16:34 -# @Author : lidanyang -# @File : base.py -# @Desc : Build base class to generate schema for tool -from typing import Any, List, Optional, get_type_hints - - -class NoDefault: - """ - A class to represent a missing default value. - - This is used to distinguish between a default value of None and a missing default value. - """ - pass - - -def tool_field( - description: str, default: Any = NoDefault(), enum: Optional[List[Any]] = None, **kwargs -): - """ - Create a field for a tool parameter. - - Args: - description (str): A description of the field. - default (Any, optional): The default value for the field. Defaults to None. - enum (Optional[List[Any]], optional): A list of possible values for the field. Defaults to None. - **kwargs: Additional keyword arguments. - - Returns: - dict: A dictionary representing the field with provided attributes. - """ - field_info = { - "description": description, - "default": default, - "enum": enum, - } - field_info.update(kwargs) - return field_info - - -class ToolSchema: - @staticmethod - def format_type(type_hint): - """ - Format a type hint into a string representation. - - Args: - type_hint (type): The type hint to format. - - Returns: - str: A string representation of the type hint. - """ - if isinstance(type_hint, type): - # Handle built-in types separately - if type_hint.__module__ == "builtins": - return type_hint.__name__ - else: - return f"{type_hint.__module__}.{type_hint.__name__}" - elif hasattr(type_hint, "__origin__") and hasattr(type_hint, "__args__"): - # Handle generic types (like List[int]) - origin_type = ToolSchema.format_type(type_hint.__origin__) - args_type = ", ".join( - [ToolSchema.format_type(t) for t in type_hint.__args__] - ) - return f"{origin_type}[{args_type}]" - else: - return str(type_hint) - - @classmethod - def schema(cls): - """ - Generate a schema dictionary for the class. - - The schema includes the class name, description, and information about - each class parameter based on type hints and field definitions. - - Returns: - dict: A dictionary representing the schema of the class. - """ - schema = { - "name": cls.__name__, - "description": cls.__doc__, - "parameters": {"type": "object", "properties": {}, "required": []}, - } - type_hints = get_type_hints(cls) - for attr, type_hint in type_hints.items(): - value = getattr(cls, attr, None) - if isinstance(value, dict): - # Process each attribute that is defined using the field function - prop_info = {k: v for k, v in value.items() if v is not None or k == "default"} - if isinstance(prop_info["default"], NoDefault): - del prop_info["default"] - prop_info["type"] = ToolSchema.format_type(type_hint) - schema["parameters"]["properties"][attr] = prop_info - # Check for required fields - if "default" not in prop_info: - schema["parameters"]["required"].append(attr) - return schema diff --git a/metagpt/tools/functions/schemas/data_preprocess.py b/metagpt/tools/functions/schemas/data_preprocess.py deleted file mode 100644 index 16b97aeac..000000000 --- a/metagpt/tools/functions/schemas/data_preprocess.py +++ /dev/null @@ -1,67 +0,0 @@ - -import pandas as pd - -from metagpt.tools.functions.schemas.base import tool_field, ToolSchema - - -class FillMissingValue(ToolSchema): - """Completing missing values with simple strategies""" - df: pd.DataFrame = tool_field(description="input dataframe") - features: list = tool_field(description="columns to be processed") - strategy: str = tool_field( - description="the imputation strategy", - default='mean', - enum=['mean', 'median', 'most_frequent', 'constant'] - ) - fill_value: int = tool_field( - description="fill_value is used to replace all occurrences of missing_values", default=None) - - -class SplitBins(ToolSchema): - """Bin continuous data into intervals and return the bin identifier encoded as an integer value""" - df: pd.DataFrame = tool_field(description="input dataframe") - features: list = tool_field(description="columns to be processed") - strategy: str = tool_field(description="Strategy used to define the widths of the bins", default='quantile') - - -class MinMaxScale(ToolSchema): - """Transform features by scaling each feature to a range, witch is (0, 1)""" - df: pd.DataFrame = tool_field(description="input dataframe") - features: list = tool_field(description="columns to be processed") - - -class StandardScale(ToolSchema): - """Standardize features by removing the mean and scaling to unit variance""" - df: pd.DataFrame = tool_field(description="input dataframe") - features: list = tool_field(description="columns to be processed") - - -class LogTransform(ToolSchema): - """Performs a logarithmic transformation on the specified columns""" - df: pd.DataFrame = tool_field(description="input dataframe") - features: list = tool_field(description="columns to be processed") - - -class MaxAbsScale(ToolSchema): - """Scale each feature by its maximum absolute value""" - df: pd.DataFrame = tool_field(description="input dataframe") - features: list = tool_field(description="columns to be processed") - - -class RobustScale(ToolSchema): - """Scale features using statistics that are robust to outliers, the quantile_range is (25.0, 75.0)""" - df: pd.DataFrame = tool_field(description="input dataframe") - features: list = tool_field(description="columns to be processed") - - -class OrdinalEncode(ToolSchema): - """Encode categorical features as an integer array""" - df: pd.DataFrame = tool_field(description="input dataframe") - features: list = tool_field(description="columns to be processed") - - -class OneHotEncoding(ToolSchema): - """Apply one-hot encoding to specified categorical columns, the original columns will be dropped.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - cols: list = tool_field(description="Categorical columns to be one-hot encoded and dropped.") diff --git a/metagpt/tools/functions/schemas/feature_engineering.py b/metagpt/tools/functions/schemas/feature_engineering.py deleted file mode 100644 index 5c89d9b16..000000000 --- a/metagpt/tools/functions/schemas/feature_engineering.py +++ /dev/null @@ -1,110 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/17 10:34 -# @Author : lidanyang -# @File : feature_engineering.py -# @Desc : Schema for feature engineering functions -from typing import List - -import pandas as pd - -from metagpt.tools.functions.schemas.base import ToolSchema, tool_field - - -class PolynomialExpansion(ToolSchema): - """Add polynomial and interaction features from selected numeric columns, excluding the bias column.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - cols: list = tool_field(description="Columns for polynomial expansion.") - degree: int = tool_field(description="Degree of polynomial features.", default=2) - - -class FrequencyEncoding(ToolSchema): - """Add value counts of categorical columns as new features.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - cols: list = tool_field(description="Categorical columns to be frequency encoded.") - - -class TargetMeanEncoder(ToolSchema): - """Encodes a categorical column by the mean of the label column, and adds the result as a new feature.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - col: str = tool_field(description="Column to be mean encoded.") - label: str = tool_field(description="Predicted label column.") - - -class KFoldTargetMeanEncoder(ToolSchema): - """Adds a new feature to the DataFrame by k-fold mean encoding of a categorical column using the label column.""" - df: pd.DataFrame = tool_field(description="DataFrame to process.") - col: str = tool_field(description="Column to be k-fold mean encoded.") - label: str = tool_field(description="Predicted label column.") - n_splits: int = tool_field(description="Number of splits for K-fold.", default=5) - random_state: int = tool_field(description="Random seed.", default=2021) - - -class CatCross(ToolSchema): - """Add pairwise crossed features and convert them to numerical features.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - cols: list = tool_field(description="Columns to be pairwise crossed.") - max_cat_num: int = tool_field( - description="Maximum unique categories per crossed feature.", default=100 - ) - - -class GroupStat(ToolSchema): - """Aggregate specified column in a DataFrame grouped by another column, adding new features named '__by_'.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - group_col: str = tool_field(description="Column used for grouping.") - agg_col: str = tool_field(description="Column on which aggregation is performed.") - agg_funcs: list = tool_field( - description="""List of aggregation functions to apply, such as ['mean', 'std']. - Each function must be supported by pandas.""" - ) - - -class ExtractTimeComps(ToolSchema): - """Extract and add specific time components as new features from a designated time column.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - time_col: str = tool_field( - description="The name of the column containing time data." - ) - time_comps: List[str] = tool_field( - description="""List of time components to extract. - Each component must be in ['year', 'month', 'day', 'hour', 'dayofweek', 'is_weekend'].""" - ) - - -class FeShiftByTime(ToolSchema): - """Shift column values based on specified time intervals and add the resulting new features to the DataFrame. New features are named in the format of '__lag__'.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - time_col: str = tool_field(description="Column for time-based shifting.") - group_col: str = tool_field(description="Column for grouping before shifting.") - shift_col: str = tool_field(description="Column to shift.") - periods: list = tool_field(description="Time intervals for shifting.") - freq: str = tool_field( - description="Frequency unit for time intervals (e.g., 'D', 'M').", - enum=["D", "M", "Y", "W", "H"], - ) - - -class FeRollingByTime(ToolSchema): - """Calculate rolling statistics for a DataFrame column over time intervals.""" - - df: pd.DataFrame = tool_field(description="DataFrame to process.") - time_col: str = tool_field(description="Column for time-based rolling.") - group_col: str = tool_field(description="Column for grouping before rolling.") - rolling_col: str = tool_field(description="Column for rolling calculations.") - periods: list = tool_field(description="Window sizes for rolling.") - freq: str = tool_field( - description="Frequency unit for time windows (e.g., 'D', 'M').", - enum=["D", "M", "Y", "W", "H"], - ) - agg_funcs: list = tool_field( - description="""List of aggregation functions for rolling, like ['mean', 'std']. - Each function must be in ['mean', 'std', 'min', 'max', 'median', 'sum', 'count'].""" - ) diff --git a/metagpt/tools/functions/schemas/ml_model.py b/metagpt/tools/functions/schemas/ml_model.py deleted file mode 100644 index 9268156af..000000000 --- a/metagpt/tools/functions/schemas/ml_model.py +++ /dev/null @@ -1,55 +0,0 @@ -import pandas as pd - -from metagpt.tools.functions.schemas.base import tool_field, ToolSchema - - -class LogisticRegressionClassification(ToolSchema): - """Logistic Regression (aka logit, MaxEnt) classifier""" - df: pd.DataFrame = tool_field(description="input dataframe") - label: str = tool_field(description="target name") - test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) - penalty: str = tool_field(description="Specify the norm of the penalty", default="l2") - dual: bool = tool_field(description="Dual (constrained) or primal (regularized) formulation", default="l2") - - -class RandomForestClassification(ToolSchema): - """random forest is a meta estimator that fits a number of decision tree classifiers on various sub-samples of the dataset and uses averaging to improve the predictive accuracy and control over-fitting""" - df: pd.DataFrame = tool_field(description="input dataframe") - label: str = tool_field(description="target name") - test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) - n_estimators: int = tool_field(description="The number of trees in the forest", default=100) - criterion: str = tool_field(description="The function to measure the quality of a split", default="gini") - - -class GradientBoostingClassification(ToolSchema): - """Gradient Boosting for classification.This algorithm builds an additive model in a forward stage-wise fashion""" - df: pd.DataFrame = tool_field(description="input dataframe") - label: str = tool_field(description="target name") - test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) - n_estimators: int = tool_field(description="The number of boosting stages to perform", default=100) - learning_rate: float = tool_field(description="Learning rate shrinks the contribution of each tree by learning_rate", default=0.1) - - -class LinearRegressionRegression(ToolSchema): - """Ordinary least squares Linear Regression.""" - df: pd.DataFrame = tool_field(description="input dataframe") - label: str = tool_field(description="target name") - test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) - - -class RandomForestRegression(ToolSchema): - """random forest is a meta estimator that fits a number of decision tree on various sub-samples of the dataset and uses averaging to improve the predictive accuracy and control over-fitting""" - df: pd.DataFrame = tool_field(description="input dataframe") - label: str = tool_field(description="target name") - test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) - n_estimators: int = tool_field(description="The number of trees in the forest", default=100) - criterion: str = tool_field(description="The function to measure the quality of a split", default="squared_error") - - -class GradientBoostingRegression(ToolSchema): - """Gradient Boosting for regression.This estimator builds an additive model in a forward stage-wise fashion""" - df: pd.DataFrame = tool_field(description="input dataframe") - label: str = tool_field(description="target name") - test_size: float = tool_field(description="The proportion of the test set to all the data", default=0.2) - n_estimators: int = tool_field(description="The number of boosting stages to perform", default=100) - learning_rate: float = tool_field(description="Learning rate shrinks the contribution of each tree by learning_rate", default=0.1) From 2a3f23ec62ebca8329c2748179d731025a685d0a Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 14 Dec 2023 10:32:58 +0800 Subject: [PATCH 147/637] fix unittest --- .../actions/test_write_analysis_code.py | 33 ++++++++----------- 1 file changed, 13 insertions(+), 20 deletions(-) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index 661202115..1a568cdcd 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -31,22 +31,15 @@ async def test_tool_recommendation(): step 1: 对数据集进行去重 step 2: 对数据集进行缺失值处理 """ - available_tools = [ - { - "name": "fill_missing_value", - "description": "Completing missing values with simple strategies", - }, - { - "name": "split_bins", - "description": "Bin continuous data into intervals and return the bin identifier encoded as an integer value", - }, - ] + available_tools = { + "fill_missing_value": "Completing missing values with simple strategies", + "split_bins": "Bin continuous data into intervals and return the bin identifier encoded as an integer value", + } write_code = WriteCodeWithTools() tools = await write_code._tool_recommendation(task, code_steps, available_tools) - assert len(tools) == 2 - assert tools[0] == [] - assert tools[1] == ["fill_missing_value"] + assert len(tools) == 1 + assert tools[0] == ["fill_missing_value"] @pytest.mark.asyncio @@ -57,7 +50,7 @@ async def test_write_code_with_tools(): "1": Task( task_id="1", instruction="随机生成一个pandas DataFrame数据集", - task_type="unknown", + task_type="other", dependent_task_ids=[], code=""" import pandas as pd @@ -75,6 +68,10 @@ async def test_write_code_with_tools(): instruction="对数据集进行数据清洗", task_type="data_preprocess", dependent_task_ids=["1"], + code_steps=""" + {"Step 1": "对数据集进行去重", + "Step 2": "对数据集进行缺失值处理"} + """ ), } plan = Plan( @@ -83,13 +80,9 @@ async def test_write_code_with_tools(): task_map=task_map, current_task_id="2", ) - task_guide = """ - step 1: 对数据集进行去重 - step 2: 对数据集进行缺失值处理 - """ - data_desc = "None" + column_info = "" - code = await write_code.run(messages, plan, task_guide, data_desc) + code = await write_code.run(messages, plan, column_info) assert len(code) > 0 print(code) From d84e9cae2c8dfc5345edb253f59ca1f0901cacab Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 14 Dec 2023 10:34:15 +0800 Subject: [PATCH 148/637] fix conflict --- metagpt/actions/write_analysis_code.py | 6 ++---- metagpt/roles/ml_engineer.py | 12 ++++++------ 2 files changed, 8 insertions(+), 10 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 2c45281f9..6970fb4f0 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -24,8 +24,8 @@ from metagpt.utils.common import create_func_config, remove_comments class BaseWriteAnalysisCode(Action): - DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt - REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" + DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt + # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG @@ -201,8 +201,6 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name=module_name, tool_catalog=tool_catalog, ) - - else: prompt = GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 8f06a541c..0b76711f4 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -159,12 +159,12 @@ class MLEngineer(Role): # print("*" * 10) # breakpoint() if counter > 0: - improve_code = await DebugCode().run(plan=self.plan.current_task.instruction, - # finished_code=code_context, - # finished_code_result=code_result, - code=code, - runtime_result=self.working_memory.get(), - context=debug_context) + improve_code = await DebugCode().run( + plan=self.plan.current_task.instruction, + code=code, + runtime_result=self.working_memory.get(), + context=debug_context + ) if improve_code != "": code = improve_code From e4ee3efeb89f24762baa2758a23962fb544e6df1 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 10:46:43 +0800 Subject: [PATCH 149/637] =?UTF-8?q?update:=20=E6=8C=89code=20step=20?= =?UTF-8?q?=E9=80=90=E4=B8=80=E7=94=9F=E6=88=90=E4=BB=A3=E7=A0=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/actions/write_analysis_code.py | 30 ++++++++++++++++++++++++-- 1 file changed, 28 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 3e91f4b14..1cfc28811 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -202,8 +202,34 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name=module_name, tool_catalog=tool_catalog, ) - - + code_steps_ = eval(code_steps) + print(code_steps_) + + new_code = "" + tool_context = "" + for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): + prompt = TOOL_USAGE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + special_prompt=special_prompt, + code_steps=step_instruction, + module_name=module_name, + tool_catalog=tool_catalog, + ) + + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + + rsp = await self.llm.aask_code(prompt, **tool_config) + logger.info(f"rsp is: {rsp}") + new_code = new_code + "\n\n" + rsp["code"] + code_context = code_context + "\n\n" + new_code + tool_context = tool_context + "\n\n" + prompt + context = [Message(content=tool_context, role="user")] + return context, new_code + + else: prompt = GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, From 31bd653f07a3d974ef163bfaf9f89e9cb133da9e Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 10:47:05 +0800 Subject: [PATCH 150/637] =?UTF-8?q?update:=20=E5=8E=BB=E9=99=A4=E5=9B=9E?= =?UTF-8?q?=E8=BD=A6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/actions/write_code_steps.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py index 9e06bc91e..89bf8980f 100644 --- a/metagpt/actions/write_code_steps.py +++ b/metagpt/actions/write_code_steps.py @@ -106,7 +106,7 @@ class WriteCodeSteps(Action): def process_task(task): task_dict = task.dict() # ptask = {k: task_dict[k] for k in task_dict if k in select_task_keys } - ptask = f"task_id_{task_dict['task_id']}:{task_dict['instruction']}\n" + ptask = f"task_id_{task_dict['task_id']}:{task_dict['instruction']}" return ptask From 41e872a8c0cee55c81fefeb82f737ce1210721c6 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 10:47:47 +0800 Subject: [PATCH 151/637] =?UTF-8?q?update:=20=E6=9B=B4=E6=96=B0prompt?= =?UTF-8?q?=EF=BC=8C=E7=BB=99=E5=87=BA=E5=8D=95step-code=E7=A4=BA=E4=BE=8B?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/prompts/ml_engineer.py | 20 ++++++++++++++++++-- metagpt/roles/ml_engineer.py | 18 +++++++++++------- 2 files changed, 29 insertions(+), 9 deletions(-) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 2d2d3315a..ae6938ee0 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -208,8 +208,10 @@ Follow steps below when you writing code if it's convenient. Each Class tool is described in JSON format. When you call it, import the tool from `{module_name}` first. {tool_catalog} -# Output Example: -For "fill missing value and handle outliers", the output code be like when there are training data and test data: +# Step Example: +Here is a coding example for each code step: +[Step 1]: Handle missing values by imputing or dropping them. For numerical columns, use median or mean imputation +[Code] ```python # Tools used: ['FillMissingValue'] from metagpt.tools.functions.libs.data_preprocess import FillMissingValue @@ -227,12 +229,26 @@ for col in num_cols: train_processed[col] = train_processed[col].clip(low, high) test_processed[col] = test_processed[col].clip(low, high) ```end +[Step 2]: xxx +[Code]: +```python +# Tools used: [xxx] +from metagpt.tools.functions.libs.xxx import +```end +[Step 3]: xxx +[Code]: +```python +# Tools used: [xxx] +from metagpt.tools.functions.libs.xxx import +```end # Constraints: - Prioritize using pre-defined tools for the same functionality. - Copy DataFrame before processing if needed. +- Strictly follow the code steps to write code """ #- If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. +#For "fill missing value and handle outliers", the output code be like when there are training data and test data: DATA_PREPROCESS_PROMPT = """ The current task is about data preprocessing, please note the following: diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index c735eb983..357fdbe09 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -33,14 +33,13 @@ STRUCTURAL_CONTEXT = """ ## Current Task {current_task} ## Packages Installed -scikit-learn pandas numpy -lightgbm -xgboost -catboost """ - +# scikit-learn +# lightgbm +# xgboost +# catboost def truncate(result: str, keep_len: int = 1000) -> str: desc = "Truncated to show only the last 1000 characters\n" @@ -290,11 +289,16 @@ if __name__ == "__main__": # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." - data_path = f"{DATA_PATH}/titanic" - requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + # data_path = f"{DATA_PATH}/titanic" + # requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." # requirement = f"Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." + # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" + # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + + data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" + requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." async def main(requirement: str = requirement, auto_run: bool = True): role = MLEngineer(goal=requirement, auto_run=auto_run) await role.run(requirement) From 44334c0c9aa6b8a0d6314d3b24623d9633ce7c2d Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 14 Dec 2023 10:59:42 +0800 Subject: [PATCH 152/637] drop old schema import --- metagpt/tools/functions/libs/data_preprocess.py | 7 +++++-- metagpt/tools/functions/libs/feature_engineering.py | 2 +- 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index fa70bf8fc..ec3580889 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -1,4 +1,5 @@ import numpy as np +import pandas as pd from sklearn.impute import SimpleImputer from sklearn.preprocessing import LabelEncoder from sklearn.preprocessing import MaxAbsScaler @@ -9,7 +10,6 @@ from sklearn.preprocessing import RobustScaler from sklearn.preprocessing import StandardScaler from metagpt.tools.functions.libs.base import MLProcess -from metagpt.tools.functions.schemas.data_preprocess import * class FillMissingValue(MLProcess): @@ -141,7 +141,10 @@ def get_column_info(df: pd.DataFrame) -> dict: for i in df.columns: nan_freq = float("%.2g" % (df[i].isna().mean() * 100)) n_unique = df[i].nunique() - data.append([i, df[i].dtype, nan_freq, n_unique]) + data_type = str(df[i].dtype).replace("dtype('", "").replace("')", "") + if data_type == "O": + data_type = "object" + data.append([i, data_type, nan_freq, n_unique]) samples = pd.DataFrame( data, diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index de54e4db0..1ec2b9675 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -7,6 +7,7 @@ import itertools import numpy as np +import pandas as pd from dateutil.relativedelta import relativedelta from joblib import Parallel, delayed from pandas.api.types import is_numeric_dtype @@ -15,7 +16,6 @@ from sklearn.model_selection import KFold from sklearn.preprocessing import PolynomialFeatures, KBinsDiscretizer from metagpt.tools.functions.libs.base import MLProcess -from metagpt.tools.functions.schemas.feature_engineering import * class PolynomialExpansion(MLProcess): From 5940c8d908b12d8c99cc03305dec4fcf8bcc3dd8 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 14 Dec 2023 12:56:01 +0800 Subject: [PATCH 153/637] remove old comments --- metagpt/roles/ml_engineer.py | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 0b76711f4..51faf1e0d 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -261,14 +261,12 @@ class MLEngineer(Role): """find useful memories only to reduce context length and improve performance""" # TODO dataset description , code steps if task_exclude_field is None: + # Shorten the context as we don't need code steps after we get the codes. + # This doesn't affect current_task below, which should hold the code steps task_exclude_field = {'code_steps'} user_requirement = self.plan.goal data_desc = self.plan.context tasks = [task.dict(exclude=task_exclude_field) for task in self.plan.tasks] - # for task in tasks: - # # Shorten the context as we don't need code steps after we get the codes. - # # This doesn't affect current_task below, which should hold the code steps - # task.pop("code_steps") tasks = json.dumps(tasks, indent=4, ensure_ascii=False) current_task = self.plan.current_task.json() if self.plan.current_task else {} context = STRUCTURAL_CONTEXT.format( From ef6e4a1b77a21cefeb165301dd1d47b5c273fdbb Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 14 Dec 2023 13:46:27 +0800 Subject: [PATCH 154/637] debug only when use_tools --- metagpt/roles/ml_engineer.py | 10 +++------- 1 file changed, 3 insertions(+), 7 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 51faf1e0d..3755e7bac 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -147,7 +147,6 @@ class MLEngineer(Role): ) counter = 0 - improve_code = "" success = False debug_context = [] @@ -158,17 +157,14 @@ class MLEngineer(Role): # print(context) # print("*" * 10) # breakpoint() - if counter > 0: - improve_code = await DebugCode().run( + if counter > 0 and self.use_tools: + code = await DebugCode().run( plan=self.plan.current_task.instruction, code=code, runtime_result=self.working_memory.get(), context=debug_context ) - - if improve_code != "": - code = improve_code - logger.info(f"new code \n{improve_code}") + logger.info(f"new code \n{code}") cause_by = DebugCode elif not self.use_tools or self.plan.current_task.task_type == "other": logger.info("Write code with pure generation") From 97f707784bd8558b3bbd138d9380af55bb85f9a4 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 14 Dec 2023 13:56:23 +0800 Subject: [PATCH 155/637] reformat --- metagpt/actions/debug_code.py | 124 ++++++++++++++++++---------------- 1 file changed, 64 insertions(+), 60 deletions(-) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 58d006a08..3e1705d8e 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -1,57 +1,56 @@ from typing import Dict, List, Union, Tuple, Optional, Any -from metagpt.actions import Action from metagpt.logs import logger from metagpt.schema import Message, Plan from metagpt.utils.common import CodeParser, create_func_config from metagpt.actions.write_analysis_code import BaseWriteAnalysisCode -DEBUG_REFLECTION_EXAMPLE = '''Example 1: - [previous impl]: - ```python - def add(a: int, b: int) -> int: - """ - Given integers a and b, return the total value of a and b. - """ - return a - b - ``` +DEBUG_REFLECTION_EXAMPLE = ''' +Example 1: +[previous impl]: +```python +def add(a: int, b: int) -> int: + """ + Given integers a and b, return the total value of a and b. + """ + return a - b +``` - [runtime Error]: - Tested passed: +[runtime Error]: +Tested passed: - Tests failed: - assert add(1, 2) == 3 # output: -1 - assert add(1, 2) == 4 # output: -1 +Tests failed: +assert add(1, 2) == 3 # output: -1 +assert add(1, 2) == 4 # output: -1 - [reflection on previous impl]: - The implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input. +[reflection on previous impl]: +The implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input. - [improved impl]: - ```python - def add(a: int, b: int) -> int: - """ - Given integers a and b, return the total value of a and b. - """ - return a + b - ``` - ''' +[improved impl]: +```python +def add(a: int, b: int) -> int: + """ + Given integers a and b, return the total value of a and b. + """ + return a + b +``` +''' REFLECTION_PROMPT = """ - Here is an example for you. - {debug_example} - [context] - {context} - - [previous impl] - {code} - [runtime Error] - {runtime_result} +Here is an example for you. +{debug_example} +[context] +{context} - Analysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step. - [reflection on previous impl]: - xxx +[previous impl] +{code} +[runtime Error] +{runtime_result} - """ +Analysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step. +[reflection on previous impl]: +xxx +""" CODE_REFLECTION = { "name": "execute_reflection_code", @@ -85,10 +84,10 @@ class DebugCode(BaseWriteAnalysisCode): name: str = "debugcode" context: Optional[str] = None llm: None - + def __init__(self, **kwargs: Any): super().__init__(**kwargs) - + async def run_reflection( self, # goal, @@ -100,23 +99,26 @@ class DebugCode(BaseWriteAnalysisCode): ) -> dict: info = [] # finished_code_and_result = finished_code + "\n [finished results]\n\n" + finished_code_result - reflection_prompt = REFLECTION_PROMPT.format(debug_example=DEBUG_REFLECTION_EXAMPLE, - context=context, - # goal=goal, - # finished_code=finished_code_and_result, - code=code, - runtime_result=runtime_result - ) + reflection_prompt = REFLECTION_PROMPT.format( + debug_example=DEBUG_REFLECTION_EXAMPLE, + context=context, + # goal=goal, + # finished_code=finished_code_and_result, + code=code, + runtime_result=runtime_result, + ) system_prompt = "You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation " info.append(Message(role="system", content=system_prompt)) info.append(Message(role="user", content=reflection_prompt)) - + # msg = messages_to_str(info) # resp = await self.llm.aask(msg=msg) - resp = await self.llm.aask_code(messages=info, **create_func_config(CODE_REFLECTION)) + resp = await self.llm.aask_code( + messages=info, **create_func_config(CODE_REFLECTION) + ) logger.info(f"reflection is {resp}") return resp - + # async def rewrite_code(self, reflection: str = "", context: List[Message] = None) -> str: # """ # 根据reflection重写代码 @@ -131,14 +133,16 @@ class DebugCode(BaseWriteAnalysisCode): # resp = await self.llm.aask(msg=msg) # improv_code = CodeParser.parse_code(block=None, text=resp) # return improv_code - - async def run(self, - context: List[Message] = None, - plan: str = "", - # finished_code: str = "", - # finished_code_result: str = "", - code: str = "", - runtime_result: str = "") -> str: + + async def run( + self, + context: List[Message] = None, + plan: str = "", + # finished_code: str = "", + # finished_code_result: str = "", + code: str = "", + runtime_result: str = "", + ) -> str: """ 根据当前运行代码和报错信息进行reflection和纠错 """ @@ -152,5 +156,5 @@ class DebugCode(BaseWriteAnalysisCode): ) # 根据reflection结果重写代码 # improv_code = await self.rewrite_code(reflection, context=context) - improv_code = reflection['improved_impl'] + improv_code = reflection["improved_impl"] return improv_code From 2da141abbe43fa2c046a8f4bbdb0edc9325b03d3 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 14 Dec 2023 13:57:39 +0800 Subject: [PATCH 156/637] recover code --- metagpt/tools/web_browser_engine.py | 2 +- metagpt/utils/__init__.py | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/metagpt/tools/web_browser_engine.py b/metagpt/tools/web_browser_engine.py index 7228ae9cf..453d87f31 100644 --- a/metagpt/tools/web_browser_engine.py +++ b/metagpt/tools/web_browser_engine.py @@ -7,7 +7,7 @@ from typing import Any, Callable, Coroutine, Literal, overload from metagpt.config import CONFIG from metagpt.tools import WebBrowserEngineType -# from metagpt.utils.parse_html import WebPage +from metagpt.utils.parse_html import WebPage class WebBrowserEngine: diff --git a/metagpt/utils/__init__.py b/metagpt/utils/__init__.py index 86cac50db..f13175cf8 100644 --- a/metagpt/utils/__init__.py +++ b/metagpt/utils/__init__.py @@ -6,7 +6,7 @@ @File : __init__.py """ -# from metagpt.utils.read_document import read_docx +from metagpt.utils.read_document import read_docx from metagpt.utils.singleton import Singleton from metagpt.utils.token_counter import ( TOKEN_COSTS, @@ -16,7 +16,7 @@ from metagpt.utils.token_counter import ( __all__ = [ - # "read_docx", + "read_docx", "Singleton", "TOKEN_COSTS", "count_message_tokens", From 234ffdab355f729ddfc385aa20bb6676e314174d Mon Sep 17 00:00:00 2001 From: Zhou <1359698378@qq.com> Date: Thu, 14 Dec 2023 14:37:27 +0800 Subject: [PATCH 157/637] remove old typing-extensions version --- requirements.txt | 1 - 1 file changed, 1 deletion(-) diff --git a/requirements.txt b/requirements.txt index 1d1bc95a1..2328de2a1 100644 --- a/requirements.txt +++ b/requirements.txt @@ -35,7 +35,6 @@ tqdm==4.64.0 # webdriver_manager<3.9 anthropic==0.3.6 typing-inspect==0.8.0 -typing_extensions==4.5.0 libcst==1.0.1 qdrant-client==1.4.0 pytest-mock==3.11.1 From 70fdb1905f8b6a615b4557cc9278454381d26983 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 15:33:00 +0800 Subject: [PATCH 158/637] =?UTF-8?q?=E6=9B=B4=E6=96=B0=EF=BC=9A=E4=BF=AE?= =?UTF-8?q?=E6=94=B9execute=5Fcode=20=E5=88=9D=E5=A7=8B=E5=8C=96=20?= =?UTF-8?q?=E5=A2=9E=E5=8A=A0resume=E5=8A=9F=E8=83=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- metagpt/actions/execute_code.py | 8 ++- metagpt/roles/ml_engineer.py | 119 ++++++++++++++++++++++++++------ 2 files changed, 102 insertions(+), 25 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 6fd980494..54d2cf348 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -45,9 +45,12 @@ class ExecuteCode(ABC): class ExecutePyCode(ExecuteCode, Action): """execute code, return result to llm, and display it.""" - def __init__(self, name: str = "python_executor", context=None, llm=None): + def __init__(self, name: str = "python_executor", context=None, llm=None, nb=None): super().__init__(name, context, llm) - self.nb = nbformat.v4.new_notebook() + if nb is None: + self.nb = nbformat.v4.new_notebook() + else: + self.nb = nb self.nb_client = NotebookClient(self.nb) self.console = Console() self.interaction = "ipython" if self.is_ipython() else "terminal" @@ -158,6 +161,7 @@ class ExecutePyCode(ExecuteCode, Action): def save_notebook(self, path: str): path = Path(path) + print(path) path.parent.mkdir(parents=True, exist_ok=True) nbformat.write(self.nb, path) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 357fdbe09..e8b0bda16 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -4,7 +4,8 @@ from datetime import datetime from typing import List import fire -import pandas as pd +import nbformat +from pathlib import Path from metagpt.actions import Action from metagpt.actions.execute_code import ExecutePyCode @@ -36,6 +37,8 @@ STRUCTURAL_CONTEXT = """ pandas numpy """ + + # scikit-learn # lightgbm # xgboost @@ -129,17 +132,18 @@ class MLEngineer(Role): task.code_steps = code_steps self.plan.finish_current_task() self.working_memory.clear() - + success, new_code = await self._update_data_columns() if success: task.code = task.code + "\n\n" + new_code + else: # update plan according to user's feedback and to take on changed tasks await self._update_plan() - + time = datetime.now().strftime('%Y-%m-%d_%H-%M-%S') self.execute_code.save_notebook(f"{DATA_PATH}/notebooks/ml_{time}.ipynb") - + async def _update_data_columns(self): rsp = await UpdateDataColumns().run(self.plan) is_update, code = rsp["is_update"], rsp["code"] @@ -149,7 +153,7 @@ class MLEngineer(Role): if success: self.data_desc["column_info"] = result return success, code - + async def _write_and_exec_code(self, max_retry: int = 3): code_steps = ( await WriteCodeSteps().run(self.plan) @@ -162,23 +166,15 @@ class MLEngineer(Role): success = False debug_context = [] - finished_tasks = self.plan.get_finished_tasks() - code_context = [task.code for task in finished_tasks] - code_result = [task.result for task in finished_tasks] - code_context = "\n\n".join(code_context) - code_result = "\n\n".join(code_result) - while not success and counter < max_retry: context = self.get_useful_memories() - + if counter > 0: improve_code = await DebugCode().run(plan=self.plan.current_task.instruction, - # finished_code=code_context, - # finished_code_result=code_result, code=code, runtime_result=self.working_memory.get(), context=debug_context) - + if improve_code != "": code = improve_code logger.info(f"new code \n{improve_code}") @@ -217,7 +213,7 @@ class MLEngineer(Role): ) if "!pip" in code: - success = False + success = False # if not success: # await self._ask_review() @@ -294,14 +290,91 @@ if __name__ == "__main__": # requirement = f"Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." - # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" - # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" - requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." - async def main(requirement: str = requirement, auto_run: bool = True): - role = MLEngineer(goal=requirement, auto_run=auto_run) - await role.run(requirement) + data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" + requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" + # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." + + + save_dir = "" + save_dir = DATA_PATH / "save" / "2023-12-14_15-11-40" + + + def load_history(save_dir: str = save_dir): + """ + Load history from the specified save directory. + + Args: + save_dir (str): The directory from which to load the history. + + Returns: + Tuple: A tuple containing the loaded plan and notebook. + """ + + plan_path = Path(save_dir) / "plan.json" + nb_path = Path(save_dir) / "history_nb.ipynb" + plan = json.load(open(plan_path, "r", encoding="utf-8")) + nb = nbformat.read(open(nb_path, "r", encoding="utf-8"), as_version=nbformat.NO_CONVERT) + return plan, nb + + + async def save_history(role: Role = MLEngineer, save_dir: str = save_dir): + """ + Save history to the specified directory. + + Args: + role (Role): The role containing the plan and execute_code attributes. + save_dir (str): The directory to save the history. + + Returns: + Path: The path to the saved history directory. + """ + save_path = Path(save_dir) if save_dir else DATA_PATH / "save" / datetime.now().strftime( + '%Y-%m-%d_%H-%M-%S') + # overwrite + save_path.mkdir(parents=True, exist_ok=True) + + plan = role.plan.dict() + logger.info(f"Plan is {plan}") + + with open(save_path / "plan.json", "w", encoding="utf-8") as plan_file: + json.dump(plan, plan_file, indent=4, ensure_ascii=False) + + role.execute_code.save_notebook(path=save_path / "history_nb.ipynb") + return save_path + + + async def main(requirement: str = requirement, auto_run: bool = True, save_dir: str = save_dir): + """ + The main function to run the MLEngineer with optional history loading. + + Args: + requirement (str): The requirement for the MLEngineer. + auto_run (bool): Whether to auto-run the MLEngineer. + save_dir (str): The directory from which to load the history or to save the new history. + + Raises: + Exception: If an error occurs during execution, log the error and save the history. + """ + if save_dir: + logger.info("Resuming from history trajectory") + plan, nb = load_history(save_dir) + role = MLEngineer(goal=requirement, auto_run=auto_run) + role.plan = Plan(**plan) + role.execute_code = ExecutePyCode(nb) + import pdb;pdb.set_trace() + else: + logger.info("Run from scratch") + role = MLEngineer(goal=requirement, auto_run=auto_run) + + try: + await role.run(requirement) + except Exception as e: + + save_path = await save_history(role, save_dir) + + logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") fire.Fire(main) From 9d046a7ce570f65b01d9732b5651ea3584e23b4f Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 15:48:00 +0800 Subject: [PATCH 159/637] rm runtime result --- .../catboost_info/catboost_training.json | 1004 ----------------- .../catboost_info/learn/events.out.tfevents | Bin 54870 -> 0 bytes metagpt/roles/catboost_info/learn_error.tsv | 1001 ---------------- metagpt/roles/catboost_info/time_left.tsv | 1001 ---------------- 4 files changed, 3006 deletions(-) delete mode 100644 metagpt/roles/catboost_info/catboost_training.json delete mode 100644 metagpt/roles/catboost_info/learn/events.out.tfevents delete mode 100644 metagpt/roles/catboost_info/learn_error.tsv delete mode 100644 metagpt/roles/catboost_info/time_left.tsv diff --git a/metagpt/roles/catboost_info/catboost_training.json b/metagpt/roles/catboost_info/catboost_training.json deleted file mode 100644 index 68f95dbe6..000000000 --- a/metagpt/roles/catboost_info/catboost_training.json +++ /dev/null @@ -1,1004 +0,0 @@ -{ -"meta":{"test_sets":[],"test_metrics":[],"learn_metrics":[{"best_value":"Min","name":"Logloss"}],"launch_mode":"Train","parameters":"","iteration_count":1000,"learn_sets":["learn"],"name":"experiment"}, -"iterations":[ -{"learn":[0.6882514366],"iteration":0,"passed_time":0.2209602877,"remaining_time":220.7393274}, -{"learn":[0.6847159377],"iteration":1,"passed_time":0.2329893569,"remaining_time":116.2616891}, -{"learn":[0.6795580406],"iteration":2,"passed_time":0.25602594,"remaining_time":85.08595405}, -{"learn":[0.6750537284],"iteration":3,"passed_time":0.2782187523,"remaining_time":69.27646932}, -{"learn":[0.6699312491],"iteration":4,"passed_time":0.2999463484,"remaining_time":59.68932333}, -{"learn":[0.6650724516],"iteration":5,"passed_time":0.3218397014,"remaining_time":53.31811054}, -{"learn":[0.6606555729],"iteration":6,"passed_time":0.3459084069,"remaining_time":49.06957829}, -{"learn":[0.6561724634],"iteration":7,"passed_time":0.3679667854,"remaining_time":45.62788139}, -{"learn":[0.6519794747],"iteration":8,"passed_time":0.3899457437,"remaining_time":42.93735911}, -{"learn":[0.6492472012],"iteration":9,"passed_time":0.4007993018,"remaining_time":39.67913088}, -{"learn":[0.6446640682],"iteration":10,"passed_time":0.4235608008,"remaining_time":38.08196655}, -{"learn":[0.6400603726],"iteration":11,"passed_time":0.4457455009,"remaining_time":36.69971291}, -{"learn":[0.637483839],"iteration":12,"passed_time":0.4554558546,"remaining_time":34.57960988}, -{"learn":[0.6334773178],"iteration":13,"passed_time":0.4733180286,"remaining_time":33.33511259}, -{"learn":[0.6286841787],"iteration":14,"passed_time":0.4951142595,"remaining_time":32.51250304}, -{"learn":[0.6262362324],"iteration":15,"passed_time":0.506736055,"remaining_time":31.16426738}, -{"learn":[0.6227706725],"iteration":16,"passed_time":0.5269891476,"remaining_time":30.47237247}, -{"learn":[0.618564194],"iteration":17,"passed_time":0.5488836139,"remaining_time":29.94465049}, -{"learn":[0.6154841122],"iteration":18,"passed_time":0.5635392104,"remaining_time":29.09641923}, -{"learn":[0.6112592312],"iteration":19,"passed_time":0.5852540048,"remaining_time":28.67744624}, -{"learn":[0.6077881571],"iteration":20,"passed_time":0.6073336225,"remaining_time":28.31331507}, -{"learn":[0.6037553183],"iteration":21,"passed_time":0.6292302093,"remaining_time":27.97214294}, -{"learn":[0.6006649251],"iteration":22,"passed_time":0.6456397926,"remaining_time":27.42565554}, -{"learn":[0.5975849834],"iteration":23,"passed_time":0.6645987882,"remaining_time":27.02701739}, -{"learn":[0.5940831045],"iteration":24,"passed_time":0.6864422193,"remaining_time":26.77124655}, -{"learn":[0.5916771489],"iteration":25,"passed_time":0.7039365724,"remaining_time":26.37054698}, -{"learn":[0.5894338237],"iteration":26,"passed_time":0.7168747032,"remaining_time":25.83404023}, -{"learn":[0.5875190394],"iteration":27,"passed_time":0.7261948992,"remaining_time":25.20933721}, -{"learn":[0.5844895773],"iteration":28,"passed_time":0.7450785696,"remaining_time":24.9472859}, -{"learn":[0.5810267327],"iteration":29,"passed_time":0.7672816003,"remaining_time":24.80877174}, -{"learn":[0.5778936903],"iteration":30,"passed_time":0.7890965402,"remaining_time":24.66563056}, -{"learn":[0.576124503],"iteration":31,"passed_time":0.7996305702,"remaining_time":24.18882475}, -{"learn":[0.5735057785],"iteration":32,"passed_time":0.8142960813,"remaining_time":23.86134275}, -{"learn":[0.570293767],"iteration":33,"passed_time":0.8360974032,"remaining_time":23.75500269}, -{"learn":[0.5672457801],"iteration":34,"passed_time":0.8581203558,"remaining_time":23.6596041}, -{"learn":[0.5649423522],"iteration":35,"passed_time":0.8765849625,"remaining_time":23.47299733}, -{"learn":[0.5615275613],"iteration":36,"passed_time":0.9024578152,"remaining_time":23.48829395}, -{"learn":[0.5581402135],"iteration":37,"passed_time":0.9273196032,"remaining_time":23.47582785}, -{"learn":[0.555577741],"iteration":38,"passed_time":0.9458885498,"remaining_time":23.30766401}, -{"learn":[0.5523266666],"iteration":39,"passed_time":0.9682634562,"remaining_time":23.23832295}, -{"learn":[0.5508561568],"iteration":40,"passed_time":0.9762371787,"remaining_time":22.83442572}, -{"learn":[0.5487373589],"iteration":41,"passed_time":0.9944226088,"remaining_time":22.68230617}, -{"learn":[0.5460358061],"iteration":42,"passed_time":1.017613209,"remaining_time":22.64781026}, -{"learn":[0.5429618153],"iteration":43,"passed_time":1.039705016,"remaining_time":22.58995444}, -{"learn":[0.5411169242],"iteration":44,"passed_time":1.058342618,"remaining_time":22.46038223}, -{"learn":[0.5389150372],"iteration":45,"passed_time":1.080413363,"remaining_time":22.40683365}, -{"learn":[0.5364783846],"iteration":46,"passed_time":1.102740504,"remaining_time":22.3598234}, -{"learn":[0.534124059],"iteration":47,"passed_time":1.117903644,"remaining_time":22.17175561}, -{"learn":[0.5307002414],"iteration":48,"passed_time":1.140720347,"remaining_time":22.13928674}, -{"learn":[0.5289874066],"iteration":49,"passed_time":1.176629736,"remaining_time":22.35596498}, -{"learn":[0.5263977673],"iteration":50,"passed_time":1.206713737,"remaining_time":22.45433992}, -{"learn":[0.5243808354],"iteration":51,"passed_time":1.229893513,"remaining_time":22.42190481}, -{"learn":[0.5226468558],"iteration":52,"passed_time":1.251654021,"remaining_time":22.36445959}, -{"learn":[0.5205374561],"iteration":53,"passed_time":1.277702019,"remaining_time":22.38344648}, -{"learn":[0.5184600522],"iteration":54,"passed_time":1.300474793,"remaining_time":22.34452145}, -{"learn":[0.5161120982],"iteration":55,"passed_time":1.32256994,"remaining_time":22.29475042}, -{"learn":[0.5135670428],"iteration":56,"passed_time":1.345110102,"remaining_time":22.25331274}, -{"learn":[0.5117345666],"iteration":57,"passed_time":1.367616671,"remaining_time":22.21198111}, -{"learn":[0.5103042276],"iteration":58,"passed_time":1.379228529,"remaining_time":21.99752619}, -{"learn":[0.5088685224],"iteration":59,"passed_time":1.393836981,"remaining_time":21.83677937}, -{"learn":[0.5069392613],"iteration":60,"passed_time":1.420003792,"remaining_time":21.85874689}, -{"learn":[0.5058379484],"iteration":61,"passed_time":1.434821217,"remaining_time":21.70745648}, -{"learn":[0.504071849],"iteration":62,"passed_time":1.450886544,"remaining_time":21.57905859}, -{"learn":[0.5026509319],"iteration":63,"passed_time":1.470451497,"remaining_time":21.50535314}, -{"learn":[0.5013652681],"iteration":64,"passed_time":1.485685519,"remaining_time":21.37101477}, -{"learn":[0.4990494982],"iteration":65,"passed_time":1.507402942,"remaining_time":21.33203558}, -{"learn":[0.4975801239],"iteration":66,"passed_time":1.529043474,"remaining_time":21.29250091}, -{"learn":[0.4954073892],"iteration":67,"passed_time":1.55148063,"remaining_time":21.26441099}, -{"learn":[0.4937794274],"iteration":68,"passed_time":1.573374258,"remaining_time":21.22915121}, -{"learn":[0.4917253363],"iteration":69,"passed_time":1.595265697,"remaining_time":21.19424426}, -{"learn":[0.4897655824],"iteration":70,"passed_time":1.617828232,"remaining_time":21.16848489}, -{"learn":[0.4881025387],"iteration":71,"passed_time":1.63997141,"remaining_time":21.13740929}, -{"learn":[0.4855490154],"iteration":72,"passed_time":1.668328528,"remaining_time":21.18548693}, -{"learn":[0.4839406321],"iteration":73,"passed_time":1.694200638,"remaining_time":21.20040257}, -{"learn":[0.4825486397],"iteration":74,"passed_time":1.714501822,"remaining_time":21.14552248}, -{"learn":[0.4806147512],"iteration":75,"passed_time":1.74069548,"remaining_time":21.16319241}, -{"learn":[0.4790772126],"iteration":76,"passed_time":1.76355628,"remaining_time":21.13977203}, -{"learn":[0.4783791778],"iteration":77,"passed_time":1.771777779,"remaining_time":20.94332195}, -{"learn":[0.476922228],"iteration":78,"passed_time":1.789701952,"remaining_time":20.86475313}, -{"learn":[0.4757319459],"iteration":79,"passed_time":1.811825511,"remaining_time":20.83599338}, -{"learn":[0.4742068644],"iteration":80,"passed_time":1.833878171,"remaining_time":20.80659308}, -{"learn":[0.4726812191],"iteration":81,"passed_time":1.855740771,"remaining_time":20.77524424}, -{"learn":[0.4710468444],"iteration":82,"passed_time":1.877655908,"remaining_time":20.74470443}, -{"learn":[0.4693673381],"iteration":83,"passed_time":1.899915651,"remaining_time":20.71812782}, -{"learn":[0.4676818392],"iteration":84,"passed_time":1.929145481,"remaining_time":20.76668371}, -{"learn":[0.4657056761],"iteration":85,"passed_time":1.951233566,"remaining_time":20.73752883}, -{"learn":[0.4642140634],"iteration":86,"passed_time":1.969771174,"remaining_time":20.6712768}, -{"learn":[0.4632998746],"iteration":87,"passed_time":1.991742157,"remaining_time":20.64169144}, -{"learn":[0.4622314595],"iteration":88,"passed_time":2.014152311,"remaining_time":20.61677254}, -{"learn":[0.460592481],"iteration":89,"passed_time":2.036215085,"remaining_time":20.58839697}, -{"learn":[0.4593241639],"iteration":90,"passed_time":2.058798714,"remaining_time":20.56536298}, -{"learn":[0.4584569869],"iteration":91,"passed_time":2.073992457,"remaining_time":20.46940381}, -{"learn":[0.4577605438],"iteration":92,"passed_time":2.085384355,"remaining_time":20.33810334}, -{"learn":[0.4561556633],"iteration":93,"passed_time":2.107586742,"remaining_time":20.31354881}, -{"learn":[0.4554982999],"iteration":94,"passed_time":2.119182321,"remaining_time":20.188}, -{"learn":[0.4545744958],"iteration":95,"passed_time":2.137405105,"remaining_time":20.1272314}, -{"learn":[0.4540947284],"iteration":96,"passed_time":2.145521477,"remaining_time":19.97325664}, -{"learn":[0.4540085087],"iteration":97,"passed_time":2.150456003,"remaining_time":19.7929726}, -{"learn":[0.4531926189],"iteration":98,"passed_time":2.164042187,"remaining_time":19.6949698}, -{"learn":[0.4527317281],"iteration":99,"passed_time":2.176278456,"remaining_time":19.5865061}, -{"learn":[0.4519466344],"iteration":100,"passed_time":2.196543022,"remaining_time":19.55140769}, -{"learn":[0.450688372],"iteration":101,"passed_time":2.226552191,"remaining_time":19.60239085}, -{"learn":[0.4488322057],"iteration":102,"passed_time":2.2501016,"remaining_time":19.595545}, -{"learn":[0.4478782417],"iteration":103,"passed_time":2.272366787,"remaining_time":19.57731386}, -{"learn":[0.4468232543],"iteration":104,"passed_time":2.294253479,"remaining_time":19.55577965}, -{"learn":[0.4457592303],"iteration":105,"passed_time":2.316373853,"remaining_time":19.53620967}, -{"learn":[0.4447781675],"iteration":106,"passed_time":2.334811885,"remaining_time":19.48585994}, -{"learn":[0.4438522903],"iteration":107,"passed_time":2.36016004,"remaining_time":19.49317366}, -{"learn":[0.4430406598],"iteration":108,"passed_time":2.375441273,"remaining_time":19.41759793}, -{"learn":[0.4420804721],"iteration":109,"passed_time":2.397764964,"remaining_time":19.40009835}, -{"learn":[0.4413288303],"iteration":110,"passed_time":2.431630964,"remaining_time":19.47495429}, -{"learn":[0.4406769097],"iteration":111,"passed_time":2.448104093,"remaining_time":19.40996816}, -{"learn":[0.439927404],"iteration":112,"passed_time":2.471054306,"remaining_time":19.39668291}, -{"learn":[0.4387131738],"iteration":113,"passed_time":2.493169175,"remaining_time":19.37673587}, -{"learn":[0.4379161685],"iteration":114,"passed_time":2.515201611,"remaining_time":19.35611674}, -{"learn":[0.4368122477],"iteration":115,"passed_time":2.536876581,"remaining_time":19.33274912}, -{"learn":[0.4363066446],"iteration":116,"passed_time":2.558814198,"remaining_time":19.31139262}, -{"learn":[0.4352350106],"iteration":117,"passed_time":2.580710273,"remaining_time":19.28971577}, -{"learn":[0.4340016567],"iteration":118,"passed_time":2.602677388,"remaining_time":19.26856117}, -{"learn":[0.4332395033],"iteration":119,"passed_time":2.625437408,"remaining_time":19.25320766}, -{"learn":[0.4323511556],"iteration":120,"passed_time":2.650764059,"remaining_time":19.25637692}, -{"learn":[0.4316166781],"iteration":121,"passed_time":2.686256274,"remaining_time":19.33223778}, -{"learn":[0.4304510199],"iteration":122,"passed_time":2.712823213,"remaining_time":19.34265007}, -{"learn":[0.4297836948],"iteration":123,"passed_time":2.744805133,"remaining_time":19.39072014}, -{"learn":[0.4286476712],"iteration":124,"passed_time":2.819134566,"remaining_time":19.73394196}, -{"learn":[0.4281044323],"iteration":125,"passed_time":2.83375925,"remaining_time":19.65639353}, -{"learn":[0.4277452421],"iteration":126,"passed_time":2.851695114,"remaining_time":19.60259712}, -{"learn":[0.4268564678],"iteration":127,"passed_time":2.873147663,"remaining_time":19.57331846}, -{"learn":[0.4262834378],"iteration":128,"passed_time":2.894895865,"remaining_time":19.54615735}, -{"learn":[0.4255606014],"iteration":129,"passed_time":2.917059119,"remaining_time":19.52185718}, -{"learn":[0.4252233113],"iteration":130,"passed_time":2.937915636,"remaining_time":19.48892128}, -{"learn":[0.424750709],"iteration":131,"passed_time":2.956196683,"remaining_time":19.43923273}, -{"learn":[0.4237267015],"iteration":132,"passed_time":2.978141095,"remaining_time":19.41389721}, -{"learn":[0.4229110275],"iteration":133,"passed_time":3.000071725,"remaining_time":19.38852324}, -{"learn":[0.4226882753],"iteration":134,"passed_time":3.011578793,"remaining_time":19.29641227}, -{"learn":[0.4220238002],"iteration":135,"passed_time":3.029874756,"remaining_time":19.24861609}, -{"learn":[0.4216236205],"iteration":136,"passed_time":3.044691237,"remaining_time":19.17933239}, -{"learn":[0.4213023831],"iteration":137,"passed_time":3.059441462,"remaining_time":19.1104242}, -{"learn":[0.4205216151],"iteration":138,"passed_time":3.081243821,"remaining_time":19.08597792}, -{"learn":[0.4202240166],"iteration":139,"passed_time":3.099524481,"remaining_time":19.0399361}, -{"learn":[0.4192685081],"iteration":140,"passed_time":3.121315434,"remaining_time":19.01567346}, -{"learn":[0.4183050279],"iteration":141,"passed_time":3.143410659,"remaining_time":18.99328412}, -{"learn":[0.4173978005],"iteration":142,"passed_time":3.165354005,"remaining_time":18.96998868}, -{"learn":[0.4162954182],"iteration":143,"passed_time":3.195654275,"remaining_time":18.9963893}, -{"learn":[0.415992639],"iteration":144,"passed_time":3.220638215,"remaining_time":18.99065982}, -{"learn":[0.4157126501],"iteration":145,"passed_time":3.240805289,"remaining_time":18.95649121}, -{"learn":[0.4155852885],"iteration":146,"passed_time":3.249262286,"remaining_time":18.85456279}, -{"learn":[0.4148007449],"iteration":147,"passed_time":3.271747229,"remaining_time":18.83465297}, -{"learn":[0.4145476092],"iteration":148,"passed_time":3.289985958,"remaining_time":18.79045671}, -{"learn":[0.4139132529],"iteration":149,"passed_time":3.312792785,"remaining_time":18.77249245}, -{"learn":[0.4134843771],"iteration":150,"passed_time":3.32803032,"remaining_time":18.71190558}, -{"learn":[0.4124055164],"iteration":151,"passed_time":3.350009767,"remaining_time":18.68952817}, -{"learn":[0.412222917],"iteration":152,"passed_time":3.358421193,"remaining_time":18.59204412}, -{"learn":[0.4115787549],"iteration":153,"passed_time":3.380687937,"remaining_time":18.57183113}, -{"learn":[0.411081548],"iteration":154,"passed_time":3.402466212,"remaining_time":18.5489287}, -{"learn":[0.4110031407],"iteration":155,"passed_time":3.411424026,"remaining_time":18.45667871}, -{"learn":[0.4105687169],"iteration":156,"passed_time":3.439257058,"remaining_time":18.46683885}, -{"learn":[0.4095109133],"iteration":157,"passed_time":3.463613233,"remaining_time":18.45798951}, -{"learn":[0.4089787829],"iteration":158,"passed_time":3.485545103,"remaining_time":18.43612221}, -{"learn":[0.4084130488],"iteration":159,"passed_time":3.507325669,"remaining_time":18.41345976}, -{"learn":[0.4077252474],"iteration":160,"passed_time":3.529486341,"remaining_time":18.39278907}, -{"learn":[0.4072843557],"iteration":161,"passed_time":3.548156286,"remaining_time":18.35404301}, -{"learn":[0.4066351535],"iteration":162,"passed_time":3.569884188,"remaining_time":18.3312458}, -{"learn":[0.4063103387],"iteration":163,"passed_time":3.584669678,"remaining_time":18.27307226}, -{"learn":[0.4059185096],"iteration":164,"passed_time":3.599569148,"remaining_time":18.21600144}, -{"learn":[0.4052304697],"iteration":165,"passed_time":3.621153956,"remaining_time":18.1930265}, -{"learn":[0.4050859305],"iteration":166,"passed_time":3.629460067,"remaining_time":18.10383375}, -{"learn":[0.4044742571],"iteration":167,"passed_time":3.65097424,"remaining_time":18.08101529}, -{"learn":[0.4041975026],"iteration":168,"passed_time":3.679528258,"remaining_time":18.0928283}, -{"learn":[0.4035590426],"iteration":169,"passed_time":3.703328795,"remaining_time":18.08095824}, -{"learn":[0.4030297385],"iteration":170,"passed_time":3.728541368,"remaining_time":18.07579412}, -{"learn":[0.4025456368],"iteration":171,"passed_time":3.762066103,"remaining_time":18.11041124}, -{"learn":[0.402334931],"iteration":172,"passed_time":3.773807336,"remaining_time":18.0401079}, -{"learn":[0.401831618],"iteration":173,"passed_time":3.795813183,"remaining_time":18.01920511}, -{"learn":[0.4017743873],"iteration":174,"passed_time":3.807120561,"remaining_time":17.94785407}, -{"learn":[0.4011280658],"iteration":175,"passed_time":3.829251163,"remaining_time":17.92785772}, -{"learn":[0.4005475562],"iteration":176,"passed_time":3.851096112,"remaining_time":17.90650904}, -{"learn":[0.4000603326],"iteration":177,"passed_time":3.883514366,"remaining_time":17.93398207}, -{"learn":[0.3990424765],"iteration":178,"passed_time":3.954211249,"remaining_time":18.13635439}, -{"learn":[0.3987132685],"iteration":179,"passed_time":4.006291434,"remaining_time":18.2508832}, -{"learn":[0.3980735307],"iteration":180,"passed_time":4.04072907,"remaining_time":18.28374093}, -{"learn":[0.3976839336],"iteration":181,"passed_time":4.066100873,"remaining_time":18.27511272}, -{"learn":[0.3970583957],"iteration":182,"passed_time":4.088135325,"remaining_time":18.25140197}, -{"learn":[0.3968093954],"iteration":183,"passed_time":4.109985916,"remaining_time":18.22689406}, -{"learn":[0.3965549511],"iteration":184,"passed_time":4.121652408,"remaining_time":18.1575498}, -{"learn":[0.3961498711],"iteration":185,"passed_time":4.13684245,"remaining_time":18.10424599}, -{"learn":[0.3958574586],"iteration":186,"passed_time":4.158257951,"remaining_time":18.07841558}, -{"learn":[0.3958036693],"iteration":187,"passed_time":4.166355942,"remaining_time":17.99511183}, -{"learn":[0.3954652307],"iteration":188,"passed_time":4.184386925,"remaining_time":17.95522644}, -{"learn":[0.3949970516],"iteration":189,"passed_time":4.215870756,"remaining_time":17.9729227}, -{"learn":[0.3946423615],"iteration":190,"passed_time":4.243567046,"remaining_time":17.97406147}, -{"learn":[0.3937733218],"iteration":191,"passed_time":4.267948498,"remaining_time":17.96094993}, -{"learn":[0.3931025335],"iteration":192,"passed_time":4.290856304,"remaining_time":17.94155978}, -{"learn":[0.3925366206],"iteration":193,"passed_time":4.31302246,"remaining_time":17.91905208}, -{"learn":[0.391865258],"iteration":194,"passed_time":4.33481009,"remaining_time":17.89498524}, -{"learn":[0.3914533348],"iteration":195,"passed_time":4.356400319,"remaining_time":17.87013192}, -{"learn":[0.3912476553],"iteration":196,"passed_time":4.37834033,"remaining_time":17.84673749}, -{"learn":[0.3905329154],"iteration":197,"passed_time":4.400898981,"remaining_time":17.82586355}, -{"learn":[0.3898798389],"iteration":198,"passed_time":4.423174721,"remaining_time":17.80383393}, -{"learn":[0.3891654603],"iteration":199,"passed_time":4.455398968,"remaining_time":17.82159587}, -{"learn":[0.3887747769],"iteration":200,"passed_time":4.477551848,"remaining_time":17.7988255}, -{"learn":[0.3884753081],"iteration":201,"passed_time":4.499497174,"remaining_time":17.77524131}, -{"learn":[0.3877793274],"iteration":202,"passed_time":4.521143304,"remaining_time":17.75049859}, -{"learn":[0.3874664375],"iteration":203,"passed_time":4.543454079,"remaining_time":17.72837964}, -{"learn":[0.3871008215],"iteration":204,"passed_time":4.565293203,"remaining_time":17.70442974}, -{"learn":[0.3865209415],"iteration":205,"passed_time":4.587377641,"remaining_time":17.68144586}, -{"learn":[0.3859273739],"iteration":206,"passed_time":4.609581413,"remaining_time":17.65892783}, -{"learn":[0.3855620818],"iteration":207,"passed_time":4.631576096,"remaining_time":17.63561667}, -{"learn":[0.3851648115],"iteration":208,"passed_time":4.653493816,"remaining_time":17.61202684}, -{"learn":[0.3843406643],"iteration":209,"passed_time":4.675584403,"remaining_time":17.58910323}, -{"learn":[0.3842506785],"iteration":210,"passed_time":4.702482855,"remaining_time":17.58416575}, -{"learn":[0.3840729725],"iteration":211,"passed_time":4.725032901,"remaining_time":17.56285814}, -{"learn":[0.3836865443],"iteration":212,"passed_time":4.759392503,"remaining_time":17.58517324}, -{"learn":[0.3834504777],"iteration":213,"passed_time":4.787775158,"remaining_time":17.58500595}, -{"learn":[0.383221126],"iteration":214,"passed_time":4.810450824,"remaining_time":17.56373906}, -{"learn":[0.382875332],"iteration":215,"passed_time":4.833018275,"remaining_time":17.54206633}, -{"learn":[0.3827132502],"iteration":216,"passed_time":4.854566753,"remaining_time":17.51670861}, -{"learn":[0.3825534206],"iteration":217,"passed_time":4.872745824,"remaining_time":17.47929924}, -{"learn":[0.3821802218],"iteration":218,"passed_time":4.894785533,"remaining_time":17.45583334}, -{"learn":[0.3818052054],"iteration":219,"passed_time":4.916556405,"remaining_time":17.43142725}, -{"learn":[0.3816117561],"iteration":220,"passed_time":4.938252004,"remaining_time":17.40677969}, -{"learn":[0.3812172842],"iteration":221,"passed_time":4.960909917,"remaining_time":17.38553115}, -{"learn":[0.3810255507],"iteration":222,"passed_time":4.988823751,"remaining_time":17.3825832}, -{"learn":[0.3809283758],"iteration":223,"passed_time":5.010867096,"remaining_time":17.3590753}, -{"learn":[0.3808446585],"iteration":224,"passed_time":5.03372608,"remaining_time":17.33838983}, -{"learn":[0.3801708307],"iteration":225,"passed_time":5.055982895,"remaining_time":17.31562284}, -{"learn":[0.3798560036],"iteration":226,"passed_time":5.078222381,"remaining_time":17.29280132}, -{"learn":[0.3794923916],"iteration":227,"passed_time":5.100240524,"remaining_time":17.26923546}, -{"learn":[0.3793373475],"iteration":228,"passed_time":5.121943623,"remaining_time":17.24462242}, -{"learn":[0.3791100641],"iteration":229,"passed_time":5.144086578,"remaining_time":17.22150724}, -{"learn":[0.3786186348],"iteration":230,"passed_time":5.16685425,"remaining_time":17.20048017}, -{"learn":[0.3783799747],"iteration":231,"passed_time":5.20100571,"remaining_time":17.21712235}, -{"learn":[0.3779793355],"iteration":232,"passed_time":5.262317101,"remaining_time":17.32273483}, -{"learn":[0.3777902426],"iteration":233,"passed_time":5.311745138,"remaining_time":17.38802041}, -{"learn":[0.377759024],"iteration":234,"passed_time":5.326918408,"remaining_time":17.3408195}, -{"learn":[0.3775407801],"iteration":235,"passed_time":5.366940659,"remaining_time":17.37433332}, -{"learn":[0.3772180796],"iteration":236,"passed_time":5.410539935,"remaining_time":17.41874249}, -{"learn":[0.3770122934],"iteration":237,"passed_time":5.434735311,"remaining_time":17.40028701}, -{"learn":[0.3765490897],"iteration":238,"passed_time":5.464453118,"remaining_time":17.39936746}, -{"learn":[0.3764835925],"iteration":239,"passed_time":5.490098561,"remaining_time":17.38531211}, -{"learn":[0.3759776808],"iteration":240,"passed_time":5.513007919,"remaining_time":17.36254361}, -{"learn":[0.3757142158],"iteration":241,"passed_time":5.534974237,"remaining_time":17.33682013}, -{"learn":[0.3752871326],"iteration":242,"passed_time":5.556902667,"remaining_time":17.31100954}, -{"learn":[0.3748925461],"iteration":243,"passed_time":5.578824076,"remaining_time":17.28520902}, -{"learn":[0.3746214472],"iteration":244,"passed_time":5.600775557,"remaining_time":17.25953284}, -{"learn":[0.3740866821],"iteration":245,"passed_time":5.622820585,"remaining_time":17.23417366}, -{"learn":[0.3735711811],"iteration":246,"passed_time":5.645061274,"remaining_time":17.20943781}, -{"learn":[0.3732294839],"iteration":247,"passed_time":5.666748819,"remaining_time":17.1830448}, -{"learn":[0.3729683346],"iteration":248,"passed_time":5.678427534,"remaining_time":17.12650232}, -{"learn":[0.3727413457],"iteration":249,"passed_time":5.709521815,"remaining_time":17.12856544}, -{"learn":[0.3723892856],"iteration":250,"passed_time":5.732680933,"remaining_time":17.10668534}, -{"learn":[0.3719863363],"iteration":251,"passed_time":5.758476679,"remaining_time":17.09262125}, -{"learn":[0.3716739793],"iteration":252,"passed_time":5.793914284,"remaining_time":17.10693269}, -{"learn":[0.3716103601],"iteration":253,"passed_time":5.808977588,"remaining_time":17.06101292}, -{"learn":[0.3715319129],"iteration":254,"passed_time":5.827444128,"remaining_time":17.02527794}, -{"learn":[0.371508316],"iteration":255,"passed_time":5.835459561,"remaining_time":16.95930435}, -{"learn":[0.3713832526],"iteration":256,"passed_time":5.847111057,"remaining_time":16.90429383}, -{"learn":[0.3710349299],"iteration":257,"passed_time":5.86851783,"remaining_time":16.87767531}, -{"learn":[0.371030348],"iteration":258,"passed_time":5.873415663,"remaining_time":16.80386489}, -{"learn":[0.3708892154],"iteration":259,"passed_time":5.894554845,"remaining_time":16.77680994}, -{"learn":[0.3705921581],"iteration":260,"passed_time":5.916265404,"remaining_time":16.75141814}, -{"learn":[0.3702244894],"iteration":261,"passed_time":5.938204167,"remaining_time":16.72669723}, -{"learn":[0.3702202347],"iteration":262,"passed_time":5.943096693,"remaining_time":16.65422914}, -{"learn":[0.3698349185],"iteration":263,"passed_time":5.965210221,"remaining_time":16.63028304}, -{"learn":[0.3696711654],"iteration":264,"passed_time":5.983506973,"remaining_time":16.59576462}, -{"learn":[0.3692863237],"iteration":265,"passed_time":6.008756231,"remaining_time":16.58055291}, -{"learn":[0.3690671729],"iteration":266,"passed_time":6.030910934,"remaining_time":16.55677047}, -{"learn":[0.3687581564],"iteration":267,"passed_time":6.053029386,"remaining_time":16.53290116}, -{"learn":[0.3683941859],"iteration":268,"passed_time":6.074786966,"remaining_time":16.50806421}, -{"learn":[0.368050992],"iteration":269,"passed_time":6.096955104,"remaining_time":16.4843601}, -{"learn":[0.3677324616],"iteration":270,"passed_time":6.118732429,"remaining_time":16.45961602}, -{"learn":[0.3674997199],"iteration":271,"passed_time":6.133897356,"remaining_time":16.41719586}, -{"learn":[0.3672484703],"iteration":272,"passed_time":6.155501408,"remaining_time":16.3921228}, -{"learn":[0.3670583272],"iteration":273,"passed_time":6.177355776,"remaining_time":16.3677383}, -{"learn":[0.366936125],"iteration":274,"passed_time":6.196658037,"remaining_time":16.33664392}, -{"learn":[0.3665623609],"iteration":275,"passed_time":6.223489438,"remaining_time":16.32538534}, -{"learn":[0.3663926512],"iteration":276,"passed_time":6.249200888,"remaining_time":16.31109113}, -{"learn":[0.3659325492],"iteration":277,"passed_time":6.275677092,"remaining_time":16.29870094}, -{"learn":[0.3656171802],"iteration":278,"passed_time":6.300903365,"remaining_time":16.28297966}, -{"learn":[0.3654984176],"iteration":279,"passed_time":6.322813287,"remaining_time":16.25866274}, -{"learn":[0.3653075674],"iteration":280,"passed_time":6.344882826,"remaining_time":16.23477136}, -{"learn":[0.3652175898],"iteration":281,"passed_time":6.366618787,"remaining_time":16.21004358}, -{"learn":[0.365164193],"iteration":282,"passed_time":6.374755754,"remaining_time":16.15088295}, -{"learn":[0.3648739127],"iteration":283,"passed_time":6.397192914,"remaining_time":16.12813425}, -{"learn":[0.3648461699],"iteration":284,"passed_time":6.408952603,"remaining_time":16.07860039}, -{"learn":[0.3646301237],"iteration":285,"passed_time":6.430803359,"remaining_time":16.05452307}, -{"learn":[0.3646030346],"iteration":286,"passed_time":6.444636163,"remaining_time":16.01054211}, -{"learn":[0.3644969437],"iteration":287,"passed_time":6.469654901,"remaining_time":15.99442462}, -{"learn":[0.3643646754],"iteration":288,"passed_time":6.495103933,"remaining_time":15.97930414}, -{"learn":[0.3639816129],"iteration":289,"passed_time":6.516903674,"remaining_time":15.95517796}, -{"learn":[0.3636656294],"iteration":290,"passed_time":6.538871352,"remaining_time":15.93147694}, -{"learn":[0.363430258],"iteration":291,"passed_time":6.575359293,"remaining_time":15.94299445}, -{"learn":[0.3632201443],"iteration":292,"passed_time":6.615051101,"remaining_time":15.96191511}, -{"learn":[0.3628359855],"iteration":293,"passed_time":6.65904192,"remaining_time":15.99076053}, -{"learn":[0.3627236518],"iteration":294,"passed_time":6.696672672,"remaining_time":16.00391266}, -{"learn":[0.3622184354],"iteration":295,"passed_time":6.758116653,"remaining_time":16.07335853}, -{"learn":[0.3621264294],"iteration":296,"passed_time":6.787336902,"remaining_time":16.0656493}, -{"learn":[0.3620089934],"iteration":297,"passed_time":6.824504579,"remaining_time":16.0765175}, -{"learn":[0.3618790034],"iteration":298,"passed_time":6.853537324,"remaining_time":16.06799219}, -{"learn":[0.3617399454],"iteration":299,"passed_time":6.877452265,"remaining_time":16.04738862}, -{"learn":[0.3616850281],"iteration":300,"passed_time":6.889464347,"remaining_time":15.99912152}, -{"learn":[0.3613225424],"iteration":301,"passed_time":6.911690035,"remaining_time":15.97470081}, -{"learn":[0.3613077955],"iteration":302,"passed_time":6.919765154,"remaining_time":15.9177436}, -{"learn":[0.3611613922],"iteration":303,"passed_time":6.941101626,"remaining_time":15.89146951}, -{"learn":[0.3606681647],"iteration":304,"passed_time":6.963247586,"remaining_time":15.86707237}, -{"learn":[0.3606662726],"iteration":305,"passed_time":6.971583113,"remaining_time":15.81136824}, -{"learn":[0.3606644378],"iteration":306,"passed_time":6.977915177,"remaining_time":15.75145022}, -{"learn":[0.3601634319],"iteration":307,"passed_time":7.004917362,"remaining_time":15.73832083}, -{"learn":[0.3599344409],"iteration":308,"passed_time":7.027043523,"remaining_time":15.71419765}, -{"learn":[0.3596092344],"iteration":309,"passed_time":7.049369762,"remaining_time":15.6905327}, -{"learn":[0.3595229983],"iteration":310,"passed_time":7.071081586,"remaining_time":15.66551515}, -{"learn":[0.3591048131],"iteration":311,"passed_time":7.093007056,"remaining_time":15.64098992}, -{"learn":[0.3589464226],"iteration":312,"passed_time":7.114835418,"remaining_time":15.61626815}, -{"learn":[0.3588198429],"iteration":313,"passed_time":7.136272187,"remaining_time":15.5907093}, -{"learn":[0.3585066029],"iteration":314,"passed_time":7.1580098,"remaining_time":15.56583083}, -{"learn":[0.3584892118],"iteration":315,"passed_time":7.172922678,"remaining_time":15.52619972}, -{"learn":[0.3580264626],"iteration":316,"passed_time":7.195573733,"remaining_time":15.50339703}, -{"learn":[0.3580001595],"iteration":317,"passed_time":7.207669718,"remaining_time":15.45795833}, -{"learn":[0.3578136454],"iteration":318,"passed_time":7.228151813,"remaining_time":15.4306313}, -{"learn":[0.3576122347],"iteration":319,"passed_time":7.255577239,"remaining_time":15.41810163}, -{"learn":[0.3573283521],"iteration":320,"passed_time":7.28056961,"remaining_time":15.4003326}, -{"learn":[0.3569801921],"iteration":321,"passed_time":7.307082032,"remaining_time":15.38571931}, -{"learn":[0.3568810457],"iteration":322,"passed_time":7.329394107,"remaining_time":15.36222851}, -{"learn":[0.35676928],"iteration":323,"passed_time":7.355222467,"remaining_time":15.34608144}, -{"learn":[0.3566134286],"iteration":324,"passed_time":7.377839391,"remaining_time":15.32320489}, -{"learn":[0.3565307492],"iteration":325,"passed_time":7.393089132,"remaining_time":15.28509839}, -{"learn":[0.3562459001],"iteration":326,"passed_time":7.415445478,"remaining_time":15.26175782}, -{"learn":[0.3559648846],"iteration":327,"passed_time":7.43874903,"remaining_time":15.24036387}, -{"learn":[0.3558310934],"iteration":328,"passed_time":7.461531161,"remaining_time":15.21789486}, -{"learn":[0.355525639],"iteration":329,"passed_time":7.488692372,"remaining_time":15.20431482}, -{"learn":[0.3553348141],"iteration":330,"passed_time":7.512358605,"remaining_time":15.18358884}, -{"learn":[0.3552290203],"iteration":331,"passed_time":7.534131988,"remaining_time":15.15903665}, -{"learn":[0.3545412457],"iteration":332,"passed_time":7.556277348,"remaining_time":15.13524622}, -{"learn":[0.3543498177],"iteration":333,"passed_time":7.578229608,"remaining_time":15.1110806}, -{"learn":[0.354267269],"iteration":334,"passed_time":7.600212936,"remaining_time":15.08698986}, -{"learn":[0.3541845198],"iteration":335,"passed_time":7.62304827,"remaining_time":15.06459539}, -{"learn":[0.3538121961],"iteration":336,"passed_time":7.644785675,"remaining_time":15.04003829}, -{"learn":[0.3535608169],"iteration":337,"passed_time":7.667040919,"remaining_time":15.0165121}, -{"learn":[0.3531983941],"iteration":338,"passed_time":7.689794841,"remaining_time":14.99396575}, -{"learn":[0.3530182856],"iteration":339,"passed_time":7.706120291,"remaining_time":14.95893939}, -{"learn":[0.3527164122],"iteration":340,"passed_time":7.737325144,"remaining_time":14.95277792}, -{"learn":[0.352633574],"iteration":341,"passed_time":7.779216311,"remaining_time":14.96703021}, -{"learn":[0.3525075494],"iteration":342,"passed_time":7.905813197,"remaining_time":15.14320487}, -{"learn":[0.3524961056],"iteration":343,"passed_time":7.927378105,"remaining_time":15.11732569}, -{"learn":[0.35245364],"iteration":344,"passed_time":7.959635347,"remaining_time":15.11177146}, -{"learn":[0.3521993518],"iteration":345,"passed_time":7.982424347,"remaining_time":15.08816625}, -{"learn":[0.3520077509],"iteration":346,"passed_time":8.011559383,"remaining_time":15.076508}, -{"learn":[0.3519447224],"iteration":347,"passed_time":8.041947635,"remaining_time":15.06709729}, -{"learn":[0.3518945368],"iteration":348,"passed_time":8.053708214,"remaining_time":15.02281962}, -{"learn":[0.3516107656],"iteration":349,"passed_time":8.075229556,"remaining_time":14.99685489}, -{"learn":[0.3515159018],"iteration":350,"passed_time":8.09696984,"remaining_time":14.9713203}, -{"learn":[0.3515000807],"iteration":351,"passed_time":8.105153638,"remaining_time":14.92085101}, -{"learn":[0.3512670327],"iteration":352,"passed_time":8.127032488,"remaining_time":14.89572244}, -{"learn":[0.3508513767],"iteration":353,"passed_time":8.148752566,"remaining_time":14.87032248}, -{"learn":[0.3504572704],"iteration":354,"passed_time":8.174282947,"remaining_time":14.8518662}, -{"learn":[0.35017651],"iteration":355,"passed_time":8.214266086,"remaining_time":14.85951505}, -{"learn":[0.3501286962],"iteration":356,"passed_time":8.260065356,"remaining_time":14.87737262}, -{"learn":[0.3498837488],"iteration":357,"passed_time":8.323683533,"remaining_time":14.92682913}, -{"learn":[0.3495626772],"iteration":358,"passed_time":8.36389557,"remaining_time":14.93386368}, -{"learn":[0.3493757992],"iteration":359,"passed_time":8.399382759,"remaining_time":14.93223602}, -{"learn":[0.3490827511],"iteration":360,"passed_time":8.424002237,"remaining_time":14.91118401}, -{"learn":[0.3488646977],"iteration":361,"passed_time":8.448150513,"remaining_time":14.88928184}, -{"learn":[0.3484613426],"iteration":362,"passed_time":8.471364531,"remaining_time":14.86572784}, -{"learn":[0.3483259359],"iteration":363,"passed_time":8.496857622,"remaining_time":14.84615782}, -{"learn":[0.3480209179],"iteration":364,"passed_time":8.522528902,"remaining_time":14.82686535}, -{"learn":[0.3476817077],"iteration":365,"passed_time":8.54443234,"remaining_time":14.80101121}, -{"learn":[0.3474725024],"iteration":366,"passed_time":8.566187938,"remaining_time":14.77492361}, -{"learn":[0.3474285581],"iteration":367,"passed_time":8.577474249,"remaining_time":14.73087969}, -{"learn":[0.3470858003],"iteration":368,"passed_time":8.599341907,"remaining_time":14.70510771}, -{"learn":[0.3469015363],"iteration":369,"passed_time":8.621413585,"remaining_time":14.67970421}, -{"learn":[0.3463432335],"iteration":370,"passed_time":8.643379322,"remaining_time":14.65413907}, -{"learn":[0.3461185251],"iteration":371,"passed_time":8.665799542,"remaining_time":14.62936052}, -{"learn":[0.3458407874],"iteration":372,"passed_time":8.701760975,"remaining_time":14.62735692}, -{"learn":[0.3457031815],"iteration":373,"passed_time":8.756442967,"remaining_time":14.65650614}, -{"learn":[0.3455128742],"iteration":374,"passed_time":8.807534181,"remaining_time":14.67922364}, -{"learn":[0.3452742962],"iteration":375,"passed_time":8.872117774,"remaining_time":14.72394014}, -{"learn":[0.3452251263],"iteration":376,"passed_time":8.899250902,"remaining_time":14.70618916}, -{"learn":[0.3450881669],"iteration":377,"passed_time":8.928085048,"remaining_time":14.69118757}, -{"learn":[0.3449793371],"iteration":378,"passed_time":8.947728076,"remaining_time":14.66105313}, -{"learn":[0.344691491],"iteration":379,"passed_time":8.969934723,"remaining_time":14.63515665}, -{"learn":[0.3443639578],"iteration":380,"passed_time":8.99241215,"remaining_time":14.60971948}, -{"learn":[0.3442136782],"iteration":381,"passed_time":9.02144383,"remaining_time":14.59490127}, -{"learn":[0.3440594348],"iteration":382,"passed_time":9.03745504,"remaining_time":14.55903332}, -{"learn":[0.3439143606],"iteration":383,"passed_time":9.059658661,"remaining_time":14.53320244}, -{"learn":[0.3437593269],"iteration":384,"passed_time":9.081463941,"remaining_time":14.50675409}, -{"learn":[0.3433892597],"iteration":385,"passed_time":9.103490585,"remaining_time":14.48068192}, -{"learn":[0.3430005665],"iteration":386,"passed_time":9.125031046,"remaining_time":14.45386055}, -{"learn":[0.3426363435],"iteration":387,"passed_time":9.147229098,"remaining_time":14.42810363}, -{"learn":[0.3425431519],"iteration":388,"passed_time":9.169451319,"remaining_time":14.40240297}, -{"learn":[0.3422194687],"iteration":389,"passed_time":9.192135544,"remaining_time":14.37744277}, -{"learn":[0.3420983186],"iteration":390,"passed_time":9.21482497,"remaining_time":14.35250232}, -{"learn":[0.3419547464],"iteration":391,"passed_time":9.240174122,"remaining_time":14.33169864}, -{"learn":[0.3418564406],"iteration":392,"passed_time":9.266114196,"remaining_time":14.31178452}, -{"learn":[0.3416273848],"iteration":393,"passed_time":9.291484275,"remaining_time":14.29096312}, -{"learn":[0.3413111854],"iteration":394,"passed_time":9.321332214,"remaining_time":14.27697719}, -{"learn":[0.3411554684],"iteration":395,"passed_time":9.340067818,"remaining_time":14.24596203}, -{"learn":[0.3410876043],"iteration":396,"passed_time":9.370184518,"remaining_time":14.23229538}, -{"learn":[0.3410429673],"iteration":397,"passed_time":9.39162494,"remaining_time":14.20542265}, -{"learn":[0.341042403],"iteration":398,"passed_time":9.406694781,"remaining_time":14.16898136}, -{"learn":[0.3409573532],"iteration":399,"passed_time":9.451310091,"remaining_time":14.17696514}, -{"learn":[0.3408655821],"iteration":400,"passed_time":9.4923409,"remaining_time":14.17933217}, -{"learn":[0.3406567823],"iteration":401,"passed_time":9.547103264,"remaining_time":14.20190983}, -{"learn":[0.3404763525],"iteration":402,"passed_time":9.579503053,"remaining_time":14.19097599}, -{"learn":[0.3403717595],"iteration":403,"passed_time":9.60199694,"remaining_time":14.16532222}, -{"learn":[0.3400880712],"iteration":404,"passed_time":9.62408225,"remaining_time":14.1390838}, -{"learn":[0.3399640397],"iteration":405,"passed_time":9.646009648,"remaining_time":14.1126348}, -{"learn":[0.3399010234],"iteration":406,"passed_time":9.667975784,"remaining_time":14.08626447}, -{"learn":[0.339826297],"iteration":407,"passed_time":9.690275059,"remaining_time":14.06039911}, -{"learn":[0.3397102041],"iteration":408,"passed_time":9.709651563,"remaining_time":14.03032781}, -{"learn":[0.3395576906],"iteration":409,"passed_time":9.732244916,"remaining_time":14.00493781}, -{"learn":[0.3392198413],"iteration":410,"passed_time":9.757549483,"remaining_time":13.98344683}, -{"learn":[0.3391475536],"iteration":411,"passed_time":9.785746366,"remaining_time":13.9660652}, -{"learn":[0.3389268153],"iteration":412,"passed_time":9.814648628,"remaining_time":13.94963376}, -{"learn":[0.3389266721],"iteration":413,"passed_time":9.823336433,"remaining_time":13.90452935}, -{"learn":[0.3388830216],"iteration":414,"passed_time":9.838841117,"remaining_time":13.86920977}, -{"learn":[0.3388260339],"iteration":415,"passed_time":9.861509022,"remaining_time":13.84404151}, -{"learn":[0.3385376482],"iteration":416,"passed_time":9.884074666,"remaining_time":13.81874228}, -{"learn":[0.3384459997],"iteration":417,"passed_time":9.906639396,"remaining_time":13.79345485}, -{"learn":[0.3384035804],"iteration":418,"passed_time":9.928521171,"remaining_time":13.76723341}, -{"learn":[0.338206978],"iteration":419,"passed_time":9.950592651,"remaining_time":13.74129461}, -{"learn":[0.3381563107],"iteration":420,"passed_time":9.972312358,"remaining_time":13.71489039}, -{"learn":[0.3380916237],"iteration":421,"passed_time":9.994182293,"remaining_time":13.68871414}, -{"learn":[0.3379155117],"iteration":422,"passed_time":10.01716742,"remaining_time":13.66407943}, -{"learn":[0.3377370109],"iteration":423,"passed_time":10.0451664,"remaining_time":13.64626379}, -{"learn":[0.3376685347],"iteration":424,"passed_time":10.06750274,"remaining_time":13.620739}, -{"learn":[0.3375387678],"iteration":425,"passed_time":10.08578742,"remaining_time":13.58976991}, -{"learn":[0.3375264339],"iteration":426,"passed_time":10.10044959,"remaining_time":13.5539991}, -{"learn":[0.3375078804],"iteration":427,"passed_time":10.10840815,"remaining_time":13.5093679}, -{"learn":[0.337340127],"iteration":428,"passed_time":10.13027685,"remaining_time":13.4834221}, -{"learn":[0.3372056844],"iteration":429,"passed_time":10.15208,"remaining_time":13.45740837}, -{"learn":[0.336977263],"iteration":430,"passed_time":10.18504522,"remaining_time":13.44615019}, -{"learn":[0.3367988208],"iteration":431,"passed_time":10.2187855,"remaining_time":13.43581056}, -{"learn":[0.3367285927],"iteration":432,"passed_time":10.23098873,"remaining_time":13.39716076}, -{"learn":[0.3366052887],"iteration":433,"passed_time":10.2766679,"remaining_time":13.40229039}, -{"learn":[0.3363612465],"iteration":434,"passed_time":10.32772274,"remaining_time":13.41416862}, -{"learn":[0.3362435567],"iteration":435,"passed_time":10.37212705,"remaining_time":13.41715518}, -{"learn":[0.3361462199],"iteration":436,"passed_time":10.40813808,"remaining_time":13.40911154}, -{"learn":[0.3361461347],"iteration":437,"passed_time":10.41298218,"remaining_time":13.36094973}, -{"learn":[0.3360079098],"iteration":438,"passed_time":10.43960066,"remaining_time":13.34081086}, -{"learn":[0.3358086626],"iteration":439,"passed_time":10.46151863,"remaining_time":13.31466007}, -{"learn":[0.3356011997],"iteration":440,"passed_time":10.48387624,"remaining_time":13.28908576}, -{"learn":[0.3354146851],"iteration":441,"passed_time":10.50689004,"remaining_time":13.26435439}, -{"learn":[0.3352191524],"iteration":442,"passed_time":10.53608105,"remaining_time":13.24739762}, -{"learn":[0.3348674793],"iteration":443,"passed_time":10.55831293,"remaining_time":13.22167115}, -{"learn":[0.3348375647],"iteration":444,"passed_time":10.56995489,"remaining_time":13.18275273}, -{"learn":[0.3346763511],"iteration":445,"passed_time":10.59183961,"remaining_time":13.15667969}, -{"learn":[0.3344469846],"iteration":446,"passed_time":10.61412676,"remaining_time":13.13112326}, -{"learn":[0.3341331513],"iteration":447,"passed_time":10.63614375,"remaining_time":13.10524854}, -{"learn":[0.3338919982],"iteration":448,"passed_time":10.65810123,"remaining_time":13.07931798}, -{"learn":[0.3336666898],"iteration":449,"passed_time":10.67988947,"remaining_time":13.05319824}, -{"learn":[0.3336666289],"iteration":450,"passed_time":10.68502464,"remaining_time":13.00682601}, -{"learn":[0.3336154028],"iteration":451,"passed_time":10.70821328,"remaining_time":12.98252407}, -{"learn":[0.3333581167],"iteration":452,"passed_time":10.73100265,"remaining_time":12.95774492}, -{"learn":[0.333290786],"iteration":453,"passed_time":10.74255886,"remaining_time":12.91946506}, -{"learn":[0.3332008443],"iteration":454,"passed_time":10.76369789,"remaining_time":12.89278099}, -{"learn":[0.3331038516],"iteration":455,"passed_time":10.79258158,"remaining_time":12.87536048}, -{"learn":[0.3329428533],"iteration":456,"passed_time":10.81736692,"remaining_time":12.85302021}, -{"learn":[0.3329254389],"iteration":457,"passed_time":10.83350231,"remaining_time":12.82043286}, -{"learn":[0.3326182973],"iteration":458,"passed_time":10.85619419,"remaining_time":12.795645}, -{"learn":[0.3321795229],"iteration":459,"passed_time":10.87927788,"remaining_time":12.77132621}, -{"learn":[0.3320965842],"iteration":460,"passed_time":10.90369744,"remaining_time":12.74857466}, -{"learn":[0.3320919904],"iteration":461,"passed_time":10.9128379,"remaining_time":12.70802336}, -{"learn":[0.3317616467],"iteration":462,"passed_time":10.94039362,"remaining_time":12.68896625}, -{"learn":[0.3314515976],"iteration":463,"passed_time":10.96485849,"remaining_time":12.66630204}, -{"learn":[0.3314302988],"iteration":464,"passed_time":10.97934712,"remaining_time":12.63215206}, -{"learn":[0.331326468],"iteration":465,"passed_time":11.02336518,"remaining_time":12.63192491}, -{"learn":[0.3312172495],"iteration":466,"passed_time":11.06920655,"remaining_time":12.6335912}, -{"learn":[0.3311362468],"iteration":467,"passed_time":11.11033467,"remaining_time":12.62969667}, -{"learn":[0.3308259578],"iteration":468,"passed_time":11.15241621,"remaining_time":12.62672283}, -{"learn":[0.3306543113],"iteration":469,"passed_time":11.19050052,"remaining_time":12.61907506}, -{"learn":[0.3304472924],"iteration":470,"passed_time":11.22114388,"remaining_time":12.60294079}, -{"learn":[0.3302050118],"iteration":471,"passed_time":11.24489435,"remaining_time":12.57903435}, -{"learn":[0.3302042287],"iteration":472,"passed_time":11.25318644,"remaining_time":12.5379054}, -{"learn":[0.3302042278],"iteration":473,"passed_time":11.2585324,"remaining_time":12.49364566}, -{"learn":[0.330077596],"iteration":474,"passed_time":11.28675622,"remaining_time":12.47483582}, -{"learn":[0.3299967493],"iteration":475,"passed_time":11.30915867,"remaining_time":12.44957803}, -{"learn":[0.3298204689],"iteration":476,"passed_time":11.33547981,"remaining_time":12.42862881}, -{"learn":[0.3296573275],"iteration":477,"passed_time":11.36870769,"remaining_time":12.41519961}, -{"learn":[0.3295569924],"iteration":478,"passed_time":11.39100902,"remaining_time":12.38980313}, -{"learn":[0.329329664],"iteration":479,"passed_time":11.41344547,"remaining_time":12.36456593}, -{"learn":[0.329131173],"iteration":480,"passed_time":11.43771739,"remaining_time":12.34132084}, -{"learn":[0.3289868495],"iteration":481,"passed_time":11.45959838,"remaining_time":12.31550199}, -{"learn":[0.3288599313],"iteration":482,"passed_time":11.481945,"remaining_time":12.29019786}, -{"learn":[0.3285822513],"iteration":483,"passed_time":11.50452789,"remaining_time":12.26515783}, -{"learn":[0.3282988587],"iteration":484,"passed_time":11.52631836,"remaining_time":12.23928651}, -{"learn":[0.3282849959],"iteration":485,"passed_time":11.54937837,"remaining_time":12.21477465}, -{"learn":[0.3281045884],"iteration":486,"passed_time":11.57297806,"remaining_time":12.19083726}, -{"learn":[0.3279303325],"iteration":487,"passed_time":11.59488577,"remaining_time":12.16512606}, -{"learn":[0.3277856921],"iteration":488,"passed_time":11.61756328,"remaining_time":12.14023484}, -{"learn":[0.3277359886],"iteration":489,"passed_time":11.63917325,"remaining_time":12.11424154}, -{"learn":[0.3275212786],"iteration":490,"passed_time":11.66039364,"remaining_time":12.08786225}, -{"learn":[0.3273286251],"iteration":491,"passed_time":11.68281481,"remaining_time":12.06274374}, -{"learn":[0.3271903876],"iteration":492,"passed_time":11.70660697,"remaining_time":12.03904611}, -{"learn":[0.3270655317],"iteration":493,"passed_time":11.73027173,"remaining_time":12.0152176}, -{"learn":[0.3270389056],"iteration":494,"passed_time":11.73848459,"remaining_time":11.97562569}, -{"learn":[0.326813723],"iteration":495,"passed_time":11.76041241,"remaining_time":11.95009648}, -{"learn":[0.3265823051],"iteration":496,"passed_time":11.7841956,"remaining_time":11.92645954}, -{"learn":[0.326481471],"iteration":497,"passed_time":11.80409698,"remaining_time":11.898909}, -{"learn":[0.3263322725],"iteration":498,"passed_time":11.82616979,"remaining_time":11.87356926}, -{"learn":[0.3261136043],"iteration":499,"passed_time":11.86466859,"remaining_time":11.86466859}, -{"learn":[0.3259491378],"iteration":500,"passed_time":11.88944591,"remaining_time":11.84198305}, -{"learn":[0.3255755285],"iteration":501,"passed_time":11.91145079,"remaining_time":11.81653883}, -{"learn":[0.325431063],"iteration":502,"passed_time":11.9334484,"remaining_time":11.79110111}, -{"learn":[0.3251221632],"iteration":503,"passed_time":11.9553648,"remaining_time":11.7655971}, -{"learn":[0.3249533874],"iteration":504,"passed_time":11.97708778,"remaining_time":11.73991773}, -{"learn":[0.3245557056],"iteration":505,"passed_time":11.99903462,"remaining_time":11.71447253}, -{"learn":[0.3244819748],"iteration":506,"passed_time":12.02060642,"remaining_time":11.68867646}, -{"learn":[0.3244571553],"iteration":507,"passed_time":12.0398824,"remaining_time":11.66067351}, -{"learn":[0.3243944087],"iteration":508,"passed_time":12.06431851,"remaining_time":11.63768249}, -{"learn":[0.3243932505],"iteration":509,"passed_time":12.07556097,"remaining_time":11.60200956}, -{"learn":[0.3243735511],"iteration":510,"passed_time":12.08694504,"remaining_time":11.56656776}, -{"learn":[0.3242641705],"iteration":511,"passed_time":12.10916608,"remaining_time":11.54154892}, -{"learn":[0.3238891977],"iteration":512,"passed_time":12.13107379,"remaining_time":11.51624354}, -{"learn":[0.3237673065],"iteration":513,"passed_time":12.15282633,"remaining_time":11.49080466}, -{"learn":[0.3236015493],"iteration":514,"passed_time":12.17484746,"remaining_time":11.46563305}, -{"learn":[0.3233656351],"iteration":515,"passed_time":12.19863579,"remaining_time":11.44213125}, -{"learn":[0.3232732615],"iteration":516,"passed_time":12.22211573,"remaining_time":11.41834023}, -{"learn":[0.3231361121],"iteration":517,"passed_time":12.24498624,"remaining_time":11.39398333}, -{"learn":[0.32299658],"iteration":518,"passed_time":12.2674654,"remaining_time":11.36926947}, -{"learn":[0.3229860023],"iteration":519,"passed_time":12.27596326,"remaining_time":11.3316584}, -{"learn":[0.322752282],"iteration":520,"passed_time":12.30415749,"remaining_time":11.31226764}, -{"learn":[0.3227072424],"iteration":521,"passed_time":12.32641801,"remaining_time":11.2874096}, -{"learn":[0.3226090551],"iteration":522,"passed_time":12.35471927,"remaining_time":11.26807092}, -{"learn":[0.3224092365],"iteration":523,"passed_time":12.39195665,"remaining_time":11.25681558}, -{"learn":[0.3224092358],"iteration":524,"passed_time":12.39737271,"remaining_time":11.21667054}, -{"learn":[0.322391538],"iteration":525,"passed_time":12.40533802,"remaining_time":11.1789548}, -{"learn":[0.3222006912],"iteration":526,"passed_time":12.42948673,"remaining_time":11.15587708}, -{"learn":[0.3221324811],"iteration":527,"passed_time":12.4519036,"remaining_time":11.13124716}, -{"learn":[0.3221061996],"iteration":528,"passed_time":12.46338891,"remaining_time":11.09689258}, -{"learn":[0.3220694662],"iteration":529,"passed_time":12.48500417,"remaining_time":11.07160747}, -{"learn":[0.3219756017],"iteration":530,"passed_time":12.5070657,"remaining_time":11.04673035}, -{"learn":[0.3219724232],"iteration":531,"passed_time":12.51552779,"remaining_time":11.00990039}, -{"learn":[0.3215889876],"iteration":532,"passed_time":12.53731651,"remaining_time":10.98485331}, -{"learn":[0.3214944521],"iteration":533,"passed_time":12.56633158,"remaining_time":10.96612456}, -{"learn":[0.3214735687],"iteration":534,"passed_time":12.57450905,"remaining_time":10.92924618}, -{"learn":[0.3211667307],"iteration":535,"passed_time":12.59607083,"remaining_time":10.90406132}, -{"learn":[0.3210361058],"iteration":536,"passed_time":12.6184051,"remaining_time":10.87955598}, -{"learn":[0.3209803394],"iteration":537,"passed_time":12.63658166,"remaining_time":10.85148834}, -{"learn":[0.3209803394],"iteration":538,"passed_time":12.64103614,"remaining_time":10.81172108}, -{"learn":[0.3209335447],"iteration":539,"passed_time":12.65243667,"remaining_time":10.77800161}, -{"learn":[0.3208255558],"iteration":540,"passed_time":12.67388975,"remaining_time":10.75289352}, -{"learn":[0.3204879029],"iteration":541,"passed_time":12.69799169,"remaining_time":10.73003726}, -{"learn":[0.3203255553],"iteration":542,"passed_time":12.72041289,"remaining_time":10.70576186}, -{"learn":[0.3203078373],"iteration":543,"passed_time":12.7392844,"remaining_time":10.67851781}, -{"learn":[0.3201807045],"iteration":544,"passed_time":12.76135281,"remaining_time":10.65397345}, -{"learn":[0.3200168479],"iteration":545,"passed_time":12.78311559,"remaining_time":10.62918403}, -{"learn":[0.3199792981],"iteration":546,"passed_time":12.80212903,"remaining_time":10.6021288}, -{"learn":[0.3198294345],"iteration":547,"passed_time":12.82745361,"remaining_time":10.58030845}, -{"learn":[0.3198294344],"iteration":548,"passed_time":12.83217461,"remaining_time":10.54154964}, -{"learn":[0.319647278],"iteration":549,"passed_time":12.88575667,"remaining_time":10.54289182}, -{"learn":[0.3196091956],"iteration":550,"passed_time":12.91083071,"remaining_time":10.52080397}, -{"learn":[0.3194977614],"iteration":551,"passed_time":12.95694549,"remaining_time":10.51578184}, -{"learn":[0.3194412347],"iteration":552,"passed_time":12.97875463,"remaining_time":10.49096442}, -{"learn":[0.319377863],"iteration":553,"passed_time":13.00016171,"remaining_time":10.46583416}, -{"learn":[0.319212072],"iteration":554,"passed_time":13.02191225,"remaining_time":10.44099271}, -{"learn":[0.3192120719],"iteration":555,"passed_time":13.02651986,"remaining_time":10.40247269}, -{"learn":[0.3191064217],"iteration":556,"passed_time":13.05525834,"remaining_time":10.38326651}, -{"learn":[0.3190443835],"iteration":557,"passed_time":13.07782907,"remaining_time":10.35914059}, -{"learn":[0.3189403324],"iteration":558,"passed_time":13.09956572,"remaining_time":10.33436222}, -{"learn":[0.3187465621],"iteration":559,"passed_time":13.12216052,"remaining_time":10.31026898}, -{"learn":[0.3185833798],"iteration":560,"passed_time":13.14414249,"remaining_time":10.28570152}, -{"learn":[0.3184379867],"iteration":561,"passed_time":13.16591073,"remaining_time":10.26097669}, -{"learn":[0.3182631548],"iteration":562,"passed_time":13.18804655,"remaining_time":10.23654768}, -{"learn":[0.3181986719],"iteration":563,"passed_time":13.2078342,"remaining_time":10.21031155}, -{"learn":[0.3180519592],"iteration":564,"passed_time":13.23026202,"remaining_time":10.18613094}, -{"learn":[0.3180225705],"iteration":565,"passed_time":13.24844462,"remaining_time":10.15870135}, -{"learn":[0.3178969873],"iteration":566,"passed_time":13.27035077,"remaining_time":10.13414794}, -{"learn":[0.3176445672],"iteration":567,"passed_time":13.29577878,"remaining_time":10.11228245}, -{"learn":[0.317322944],"iteration":568,"passed_time":13.32134689,"remaining_time":10.09051056}, -{"learn":[0.317228846],"iteration":569,"passed_time":13.34365403,"remaining_time":10.06626532}, -{"learn":[0.3170580473],"iteration":570,"passed_time":13.37570572,"remaining_time":10.04934808}, -{"learn":[0.3167691984],"iteration":571,"passed_time":13.40324414,"remaining_time":10.02900086}, -{"learn":[0.316670053],"iteration":572,"passed_time":13.42788682,"remaining_time":10.00647063}, -{"learn":[0.3164501718],"iteration":573,"passed_time":13.45331304,"remaining_time":9.984514557}, -{"learn":[0.3163287563],"iteration":574,"passed_time":13.49501302,"remaining_time":9.974574844}, -{"learn":[0.3162894601],"iteration":575,"passed_time":13.53471092,"remaining_time":9.963051095}, -{"learn":[0.3162494869],"iteration":576,"passed_time":13.58837703,"remaining_time":9.961669818}, -{"learn":[0.3159951385],"iteration":577,"passed_time":13.63095028,"remaining_time":9.952008685}, -{"learn":[0.3157978717],"iteration":578,"passed_time":13.67218492,"remaining_time":9.941260534}, -{"learn":[0.3155850358],"iteration":579,"passed_time":13.70123406,"remaining_time":9.921583284}, -{"learn":[0.315355402],"iteration":580,"passed_time":13.72422975,"remaining_time":9.897508201}, -{"learn":[0.3150989936],"iteration":581,"passed_time":13.74689945,"remaining_time":9.873202701}, -{"learn":[0.3148114702],"iteration":582,"passed_time":13.76882133,"remaining_time":9.848367913}, -{"learn":[0.3144519215],"iteration":583,"passed_time":13.79434107,"remaining_time":9.826105965}, -{"learn":[0.3143229251],"iteration":584,"passed_time":13.84787179,"remaining_time":9.823703919}, -{"learn":[0.3141904066],"iteration":585,"passed_time":13.90335501,"remaining_time":9.82250678}, -{"learn":[0.3140027965],"iteration":586,"passed_time":13.9489919,"remaining_time":9.814197027}, -{"learn":[0.3138685082],"iteration":587,"passed_time":13.99129849,"remaining_time":9.80342683}, -{"learn":[0.3138181058],"iteration":588,"passed_time":14.03659042,"remaining_time":9.794632705}, -{"learn":[0.313574669],"iteration":589,"passed_time":14.06815424,"remaining_time":9.776174981}, -{"learn":[0.3134898049],"iteration":590,"passed_time":14.09487967,"remaining_time":9.754324512}, -{"learn":[0.313320396],"iteration":591,"passed_time":14.11692573,"remaining_time":9.729232599}, -{"learn":[0.3131676802],"iteration":592,"passed_time":14.13857285,"remaining_time":9.70387715}, -{"learn":[0.3129757867],"iteration":593,"passed_time":14.16077145,"remaining_time":9.678911124}, -{"learn":[0.3127907517],"iteration":594,"passed_time":14.18313665,"remaining_time":9.654067803}, -{"learn":[0.3125289035],"iteration":595,"passed_time":14.20720097,"remaining_time":9.630384553}, -{"learn":[0.3123599621],"iteration":596,"passed_time":14.23064686,"remaining_time":9.606282555}, -{"learn":[0.3121137456],"iteration":597,"passed_time":14.25288883,"remaining_time":9.581373427}, -{"learn":[0.3119284015],"iteration":598,"passed_time":14.27492804,"remaining_time":9.55633747}, -{"learn":[0.3119267487],"iteration":599,"passed_time":14.28632892,"remaining_time":9.52421928}, -{"learn":[0.3119112192],"iteration":600,"passed_time":14.30279762,"remaining_time":9.495534525}, -{"learn":[0.3118470126],"iteration":601,"passed_time":14.32951711,"remaining_time":9.473667455}, -{"learn":[0.3117666572],"iteration":602,"passed_time":14.35212255,"remaining_time":9.449075711}, -{"learn":[0.3116060866],"iteration":603,"passed_time":14.37601067,"remaining_time":9.425331499}, -{"learn":[0.311604484],"iteration":604,"passed_time":14.38875212,"remaining_time":9.394309237}, -{"learn":[0.3113084243],"iteration":605,"passed_time":14.43525,"remaining_time":9.385294556}, -{"learn":[0.3112684311],"iteration":606,"passed_time":14.48716047,"remaining_time":9.379660734}, -{"learn":[0.3111182078],"iteration":607,"passed_time":14.55474728,"remaining_time":9.383981801}, -{"learn":[0.3109941013],"iteration":608,"passed_time":14.60355513,"remaining_time":9.376009941}, -{"learn":[0.3108986608],"iteration":609,"passed_time":14.65277384,"remaining_time":9.368166881}, -{"learn":[0.3107318621],"iteration":610,"passed_time":14.67810413,"remaining_time":9.344979556}, -{"learn":[0.3106806721],"iteration":611,"passed_time":14.70494523,"remaining_time":9.322743053}, -{"learn":[0.3105565129],"iteration":612,"passed_time":14.73133964,"remaining_time":9.300209526}, -{"learn":[0.3103924843],"iteration":613,"passed_time":14.7562304,"remaining_time":9.276718134}, -{"learn":[0.3102082538],"iteration":614,"passed_time":14.78098881,"remaining_time":9.253139335}, -{"learn":[0.3100890351],"iteration":615,"passed_time":14.805187,"remaining_time":9.229207479}, -{"learn":[0.3099299428],"iteration":616,"passed_time":14.83016501,"remaining_time":9.205758831}, -{"learn":[0.3099148853],"iteration":617,"passed_time":14.85097273,"remaining_time":9.179727481}, -{"learn":[0.3098292521],"iteration":618,"passed_time":14.87482513,"remaining_time":9.155587033}, -{"learn":[0.3097022539],"iteration":619,"passed_time":14.89875155,"remaining_time":9.131492883}, -{"learn":[0.3095586079],"iteration":620,"passed_time":14.92966579,"remaining_time":9.111663983}, -{"learn":[0.3095365057],"iteration":621,"passed_time":14.96288268,"remaining_time":9.093198797}, -{"learn":[0.3093867074],"iteration":622,"passed_time":14.98789146,"remaining_time":9.069719231}, -{"learn":[0.3090971237],"iteration":623,"passed_time":15.01706544,"remaining_time":9.048744563}, -{"learn":[0.3090567117],"iteration":624,"passed_time":15.03814037,"remaining_time":9.022884225}, -{"learn":[0.3090514438],"iteration":625,"passed_time":15.05096904,"remaining_time":8.992112496}, -{"learn":[0.3089121979],"iteration":626,"passed_time":15.1048228,"remaining_time":8.985803678}, -{"learn":[0.3088734529],"iteration":627,"passed_time":15.13101518,"remaining_time":8.962958036}, -{"learn":[0.3087798566],"iteration":628,"passed_time":15.15559681,"remaining_time":8.939151693}, -{"learn":[0.3085569569],"iteration":629,"passed_time":15.18628248,"remaining_time":8.918927806}, -{"learn":[0.3083945767],"iteration":630,"passed_time":15.21558256,"remaining_time":8.89786048}, -{"learn":[0.3082636175],"iteration":631,"passed_time":15.24035069,"remaining_time":8.874128248}, -{"learn":[0.308097633],"iteration":632,"passed_time":15.26563185,"remaining_time":8.850690187}, -{"learn":[0.3080161232],"iteration":633,"passed_time":15.2904056,"remaining_time":8.826953388}, -{"learn":[0.3079940498],"iteration":634,"passed_time":15.31462655,"remaining_time":8.802895579}, -{"learn":[0.3077049522],"iteration":635,"passed_time":15.33624358,"remaining_time":8.777346955}, -{"learn":[0.3074875181],"iteration":636,"passed_time":15.35822191,"remaining_time":8.752016568}, -{"learn":[0.3073941113],"iteration":637,"passed_time":15.38096155,"remaining_time":8.727128651}, -{"learn":[0.3070378662],"iteration":638,"passed_time":15.4242628,"remaining_time":8.713863648}, -{"learn":[0.3069235694],"iteration":639,"passed_time":15.45524212,"remaining_time":8.693573694}, -{"learn":[0.3068787888],"iteration":640,"passed_time":15.47926165,"remaining_time":8.669352466}, -{"learn":[0.3067815049],"iteration":641,"passed_time":15.51473893,"remaining_time":8.651521084}, -{"learn":[0.3067274077],"iteration":642,"passed_time":15.5591241,"remaining_time":8.638580567}, -{"learn":[0.3067273505],"iteration":643,"passed_time":15.57439764,"remaining_time":8.609449626}, -{"learn":[0.30654485],"iteration":644,"passed_time":15.60990177,"remaining_time":8.591496323}, -{"learn":[0.306542431],"iteration":645,"passed_time":15.62539633,"remaining_time":8.562523686}, -{"learn":[0.3063441894],"iteration":646,"passed_time":15.66716512,"remaining_time":8.547927802}, -{"learn":[0.3061944627],"iteration":647,"passed_time":15.71421395,"remaining_time":8.536116217}, -{"learn":[0.3061943966],"iteration":648,"passed_time":15.72971142,"remaining_time":8.507132062}, -{"learn":[0.3059553625],"iteration":649,"passed_time":15.75314641,"remaining_time":8.48246345}, -{"learn":[0.3057317972],"iteration":650,"passed_time":15.77560955,"remaining_time":8.457277621}, -{"learn":[0.305636842],"iteration":651,"passed_time":15.79864888,"remaining_time":8.432407685}, -{"learn":[0.3056207008],"iteration":652,"passed_time":15.807586,"remaining_time":8.400049528}, -{"learn":[0.3054612738],"iteration":653,"passed_time":15.82717303,"remaining_time":8.373397353}, -{"learn":[0.3053219515],"iteration":654,"passed_time":15.85644191,"remaining_time":8.35186635}, -{"learn":[0.3053042191],"iteration":655,"passed_time":15.87826828,"remaining_time":8.326408976}, -{"learn":[0.3051397112],"iteration":656,"passed_time":15.9139373,"remaining_time":8.308189489}, -{"learn":[0.3049434285],"iteration":657,"passed_time":15.96952775,"remaining_time":8.300271261}, -{"learn":[0.3048304658],"iteration":658,"passed_time":16.04229694,"remaining_time":8.301097507}, -{"learn":[0.3047096301],"iteration":659,"passed_time":16.09940802,"remaining_time":8.293634435}, -{"learn":[0.3042903415],"iteration":660,"passed_time":16.17347815,"remaining_time":8.29471875}, -{"learn":[0.3042772412],"iteration":661,"passed_time":16.1971499,"remaining_time":8.269843906}, -{"learn":[0.3040841367],"iteration":662,"passed_time":16.23556689,"remaining_time":8.252467636}, -{"learn":[0.3039029953],"iteration":663,"passed_time":16.26662238,"remaining_time":8.231302891}, -{"learn":[0.3036718031],"iteration":664,"passed_time":16.29784551,"remaining_time":8.210192852}, -{"learn":[0.303650484],"iteration":665,"passed_time":16.32503138,"remaining_time":8.18702775}, -{"learn":[0.3035074481],"iteration":666,"passed_time":16.38176119,"remaining_time":8.178600412}, -{"learn":[0.3033167887],"iteration":667,"passed_time":16.45371574,"remaining_time":8.17759525}, -{"learn":[0.3032946087],"iteration":668,"passed_time":16.4777861,"remaining_time":8.152686399}, -{"learn":[0.3032089215],"iteration":669,"passed_time":16.54239358,"remaining_time":8.147746092}, -{"learn":[0.3030821849],"iteration":670,"passed_time":16.56585054,"remaining_time":8.122451308}, -{"learn":[0.3029894163],"iteration":671,"passed_time":16.58837822,"remaining_time":8.096708416}, -{"learn":[0.302865477],"iteration":672,"passed_time":16.6172538,"remaining_time":8.074059426}, -{"learn":[0.3028067159],"iteration":673,"passed_time":16.63609293,"remaining_time":8.046537528}, -{"learn":[0.3027061901],"iteration":674,"passed_time":16.65820596,"remaining_time":8.020617682}, -{"learn":[0.3025646938],"iteration":675,"passed_time":16.68103442,"remaining_time":7.995051998}, -{"learn":[0.3025250377],"iteration":676,"passed_time":16.69379449,"remaining_time":7.964690722}, -{"learn":[0.3020655554],"iteration":677,"passed_time":16.7164889,"remaining_time":7.93909945}, -{"learn":[0.3020534847],"iteration":678,"passed_time":16.72869792,"remaining_time":7.908559695}, -{"learn":[0.3020343145],"iteration":679,"passed_time":16.76349825,"remaining_time":7.888705057}, -{"learn":[0.3018419602],"iteration":680,"passed_time":16.8072208,"remaining_time":7.872985954}, -{"learn":[0.3016898998],"iteration":681,"passed_time":16.85202658,"remaining_time":7.857689812}, -{"learn":[0.3015708555],"iteration":682,"passed_time":16.88419123,"remaining_time":7.836440148}, -{"learn":[0.3014896934],"iteration":683,"passed_time":16.92755692,"remaining_time":7.820333312}, -{"learn":[0.301382652],"iteration":684,"passed_time":16.95983849,"remaining_time":7.799049817}, -{"learn":[0.3010326679],"iteration":685,"passed_time":16.9823736,"remaining_time":7.773273046}, -{"learn":[0.3009148994],"iteration":686,"passed_time":17.00382974,"remaining_time":7.74701413}, -{"learn":[0.30091463],"iteration":687,"passed_time":17.01200093,"remaining_time":7.714744607}, -{"learn":[0.3007108906],"iteration":688,"passed_time":17.03396364,"remaining_time":7.688770234}, -{"learn":[0.3006049621],"iteration":689,"passed_time":17.05610975,"remaining_time":7.662889886}, -{"learn":[0.3004424014],"iteration":690,"passed_time":17.07874149,"remaining_time":7.63723751}, -{"learn":[0.3002521525],"iteration":691,"passed_time":17.10159712,"remaining_time":7.611693515}, -{"learn":[0.3001677248],"iteration":692,"passed_time":17.13033905,"remaining_time":7.588764918}, -{"learn":[0.3000495606],"iteration":693,"passed_time":17.15264985,"remaining_time":7.562983939}, -{"learn":[0.2999076935],"iteration":694,"passed_time":17.17487661,"remaining_time":7.537176065}, -{"learn":[0.2997878011],"iteration":695,"passed_time":17.1986168,"remaining_time":7.512039522}, -{"learn":[0.2997774515],"iteration":696,"passed_time":17.21042258,"remaining_time":7.481718853}, -{"learn":[0.2997265735],"iteration":697,"passed_time":17.23318916,"remaining_time":7.456193591}, -{"learn":[0.2996999851],"iteration":698,"passed_time":17.25516902,"remaining_time":7.430337448}, -{"learn":[0.2995208066],"iteration":699,"passed_time":17.27751275,"remaining_time":7.40464832}, -{"learn":[0.2995205559],"iteration":700,"passed_time":17.28640479,"remaining_time":7.373231143}, -{"learn":[0.2993846391],"iteration":701,"passed_time":17.31035892,"remaining_time":7.348272021}, -{"learn":[0.2991289729],"iteration":702,"passed_time":17.33681112,"remaining_time":7.324371129}, -{"learn":[0.2990709937],"iteration":703,"passed_time":17.36675989,"remaining_time":7.301933138}, -{"learn":[0.2990700247],"iteration":704,"passed_time":17.37528246,"remaining_time":7.270508264}, -{"learn":[0.2989463307],"iteration":705,"passed_time":17.39860223,"remaining_time":7.245310278}, -{"learn":[0.2989462517],"iteration":706,"passed_time":17.4147936,"remaining_time":7.217163403}, -{"learn":[0.2987724441],"iteration":707,"passed_time":17.44933557,"remaining_time":7.196618625}, -{"learn":[0.2987179044],"iteration":708,"passed_time":17.47148817,"remaining_time":7.170949307}, -{"learn":[0.2985139439],"iteration":709,"passed_time":17.49374124,"remaining_time":7.14533093}, -{"learn":[0.2982107603],"iteration":710,"passed_time":17.51577211,"remaining_time":7.119631702}, -{"learn":[0.2981313323],"iteration":711,"passed_time":17.53812766,"remaining_time":7.094074111}, -{"learn":[0.2980878246],"iteration":712,"passed_time":17.56294973,"remaining_time":7.069518333}, -{"learn":[0.297945571],"iteration":713,"passed_time":17.58499817,"remaining_time":7.043850809}, -{"learn":[0.2977036341],"iteration":714,"passed_time":17.60727451,"remaining_time":7.018284247}, -{"learn":[0.2976532538],"iteration":715,"passed_time":17.63636028,"remaining_time":6.99542782}, -{"learn":[0.2972948225],"iteration":716,"passed_time":17.65843273,"remaining_time":6.96978586}, -{"learn":[0.2972072898],"iteration":717,"passed_time":17.68058855,"remaining_time":6.94418659}, -{"learn":[0.2970205352],"iteration":718,"passed_time":17.70485614,"remaining_time":6.919422218}, -{"learn":[0.2968625698],"iteration":719,"passed_time":17.72750556,"remaining_time":6.89402994}, -{"learn":[0.296778922],"iteration":720,"passed_time":17.75015606,"remaining_time":6.868645689}, -{"learn":[0.2965144868],"iteration":721,"passed_time":17.77211874,"remaining_time":6.843004167}, -{"learn":[0.2964393382],"iteration":722,"passed_time":17.79405049,"remaining_time":6.817360975}, -{"learn":[0.2963119084],"iteration":723,"passed_time":17.81623397,"remaining_time":6.791824}, -{"learn":[0.2960547659],"iteration":724,"passed_time":17.83883814,"remaining_time":6.766455845}, -{"learn":[0.2959726674],"iteration":725,"passed_time":17.86237455,"remaining_time":6.741447144}, -{"learn":[0.2959442898],"iteration":726,"passed_time":17.92023066,"remaining_time":6.729330081}, -{"learn":[0.2958761466],"iteration":727,"passed_time":17.93996576,"remaining_time":6.702844349}, -{"learn":[0.2955882368],"iteration":728,"passed_time":17.96248951,"remaining_time":6.677413797}, -{"learn":[0.2955462923],"iteration":729,"passed_time":18.01029763,"remaining_time":6.661342958}, -{"learn":[0.2953304881],"iteration":730,"passed_time":18.03427959,"remaining_time":6.636417524}, -{"learn":[0.2951897941],"iteration":731,"passed_time":18.05763098,"remaining_time":6.6112638}, -{"learn":[0.2950068022],"iteration":732,"passed_time":18.08062816,"remaining_time":6.585985974}, -{"learn":[0.2949951299],"iteration":733,"passed_time":18.10420309,"remaining_time":6.560923737}, -{"learn":[0.2948810476],"iteration":734,"passed_time":18.13520748,"remaining_time":6.538544192}, -{"learn":[0.2948013604],"iteration":735,"passed_time":18.15856612,"remaining_time":6.513398716}, -{"learn":[0.2947457831],"iteration":736,"passed_time":18.17803723,"remaining_time":6.486870815}, -{"learn":[0.2945614394],"iteration":737,"passed_time":18.20246632,"remaining_time":6.462122191}, -{"learn":[0.2942434298],"iteration":738,"passed_time":18.22474259,"remaining_time":6.436614093}, -{"learn":[0.2939751704],"iteration":739,"passed_time":18.24767836,"remaining_time":6.411346451}, -{"learn":[0.2938581582],"iteration":740,"passed_time":18.2701075,"remaining_time":6.38590802}, -{"learn":[0.29382581],"iteration":741,"passed_time":18.27862017,"remaining_time":6.355638818}, -{"learn":[0.2935313901],"iteration":742,"passed_time":18.30044636,"remaining_time":6.330033264}, -{"learn":[0.293385087],"iteration":743,"passed_time":18.32305332,"remaining_time":6.304706517}, -{"learn":[0.2931688515],"iteration":744,"passed_time":18.345594,"remaining_time":6.27936439}, -{"learn":[0.2929784993],"iteration":745,"passed_time":18.37302216,"remaining_time":6.255693871}, -{"learn":[0.2927616087],"iteration":746,"passed_time":18.40180032,"remaining_time":6.232470523}, -{"learn":[0.2927348935],"iteration":747,"passed_time":18.43168092,"remaining_time":6.209603732}, -{"learn":[0.2926524614],"iteration":748,"passed_time":18.45196518,"remaining_time":6.183502349}, -{"learn":[0.292532443],"iteration":749,"passed_time":18.4744705,"remaining_time":6.158156832}, -{"learn":[0.2923772006],"iteration":750,"passed_time":18.4974505,"remaining_time":6.132976263}, -{"learn":[0.292133322],"iteration":751,"passed_time":18.52078053,"remaining_time":6.107916982}, -{"learn":[0.291968671],"iteration":752,"passed_time":18.54324045,"remaining_time":6.082576881}, -{"learn":[0.2918869516],"iteration":753,"passed_time":18.56544318,"remaining_time":6.057160508}, -{"learn":[0.2917246816],"iteration":754,"passed_time":18.58865144,"remaining_time":6.032078943}, -{"learn":[0.2916077821],"iteration":755,"passed_time":18.61393085,"remaining_time":6.007670802}, -{"learn":[0.2914285282],"iteration":756,"passed_time":18.64160854,"remaining_time":5.984030217}, -{"learn":[0.2913445284],"iteration":757,"passed_time":18.66491,"remaining_time":5.958981822}, -{"learn":[0.2913443551],"iteration":758,"passed_time":18.67365163,"remaining_time":5.929314943}, -{"learn":[0.291344066],"iteration":759,"passed_time":18.68183598,"remaining_time":5.899527152}, -{"learn":[0.2911691075],"iteration":760,"passed_time":18.70671,"remaining_time":5.8750377}, -{"learn":[0.2908231947],"iteration":761,"passed_time":18.73052052,"remaining_time":5.85021507}, -{"learn":[0.2907754775],"iteration":762,"passed_time":18.75434069,"remaining_time":5.825398092}, -{"learn":[0.2906135806],"iteration":763,"passed_time":18.77785035,"remaining_time":5.800487806}, -{"learn":[0.2905947266],"iteration":764,"passed_time":18.79727277,"remaining_time":5.774325622}, -{"learn":[0.2905295128],"iteration":765,"passed_time":18.81722706,"remaining_time":5.748343516}, -{"learn":[0.2902453894],"iteration":766,"passed_time":18.8405561,"remaining_time":5.723402309}, -{"learn":[0.2900319175],"iteration":767,"passed_time":18.86818989,"remaining_time":5.699765697}, -{"learn":[0.2899129422],"iteration":768,"passed_time":18.89480903,"remaining_time":5.675813895}, -{"learn":[0.289784147],"iteration":769,"passed_time":18.92259459,"remaining_time":5.65220358}, -{"learn":[0.2896433582],"iteration":770,"passed_time":18.95342224,"remaining_time":5.629485984}, -{"learn":[0.2893137629],"iteration":771,"passed_time":18.98132175,"remaining_time":5.605882591}, -{"learn":[0.2892767962],"iteration":772,"passed_time":19.00473325,"remaining_time":5.580950126}, -{"learn":[0.2890654442],"iteration":773,"passed_time":19.02796904,"remaining_time":5.55597029}, -{"learn":[0.2889086488],"iteration":774,"passed_time":19.05106703,"remaining_time":5.530954945}, -{"learn":[0.2887625489],"iteration":775,"passed_time":19.07288043,"remaining_time":5.505573731}, -{"learn":[0.2886341007],"iteration":776,"passed_time":19.09188604,"remaining_time":5.479395864}, -{"learn":[0.2886019791],"iteration":777,"passed_time":19.11521274,"remaining_time":5.454469444}, -{"learn":[0.2883053437],"iteration":778,"passed_time":19.14616913,"remaining_time":5.431711654}, -{"learn":[0.2881800888],"iteration":779,"passed_time":19.17758908,"remaining_time":5.409063588}, -{"learn":[0.2881493372],"iteration":780,"passed_time":19.2048689,"remaining_time":5.385232125}, -{"learn":[0.2879469616],"iteration":781,"passed_time":19.22949891,"remaining_time":5.360653148}, -{"learn":[0.2878030143],"iteration":782,"passed_time":19.2543558,"remaining_time":5.336136921}, -{"learn":[0.2876184323],"iteration":783,"passed_time":19.27809709,"remaining_time":5.311312464}, -{"learn":[0.2873371785],"iteration":784,"passed_time":19.30222025,"remaining_time":5.286595354}, -{"learn":[0.287291682],"iteration":785,"passed_time":19.32579239,"remaining_time":5.261729735}, -{"learn":[0.2870997336],"iteration":786,"passed_time":19.34894999,"remaining_time":5.236755206}, -{"learn":[0.2869277017],"iteration":787,"passed_time":19.37326393,"remaining_time":5.212096386}, -{"learn":[0.2868849395],"iteration":788,"passed_time":19.40280941,"remaining_time":5.188837498}, -{"learn":[0.2866008202],"iteration":789,"passed_time":19.43469769,"remaining_time":5.166185462}, -{"learn":[0.286433075],"iteration":790,"passed_time":19.46118519,"remaining_time":5.142083065}, -{"learn":[0.2863247094],"iteration":791,"passed_time":19.48432218,"remaining_time":5.117094713}, -{"learn":[0.2861530229],"iteration":792,"passed_time":19.50804413,"remaining_time":5.092263727}, -{"learn":[0.2860601397],"iteration":793,"passed_time":19.53011588,"remaining_time":5.067007394}, -{"learn":[0.285782435],"iteration":794,"passed_time":19.55303951,"remaining_time":5.041978741}, -{"learn":[0.2857395573],"iteration":795,"passed_time":19.57553156,"remaining_time":5.016844771}, -{"learn":[0.2855575819],"iteration":796,"passed_time":19.59854296,"remaining_time":4.991849714}, -{"learn":[0.2855155269],"iteration":797,"passed_time":19.62208312,"remaining_time":4.966993471}, -{"learn":[0.2853027357],"iteration":798,"passed_time":19.65210334,"remaining_time":4.943770678}, -{"learn":[0.2852479317],"iteration":799,"passed_time":19.67522216,"remaining_time":4.918805541}, -{"learn":[0.2851592302],"iteration":800,"passed_time":19.69927286,"remaining_time":4.894076529}, -{"learn":[0.2851054653],"iteration":801,"passed_time":19.72196036,"remaining_time":4.869012657}, -{"learn":[0.2850244642],"iteration":802,"passed_time":19.74401728,"remaining_time":4.843800004}, -{"learn":[0.2847988254],"iteration":803,"passed_time":19.76599931,"remaining_time":4.818576947}, -{"learn":[0.2847954492],"iteration":804,"passed_time":19.77443812,"remaining_time":4.790081285}, -{"learn":[0.2847668538],"iteration":805,"passed_time":19.79611254,"remaining_time":4.764821133}, -{"learn":[0.2846572393],"iteration":806,"passed_time":19.81859879,"remaining_time":4.739764022}, -{"learn":[0.2843730264],"iteration":807,"passed_time":19.84128682,"remaining_time":4.714761224}, -{"learn":[0.284141588],"iteration":808,"passed_time":19.86445555,"remaining_time":4.689877639}, -{"learn":[0.284041018],"iteration":809,"passed_time":19.89360124,"remaining_time":4.66640029}, -{"learn":[0.2839261452],"iteration":810,"passed_time":19.91740236,"remaining_time":4.641663434}, -{"learn":[0.2837666765],"iteration":811,"passed_time":19.94034852,"remaining_time":4.616730937}, -{"learn":[0.2836468321],"iteration":812,"passed_time":19.97762223,"remaining_time":4.595098841}, -{"learn":[0.2835959553],"iteration":813,"passed_time":20.00746692,"remaining_time":4.571730771}, -{"learn":[0.2834490287],"iteration":814,"passed_time":20.0425443,"remaining_time":4.549534595}, -{"learn":[0.2833510414],"iteration":815,"passed_time":20.08029393,"remaining_time":4.527909416}, -{"learn":[0.2833281488],"iteration":816,"passed_time":20.13046868,"remaining_time":4.509027868}, -{"learn":[0.2832314365],"iteration":817,"passed_time":20.18125584,"remaining_time":4.490206068}, -{"learn":[0.2831363442],"iteration":818,"passed_time":20.21374897,"remaining_time":4.467263204}, -{"learn":[0.2830386545],"iteration":819,"passed_time":20.24440593,"remaining_time":4.443893986}, -{"learn":[0.2829967032],"iteration":820,"passed_time":20.26934245,"remaining_time":4.419259804}, -{"learn":[0.2828641845],"iteration":821,"passed_time":20.29211314,"remaining_time":4.394155886}, -{"learn":[0.2825780004],"iteration":822,"passed_time":20.31456744,"remaining_time":4.368989596}, -{"learn":[0.2823101112],"iteration":823,"passed_time":20.33763218,"remaining_time":4.343960272}, -{"learn":[0.2821905518],"iteration":824,"passed_time":20.3598163,"remaining_time":4.318748912}, -{"learn":[0.2819471832],"iteration":825,"passed_time":20.38282979,"remaining_time":4.293719593}, -{"learn":[0.2819286265],"iteration":826,"passed_time":20.41034151,"remaining_time":4.269636131}, -{"learn":[0.2818280875],"iteration":827,"passed_time":20.45157014,"remaining_time":4.248393797}, -{"learn":[0.2817546833],"iteration":828,"passed_time":20.48226,"remaining_time":4.224929385}, -{"learn":[0.281587275],"iteration":829,"passed_time":20.50552826,"remaining_time":4.199927474}, -{"learn":[0.2814154303],"iteration":830,"passed_time":20.52883315,"remaining_time":4.174937188}, -{"learn":[0.2813141668],"iteration":831,"passed_time":20.55206209,"remaining_time":4.149935615}, -{"learn":[0.2812636016],"iteration":832,"passed_time":20.57503672,"remaining_time":4.124887314}, -{"learn":[0.2812172885],"iteration":833,"passed_time":20.59771147,"remaining_time":4.099784297}, -{"learn":[0.281044403],"iteration":834,"passed_time":20.62116367,"remaining_time":4.074840726}, -{"learn":[0.2809039898],"iteration":835,"passed_time":20.6444309,"remaining_time":4.049864436}, -{"learn":[0.2807492094],"iteration":836,"passed_time":20.67501646,"remaining_time":4.026317424}, -{"learn":[0.2807239878],"iteration":837,"passed_time":20.70039035,"remaining_time":4.001746106}, -{"learn":[0.2804031359],"iteration":838,"passed_time":20.7230494,"remaining_time":3.976651911}, -{"learn":[0.2803187933],"iteration":839,"passed_time":20.74560852,"remaining_time":3.951544481}, -{"learn":[0.280178431],"iteration":840,"passed_time":20.7676303,"remaining_time":3.92634152}, -{"learn":[0.2800356414],"iteration":841,"passed_time":20.78986737,"remaining_time":3.901186513}, -{"learn":[0.2798041757],"iteration":842,"passed_time":20.82095241,"remaining_time":3.877686274}, -{"learn":[0.2796980093],"iteration":843,"passed_time":20.86209174,"remaining_time":3.856026435}, -{"learn":[0.2795734447],"iteration":844,"passed_time":20.91418815,"remaining_time":3.83633037}, -{"learn":[0.2794519833],"iteration":845,"passed_time":20.9700609,"remaining_time":3.817245129}, -{"learn":[0.2793843487],"iteration":846,"passed_time":21.03050779,"remaining_time":3.798899282}, -{"learn":[0.2792873292],"iteration":847,"passed_time":21.07558948,"remaining_time":3.7777}, -{"learn":[0.2790298494],"iteration":848,"passed_time":21.12625627,"remaining_time":3.757437806}, -{"learn":[0.2788323685],"iteration":849,"passed_time":21.16095811,"remaining_time":3.734286725}, -{"learn":[0.2786864684],"iteration":850,"passed_time":21.19017393,"remaining_time":3.710147963}, -{"learn":[0.2785577462],"iteration":851,"passed_time":21.21701221,"remaining_time":3.685584281}, -{"learn":[0.2783806677],"iteration":852,"passed_time":21.25896675,"remaining_time":3.663620296}, -{"learn":[0.2783359299],"iteration":853,"passed_time":21.30383455,"remaining_time":3.642107547}, -{"learn":[0.2779284052],"iteration":854,"passed_time":21.35587999,"remaining_time":3.621757424}, -{"learn":[0.2778033989],"iteration":855,"passed_time":21.40426189,"remaining_time":3.600716954}, -{"learn":[0.2776226122],"iteration":856,"passed_time":21.46367464,"remaining_time":3.581453294}, -{"learn":[0.2775176091],"iteration":857,"passed_time":21.52034093,"remaining_time":3.561641505}, -{"learn":[0.277464614],"iteration":858,"passed_time":21.56305994,"remaining_time":3.539454542}, -{"learn":[0.2773883128],"iteration":859,"passed_time":21.60634761,"remaining_time":3.517312402}, -{"learn":[0.2772957947],"iteration":860,"passed_time":21.65155055,"remaining_time":3.495430344}, -{"learn":[0.27707483],"iteration":861,"passed_time":21.70220186,"remaining_time":3.474366423}, -{"learn":[0.2770458448],"iteration":862,"passed_time":21.7451042,"remaining_time":3.452003796}, -{"learn":[0.2770211473],"iteration":863,"passed_time":21.81058221,"remaining_time":3.4331472}, -{"learn":[0.2769134335],"iteration":864,"passed_time":21.86328773,"remaining_time":3.412189415}, -{"learn":[0.2767272068],"iteration":865,"passed_time":21.91124204,"remaining_time":3.390423133}, -{"learn":[0.2766651275],"iteration":866,"passed_time":21.95803491,"remaining_time":3.368418273}, -{"learn":[0.2766236015],"iteration":867,"passed_time":21.98865411,"remaining_time":3.343896707}, -{"learn":[0.2764190352],"iteration":868,"passed_time":22.01628927,"remaining_time":3.318911271}, -{"learn":[0.2763464641],"iteration":869,"passed_time":22.03879351,"remaining_time":3.293153053}, -{"learn":[0.2762459842],"iteration":870,"passed_time":22.06084269,"remaining_time":3.267334911}, -{"learn":[0.2761180515],"iteration":871,"passed_time":22.08344645,"remaining_time":3.241606818}, -{"learn":[0.275946872],"iteration":872,"passed_time":22.10540595,"remaining_time":3.21579216}, -{"learn":[0.2757959937],"iteration":873,"passed_time":22.1273718,"remaining_time":3.189987238}, -{"learn":[0.2757647864],"iteration":874,"passed_time":22.14955984,"remaining_time":3.164222834}, -{"learn":[0.2756470463],"iteration":875,"passed_time":22.17206941,"remaining_time":3.138512109}, -{"learn":[0.2755577091],"iteration":876,"passed_time":22.20870126,"remaining_time":3.114789344}, -{"learn":[0.2754184544],"iteration":877,"passed_time":22.25209122,"remaining_time":3.091976229}, -{"learn":[0.2753339669],"iteration":878,"passed_time":22.29594659,"remaining_time":3.069180361}, -{"learn":[0.2752650527],"iteration":879,"passed_time":22.34520085,"remaining_time":3.047072843}, -{"learn":[0.2751829758],"iteration":880,"passed_time":22.39111612,"remaining_time":3.024452688}, -{"learn":[0.2749553951],"iteration":881,"passed_time":22.45076853,"remaining_time":3.003617558}, -{"learn":[0.2746921572],"iteration":882,"passed_time":22.50653576,"remaining_time":2.98217971}, -{"learn":[0.274531533],"iteration":883,"passed_time":22.549468,"remaining_time":2.958979963}, -{"learn":[0.2744750271],"iteration":884,"passed_time":22.57161604,"remaining_time":2.933034853}, -{"learn":[0.2744017506],"iteration":885,"passed_time":22.59489076,"remaining_time":2.907243281}, -{"learn":[0.2742392046],"iteration":886,"passed_time":22.61704765,"remaining_time":2.881314976}, -{"learn":[0.2740790607],"iteration":887,"passed_time":22.64893968,"remaining_time":2.856623022}, -{"learn":[0.2740407752],"iteration":888,"passed_time":22.70215534,"remaining_time":2.834577326}, -{"learn":[0.2738901483],"iteration":889,"passed_time":22.7535678,"remaining_time":2.812238717}, -{"learn":[0.2737557],"iteration":890,"passed_time":22.7960782,"remaining_time":2.788745818}, -{"learn":[0.2734666852],"iteration":891,"passed_time":22.83855164,"remaining_time":2.765205804}, -{"learn":[0.2732677571],"iteration":892,"passed_time":22.88274713,"remaining_time":2.741829724}, -{"learn":[0.2731920043],"iteration":893,"passed_time":22.97414922,"remaining_time":2.72400427}, -{"learn":[0.2729943916],"iteration":894,"passed_time":23.02020561,"remaining_time":2.700694513}, -{"learn":[0.2728667147],"iteration":895,"passed_time":23.06679197,"remaining_time":2.677395496}, -{"learn":[0.2728023199],"iteration":896,"passed_time":23.08897489,"remaining_time":2.651242378}, -{"learn":[0.2726450733],"iteration":897,"passed_time":23.11081782,"remaining_time":2.625059485}, -{"learn":[0.2724540231],"iteration":898,"passed_time":23.13305855,"remaining_time":2.598930939}, -{"learn":[0.2722533675],"iteration":899,"passed_time":23.15498608,"remaining_time":2.572776231}, -{"learn":[0.2719798723],"iteration":900,"passed_time":23.17690447,"remaining_time":2.546629903}, -{"learn":[0.271838961],"iteration":901,"passed_time":23.20607422,"remaining_time":2.521280791}, -{"learn":[0.2716784004],"iteration":902,"passed_time":23.23319485,"remaining_time":2.495703102}, -{"learn":[0.2715997004],"iteration":903,"passed_time":23.26018112,"remaining_time":2.47010773}, -{"learn":[0.2715553961],"iteration":904,"passed_time":23.28532586,"remaining_time":2.444315975}, -{"learn":[0.2715040967],"iteration":905,"passed_time":23.30905326,"remaining_time":2.418378594}, -{"learn":[0.2714110852],"iteration":906,"passed_time":23.33292333,"remaining_time":2.392460717}, -{"learn":[0.2711783584],"iteration":907,"passed_time":23.35614585,"remaining_time":2.366481738}, -{"learn":[0.2709994568],"iteration":908,"passed_time":23.3780827,"remaining_time":2.340380117}, -{"learn":[0.2709026458],"iteration":909,"passed_time":23.40132538,"remaining_time":2.314416795}, -{"learn":[0.2707555722],"iteration":910,"passed_time":23.42400284,"remaining_time":2.288404229}, -{"learn":[0.2704601695],"iteration":911,"passed_time":23.44978174,"remaining_time":2.262698238}, -{"learn":[0.2702425665],"iteration":912,"passed_time":23.47564167,"remaining_time":2.236999808}, -{"learn":[0.2701770836],"iteration":913,"passed_time":23.50186451,"remaining_time":2.211335172}, -{"learn":[0.2699971732],"iteration":914,"passed_time":23.52737442,"remaining_time":2.185603088}, -{"learn":[0.269885267],"iteration":915,"passed_time":23.5500863,"remaining_time":2.159614901}, -{"learn":[0.2697787171],"iteration":916,"passed_time":23.57260904,"remaining_time":2.133616739}, -{"learn":[0.2697367678],"iteration":917,"passed_time":23.59423385,"remaining_time":2.107545943}, -{"learn":[0.2697244921],"iteration":918,"passed_time":23.61617665,"remaining_time":2.08151285}, -{"learn":[0.2695088772],"iteration":919,"passed_time":23.63857522,"remaining_time":2.05552828}, -{"learn":[0.2692360607],"iteration":920,"passed_time":23.66064219,"remaining_time":2.029523055}, -{"learn":[0.2691868859],"iteration":921,"passed_time":23.70105795,"remaining_time":2.005078655}, -{"learn":[0.2689761885],"iteration":922,"passed_time":23.74595053,"remaining_time":1.980973121}, -{"learn":[0.2687792851],"iteration":923,"passed_time":23.78731312,"remaining_time":1.956532248}, -{"learn":[0.2683329367],"iteration":924,"passed_time":23.82670762,"remaining_time":1.931895212}, -{"learn":[0.2682383468],"iteration":925,"passed_time":23.8676536,"remaining_time":1.907350288}, -{"learn":[0.2681411189],"iteration":926,"passed_time":23.90665807,"remaining_time":1.882617086}, -{"learn":[0.2680166243],"iteration":927,"passed_time":23.95383044,"remaining_time":1.858486844}, -{"learn":[0.2680097861],"iteration":928,"passed_time":24.00201139,"remaining_time":1.834384078}, -{"learn":[0.2678649617],"iteration":929,"passed_time":24.03147583,"remaining_time":1.808820762}, -{"learn":[0.2677926982],"iteration":930,"passed_time":24.05396793,"remaining_time":1.782732317}, -{"learn":[0.2676177519],"iteration":931,"passed_time":24.07625836,"remaining_time":1.756636876}, -{"learn":[0.2675419733],"iteration":932,"passed_time":24.09823464,"remaining_time":1.730527032}, -{"learn":[0.2674892708],"iteration":933,"passed_time":24.12017091,"remaining_time":1.704423212}, -{"learn":[0.2673888832],"iteration":934,"passed_time":24.14221881,"remaining_time":1.678336067}, -{"learn":[0.2673408241],"iteration":935,"passed_time":24.16428228,"remaining_time":1.652258618}, -{"learn":[0.267061597],"iteration":936,"passed_time":24.18717108,"remaining_time":1.626245228}, -{"learn":[0.2670483357],"iteration":937,"passed_time":24.21444774,"remaining_time":1.600528529}, -{"learn":[0.2669986935],"iteration":938,"passed_time":24.23981304,"remaining_time":1.574684341}, -{"learn":[0.2667914988],"iteration":939,"passed_time":24.2623561,"remaining_time":1.548661028}, -{"learn":[0.2666629717],"iteration":940,"passed_time":24.28508323,"remaining_time":1.522656653}, -{"learn":[0.2663673411],"iteration":941,"passed_time":24.30729083,"remaining_time":1.496627249}, -{"learn":[0.2662679127],"iteration":942,"passed_time":24.32898627,"remaining_time":1.470574992}, -{"learn":[0.2661806153],"iteration":943,"passed_time":24.35133577,"remaining_time":1.444570766}, -{"learn":[0.2660864679],"iteration":944,"passed_time":24.37363009,"remaining_time":1.418571063}, -{"learn":[0.2660772833],"iteration":945,"passed_time":24.39239275,"remaining_time":1.392377599}, -{"learn":[0.2659919549],"iteration":946,"passed_time":24.4160891,"remaining_time":1.366475948}, -{"learn":[0.2659294851],"iteration":947,"passed_time":24.43827206,"remaining_time":1.340495936}, -{"learn":[0.2658864129],"iteration":948,"passed_time":24.46472254,"remaining_time":1.314753266}, -{"learn":[0.2656973273],"iteration":949,"passed_time":24.49066288,"remaining_time":1.288982257}, -{"learn":[0.2654822842],"iteration":950,"passed_time":24.51928759,"remaining_time":1.263349203}, -{"learn":[0.2654357361],"iteration":951,"passed_time":24.54243954,"remaining_time":1.237433926}, -{"learn":[0.2652483245],"iteration":952,"passed_time":24.56490013,"remaining_time":1.211490353}, -{"learn":[0.2650786501],"iteration":953,"passed_time":24.58736673,"remaining_time":1.185554371}, -{"learn":[0.2650437914],"iteration":954,"passed_time":24.60931258,"remaining_time":1.159601116}, -{"learn":[0.2647589672],"iteration":955,"passed_time":24.63138292,"remaining_time":1.133661975}, -{"learn":[0.2645760187],"iteration":956,"passed_time":24.65376702,"remaining_time":1.107745018}, -{"learn":[0.2644273667],"iteration":957,"passed_time":24.67566451,"remaining_time":1.081814102}, -{"learn":[0.2642487937],"iteration":958,"passed_time":24.69985831,"remaining_time":1.055989771}, -{"learn":[0.2641295931],"iteration":959,"passed_time":24.72576284,"remaining_time":1.030240118}, -{"learn":[0.2639643656],"iteration":960,"passed_time":24.75146344,"remaining_time":1.004481867}, -{"learn":[0.2638271273],"iteration":961,"passed_time":24.77349924,"remaining_time":0.9785789721}, -{"learn":[0.26378499],"iteration":962,"passed_time":24.79605048,"remaining_time":0.9527039126}, -{"learn":[0.263585605],"iteration":963,"passed_time":24.81817395,"remaining_time":0.926819774}, -{"learn":[0.2635169052],"iteration":964,"passed_time":24.84010583,"remaining_time":0.9009364807}, -{"learn":[0.2634345598],"iteration":965,"passed_time":24.8624533,"remaining_time":0.8750759961}, -{"learn":[0.2634097593],"iteration":966,"passed_time":24.88446723,"remaining_time":0.8492113947}, -{"learn":[0.2633306114],"iteration":967,"passed_time":24.90657225,"remaining_time":0.8233577603}, -{"learn":[0.2631360168],"iteration":968,"passed_time":24.92874625,"remaining_time":0.79751407}, -{"learn":[0.2629773219],"iteration":969,"passed_time":24.95092926,"remaining_time":0.7716782246}, -{"learn":[0.262887109],"iteration":970,"passed_time":24.9825894,"remaining_time":0.7461329482}, -{"learn":[0.2627283493],"iteration":971,"passed_time":25.00857418,"remaining_time":0.720411602}, -{"learn":[0.2626493335],"iteration":972,"passed_time":25.03449893,"remaining_time":0.6946880484}, -{"learn":[0.2625417584],"iteration":973,"passed_time":25.0570851,"remaining_time":0.6688749616}, -{"learn":[0.2623745457],"iteration":974,"passed_time":25.08006808,"remaining_time":0.6430786686}, -{"learn":[0.2622252779],"iteration":975,"passed_time":25.10275279,"remaining_time":0.6172808064}, -{"learn":[0.2620765279],"iteration":976,"passed_time":25.12650391,"remaining_time":0.5915144215}, -{"learn":[0.2619056227],"iteration":977,"passed_time":25.14859496,"remaining_time":0.565714815}, -{"learn":[0.2616868552],"iteration":978,"passed_time":25.17038647,"remaining_time":0.5399163594}, -{"learn":[0.2614966561],"iteration":979,"passed_time":25.19430345,"remaining_time":0.5141694582}, -{"learn":[0.2614164281],"iteration":980,"passed_time":25.21690348,"remaining_time":0.488400781}, -{"learn":[0.2613355646],"iteration":981,"passed_time":25.24687944,"remaining_time":0.4627737576}, -{"learn":[0.2611604482],"iteration":982,"passed_time":25.26973736,"remaining_time":0.4370147865}, -{"learn":[0.2609467509],"iteration":983,"passed_time":25.29197863,"remaining_time":0.4112516851}, -{"learn":[0.2608803914],"iteration":984,"passed_time":25.31431985,"remaining_time":0.3854972567}, -{"learn":[0.2608064874],"iteration":985,"passed_time":25.33675271,"remaining_time":0.3597510527}, -{"learn":[0.2607424461],"iteration":986,"passed_time":25.35896906,"remaining_time":0.334008711}, -{"learn":[0.2604634089],"iteration":987,"passed_time":25.38110387,"remaining_time":0.3082725167}, -{"learn":[0.2603511649],"iteration":988,"passed_time":25.40489471,"remaining_time":0.282562024}, -{"learn":[0.2603052916],"iteration":989,"passed_time":25.42704056,"remaining_time":0.2568387935}, -{"learn":[0.2601502245],"iteration":990,"passed_time":25.45103872,"remaining_time":0.231139605}, -{"learn":[0.259927874],"iteration":991,"passed_time":25.47547957,"remaining_time":0.2054474159}, -{"learn":[0.2597611063],"iteration":992,"passed_time":25.50295153,"remaining_time":0.1797791145}, -{"learn":[0.2597041835],"iteration":993,"passed_time":25.52553775,"remaining_time":0.1540776926}, -{"learn":[0.2594450403],"iteration":994,"passed_time":25.55414859,"remaining_time":0.128412807}, -{"learn":[0.2593932196],"iteration":995,"passed_time":25.57650936,"remaining_time":0.1027169051}, -{"learn":[0.2592866892],"iteration":996,"passed_time":25.59858627,"remaining_time":0.07702683932}, -{"learn":[0.2591544901],"iteration":997,"passed_time":25.62038117,"remaining_time":0.05134344924}, -{"learn":[0.2591143866],"iteration":998,"passed_time":25.64281049,"remaining_time":0.02566847897}, -{"learn":[0.2589307024],"iteration":999,"passed_time":25.66502601,"remaining_time":0} -]} \ No newline at end of file diff --git a/metagpt/roles/catboost_info/learn/events.out.tfevents b/metagpt/roles/catboost_info/learn/events.out.tfevents deleted file mode 100644 index 47650f04ff5c490dc79edd3488548c212b0c2dce..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 54870 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 Date: Thu, 14 Dec 2023 15:49:01 +0800 Subject: [PATCH 160/637] update --- .gitignore | 1 + config/config.yaml | 98 ---------------------------------------------- 2 files changed, 1 insertion(+), 98 deletions(-) delete mode 100644 config/config.yaml diff --git a/.gitignore b/.gitignore index f2ccde1d1..76283319f 100644 --- a/.gitignore +++ b/.gitignore @@ -168,3 +168,4 @@ output.wav metagpt/roles/idea_agent.py .aider* /tests/metagpt/actions/check_data.py +/config/config.yaml diff --git a/config/config.yaml b/config/config.yaml deleted file mode 100644 index 694251f17..000000000 --- a/config/config.yaml +++ /dev/null @@ -1,98 +0,0 @@ -# DO NOT MODIFY THIS FILE, create a new key.yaml, define OPENAI_API_KEY. -# The configuration of key.yaml has a higher priority and will not enter git - -#### if OpenAI -## The official OPENAI_API_BASE is https://api.openai.com/v1 -## If the official OPENAI_API_BASE is not available, we recommend using the [openai-forward](https://github.com/beidongjiedeguang/openai-forward). -## Or, you can configure OPENAI_PROXY to access official OPENAI_API_BASE. -#OPENAI_API_BASE: "https://api.openai.com/v1" -#OPENAI_PROXY: "http://127.0.0.1:8118" -#OPENAI_API_KEY: "YOUR_API_KEY" # set the value to sk-xxx if you host the openai interface for open llm model -OPENAI_API_MODEL: "gpt-4" -MAX_TOKENS: 1500 -RPM: 10 - -#### if Spark -#SPARK_APPID : "YOUR_APPID" -#SPARK_API_SECRET : "YOUR_APISecret" -#SPARK_API_KEY : "YOUR_APIKey" -#DOMAIN : "generalv2" -#SPARK_URL : "ws://spark-api.xf-yun.com/v2.1/chat" - -#### if Anthropic -#Anthropic_API_KEY: "YOUR_API_KEY" - -#### if AZURE, check https://github.com/openai/openai-cookbook/blob/main/examples/azure/chat.ipynb -#### You can use ENGINE or DEPLOYMENT mode -OPENAI_API_TYPE: "azure" -OPENAI_API_BASE: "https://deepwisdom.openai.azure.com/" -OPENAI_API_KEY: "02ae6058d09849c691176befeae2107c" -#OPENAI_API_VERSION: "2023-05-15" -OPENAI_API_VERSION: "2023-07-01-preview" -DEPLOYMENT_ID: "GPT-4" -OPENAI_API_ENGINE: "gpt-4" - -#### if zhipuai from `https://open.bigmodel.cn`. You can set here or export API_KEY="YOUR_API_KEY" -# ZHIPUAI_API_KEY: "YOUR_API_KEY" - -#### for Search - -## Supported values: serpapi/google/serper/ddg -#SEARCH_ENGINE: serpapi - -## Visit https://serpapi.com/ to get key. -#SERPAPI_API_KEY: "YOUR_API_KEY" - -## Visit https://console.cloud.google.com/apis/credentials to get key. -#GOOGLE_API_KEY: "YOUR_API_KEY" -## Visit https://programmablesearchengine.google.com/controlpanel/create to get id. -#GOOGLE_CSE_ID: "YOUR_CSE_ID" - -## Visit https://serper.dev/ to get key. -#SERPER_API_KEY: "YOUR_API_KEY" - -#### for web access - -## Supported values: playwright/selenium -#WEB_BROWSER_ENGINE: playwright - -## Supported values: chromium/firefox/webkit, visit https://playwright.dev/python/docs/api/class-browsertype -##PLAYWRIGHT_BROWSER_TYPE: chromium - -## Supported values: chrome/firefox/edge/ie, visit https://www.selenium.dev/documentation/webdriver/browsers/ -# SELENIUM_BROWSER_TYPE: chrome - -#### for TTS - -#AZURE_TTS_SUBSCRIPTION_KEY: "YOUR_API_KEY" -#AZURE_TTS_REGION: "eastus" - -#### for Stable Diffusion -## Use SD service, based on https://github.com/AUTOMATIC1111/stable-diffusion-webui -SD_URL: "YOUR_SD_URL" -SD_T2I_API: "/sdapi/v1/txt2img" - -#### for Execution -#LONG_TERM_MEMORY: false - -#### for Mermaid CLI -## If you installed mmdc (Mermaid CLI) only for metagpt then enable the following configuration. -#PUPPETEER_CONFIG: "./config/puppeteer-config.json" -#MMDC: "./node_modules/.bin/mmdc" - - -### for calc_usage -# CALC_USAGE: false - -### for Research -MODEL_FOR_RESEARCHER_SUMMARY: gpt-3.5-turbo -MODEL_FOR_RESEARCHER_REPORT: gpt-3.5-turbo-16k - -### choose the engine for mermaid conversion, -# default is nodejs, you can change it to playwright,pyppeteer or ink -# MERMAID_ENGINE: nodejs - -### browser path for pyppeteer engine, support Chrome, Chromium,MS Edge -#PYPPETEER_EXECUTABLE_PATH: "/usr/bin/google-chrome-stable" - -PROMPT_FORMAT: json #json or markdown \ No newline at end of file From 5ba3fe9be8530c34dc325ad3c9c910cbba9ed908 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 15:52:20 +0800 Subject: [PATCH 161/637] update: use default WriteCodeWithTools --- metagpt/actions/write_analysis_code.py | 25 -------- metagpt/roles/ml_engineer.py | 87 +++++++++++++------------- 2 files changed, 42 insertions(+), 70 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 1cfc28811..136a4956f 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -202,32 +202,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name=module_name, tool_catalog=tool_catalog, ) - code_steps_ = eval(code_steps) - print(code_steps_) - new_code = "" - tool_context = "" - for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): - prompt = TOOL_USAGE_PROMPT.format( - user_requirement=plan.goal, - history_code=code_context, - current_task=plan.current_task.instruction, - column_info=column_info, - special_prompt=special_prompt, - code_steps=step_instruction, - module_name=module_name, - tool_catalog=tool_catalog, - ) - - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - - rsp = await self.llm.aask_code(prompt, **tool_config) - logger.info(f"rsp is: {rsp}") - new_code = new_code + "\n\n" + rsp["code"] - code_context = code_context + "\n\n" + new_code - tool_context = tool_context + "\n\n" + prompt - context = [Message(content=tool_context, role="user")] - return context, new_code else: diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index fa006b061..b38c752a4 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,4 +1,4 @@ -from typing import List +from typing import List import json from datetime import datetime @@ -42,24 +42,24 @@ class UpdateDataColumns(Action): class MLEngineer(Role): def __init__( - self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False + self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") self._watch([DownloadData, SubmitResult]) - + self.plan = Plan(goal=goal) self.use_tools = False self.use_code_steps = False self.execute_code = ExecutePyCode() self.auto_run = auto_run self.data_desc = {} - + # memory for working on each task, discarded each time a task is done self.working_memory = Memory() - + async def _plan_and_act(self): - + ### Actions in a multi-agent multi-turn setting ### memories = self.get_memories() if memories: @@ -69,29 +69,29 @@ class MLEngineer(Role): elif latest_event == SubmitResult: # self reflect on previous plan outcomes and think about how to improve the plan, add to working memory await self._reflect() - + # get feedback for improvement from human, add to working memory await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - + ### Common Procedure in both single- and multi-agent setting ### # create initial plan and update until confirmation await self._update_plan() - + while self.plan.current_task: task = self.plan.current_task logger.info(f"ready to take on task {task}") - + # take on current task code, result, success = await self._write_and_exec_code() - + # ask for acceptance, users can other refuse and change tasks in the plan review, task_result_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - + if self.auto_run: # if human confirms the task result, then we deem the task completed, regardless of whether the code run succeeds; # if auto mode, then the code run has to succeed for the task to be considered completed task_result_confirmed = success - + if task_result_confirmed: # tick off this task and record progress task.code = code @@ -100,12 +100,13 @@ class MLEngineer(Role): self.working_memory.clear() if self.use_tools: - success, new_code = await self._update_data_columns() + success, new_code = await self._update_data_columns() if success: task.code = task.code + "\n\n" + new_code - + confirmed_and_more = (ReviewConst.CONTINUE_WORD[0] in review.lower() - and review.lower() not in ReviewConst.CONTINUE_WORD[0]) # "confirm, ... (more content, such as changing downstream tasks)" + and review.lower() not in ReviewConst.CONTINUE_WORD[ + 0]) # "confirm, ... (more content, such as changing downstream tasks)" if confirmed_and_more: self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) await self._update_plan(review) @@ -114,23 +115,23 @@ class MLEngineer(Role): # Ask the Role to redo this task with help of review feedback, # useful when the code run is successful but the procedure or result is not what we want continue - + else: # update plan according to user's feedback and to take on changed tasks await self._update_plan(review) completed_plan_memory = self.get_useful_memories() # completed plan as a outcome self._rc.memory.add(completed_plan_memory[0]) # add to persistent memory - + summary = await SummarizeAnalysis().run(self.plan) rsp = Message(content=summary, cause_by=SummarizeAnalysis) self._rc.memory.add(rsp) - + # save code using datetime.now or keywords related to the goal of your project (plan.goal). project_record = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") return rsp - + async def _update_data_columns(self): rsp = await UpdateDataColumns().run(self.plan) is_update, code = rsp["is_update"], rsp["code"] @@ -147,23 +148,19 @@ class MLEngineer(Role): if self.use_code_steps else "" ) - + counter = 0 success = False debug_context = [] - + while not success and counter < max_retry: context = self.get_useful_memories() - # print("*" * 10) - # print(context) - # print("*" * 10) - # breakpoint() if counter > 0 and self.use_tools: code = await DebugCode().run( plan=self.plan.current_task.instruction, - code=code, - runtime_result=self.working_memory.get(), + code=code, + runtime_result=self.working_memory.get(), context=debug_context ) logger.info(f"new code \n{code}") @@ -185,30 +182,30 @@ class MLEngineer(Role): ) debug_context = tool_context cause_by = WriteCodeWithTools - + self.working_memory.add( Message(content=code, role="assistant", cause_by=cause_by) ) - + result, success = await self.execute_code.run(code) print(result) self.working_memory.add( Message(content=result, role="user", cause_by=ExecutePyCode) ) - + if "!pip" in code: success = False - + counter += 1 - + if not success and counter >= max_retry: logger.info("coding failed!") review, _ = await self._ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions - + return code, result, success - + async def _ask_review(self, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): auto_run = auto_run or self.auto_run if not auto_run: @@ -218,7 +215,7 @@ class MLEngineer(Role): self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) return review, confirmed return "", True - + async def _update_plan(self, review: str = "", max_tasks: int = 3, max_retries: int = 3): plan_confirmed = False while not plan_confirmed: @@ -229,7 +226,7 @@ class MLEngineer(Role): self.working_memory.add( Message(content=rsp, role="assistant", cause_by=WritePlan) ) - + # precheck plan before asking reviews is_plan_valid, error = precheck_update_plan_from_rsp(rsp, self.plan) if not is_plan_valid and max_retries > 0: @@ -238,11 +235,11 @@ class MLEngineer(Role): self.working_memory.add(Message(content=error_msg, role="assistant", cause_by=WritePlan)) max_retries -= 1 continue - + _, plan_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - + update_plan_from_rsp(rsp, self.plan) - + self.working_memory.clear() async def _reflect(self): @@ -254,7 +251,7 @@ class MLEngineer(Role): reflection = await Reflect().run(context=context) self.working_memory.add(Message(content=reflection, role="assistant")) self.working_memory.add(Message(content=Reflect.REWRITE_PLAN_INSTRUCTION, role="user")) - + def get_useful_memories(self, task_exclude_field=None) -> List[Message]: """find useful memories only to reduce context length and improve performance""" # TODO dataset description , code steps @@ -271,9 +268,9 @@ class MLEngineer(Role): user_requirement=user_requirement, data_desc=data_desc, tasks=tasks, current_task=current_task ) context_msg = [Message(content=context, role="user")] - - return context_msg + self.get_working_memories() + return context_msg + self.get_working_memories() + def get_working_memories(self) -> List[Message]: return self.working_memory.get() @@ -298,7 +295,6 @@ if __name__ == "__main__": # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." - save_dir = "" save_dir = DATA_PATH / "save" / "2023-12-14_15-11-40" @@ -365,7 +361,8 @@ if __name__ == "__main__": role = MLEngineer(goal=requirement, auto_run=auto_run) role.plan = Plan(**plan) role.execute_code = ExecutePyCode(nb) - import pdb;pdb.set_trace() + import pdb; + pdb.set_trace() else: logger.info("Run from scratch") role = MLEngineer(goal=requirement, auto_run=auto_run) From 48d542d383bdb4bd80da68c546d3a553d8c543ed Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 14 Dec 2023 15:54:02 +0800 Subject: [PATCH 162/637] recover code --- metagpt/actions/execute_code.py | 5 ----- 1 file changed, 5 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index c5ed8964e..36e01ed0e 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -157,11 +157,6 @@ class ExecutePyCode(ExecuteCode, Action): return code, language - def save_notebook(self, path: str): - path = Path(path) - path.parent.mkdir(parents=True, exist_ok=True) - nbformat.write(self.nb, path) - async def run(self, code: Union[str, Dict, Message], language: str = "python") -> Tuple[str, bool]: code, language = self._process_code(code, language) From 82ccdde687ff55734bfe16353d1511ea34c3f4ed Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 17:18:35 +0800 Subject: [PATCH 163/637] use tools --- metagpt/roles/ml_engineer.py | 24 +++++++++++------------- 1 file changed, 11 insertions(+), 13 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index b38c752a4..bd46ae79a 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -49,8 +49,8 @@ class MLEngineer(Role): self._watch([DownloadData, SubmitResult]) self.plan = Plan(goal=goal) - self.use_tools = False - self.use_code_steps = False + self.use_tools = True + self.use_code_steps = True self.execute_code = ExecutePyCode() self.auto_run = auto_run self.data_desc = {} @@ -101,8 +101,8 @@ class MLEngineer(Role): if self.use_tools: success, new_code = await self._update_data_columns() - if success: - task.code = task.code + "\n\n" + new_code + if success: + task.code = task.code + "\n\n" + new_code confirmed_and_more = (ReviewConst.CONTINUE_WORD[0] in review.lower() and review.lower() not in ReviewConst.CONTINUE_WORD[ @@ -245,9 +245,7 @@ class MLEngineer(Role): async def _reflect(self): context = self.get_memories() context = "\n".join([str(msg) for msg in context]) - # print("*" * 10) - # print(context) - # print("*" * 10) + reflection = await Reflect().run(context=context) self.working_memory.add(Message(content=reflection, role="assistant")) self.working_memory.add(Message(content=Reflect.REWRITE_PLAN_INSTRUCTION, role="user")) @@ -296,7 +294,7 @@ if __name__ == "__main__": # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." save_dir = "" - save_dir = DATA_PATH / "save" / "2023-12-14_15-11-40" + # save_dir = DATA_PATH / "save" / "2023-12-14_16-58-03" def load_history(save_dir: str = save_dir): @@ -328,13 +326,14 @@ if __name__ == "__main__": Returns: Path: The path to the saved history directory. """ - save_path = Path(save_dir) if save_dir else DATA_PATH / "save" / datetime.now().strftime( + # save_path = Path(save_dir) if save_dir else DATA_PATH / "save" / datetime.now().strftime( + # '%Y-%m-%d_%H-%M-%S') + save_path = DATA_PATH / "save" / datetime.now().strftime( '%Y-%m-%d_%H-%M-%S') - # overwrite + # overwrite exist trajectory save_path.mkdir(parents=True, exist_ok=True) plan = role.plan.dict() - logger.info(f"Plan is {plan}") with open(save_path / "plan.json", "w", encoding="utf-8") as plan_file: json.dump(plan, plan_file, indent=4, ensure_ascii=False) @@ -361,8 +360,7 @@ if __name__ == "__main__": role = MLEngineer(goal=requirement, auto_run=auto_run) role.plan = Plan(**plan) role.execute_code = ExecutePyCode(nb) - import pdb; - pdb.set_trace() + else: logger.info("Run from scratch") role = MLEngineer(goal=requirement, auto_run=auto_run) From b5f3034cbb0e2d40032951c00d5344e18dedc66c Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 17:19:03 +0800 Subject: [PATCH 164/637] add check --- metagpt/actions/write_analysis_code.py | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 136a4956f..dda6c66cd 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -107,7 +107,8 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): if self.schema_path is not None: self._load_tools(schema_path) - + logger.info(f"available_tools: {len(self.available_tools)}") + def _load_tools(self, schema_path): """Load tools from yaml file""" yml_files = schema_path.glob("*.yml") @@ -202,7 +203,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name=module_name, tool_catalog=tool_catalog, ) - + else: From c91503cf655c9a0044f9ffba6e2f92df32bd6c39 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 17:30:01 +0800 Subject: [PATCH 165/637] rm comments --- .gitignore | 2 -- metagpt/roles/ml_engineer.py | 16 ---------------- 2 files changed, 18 deletions(-) diff --git a/.gitignore b/.gitignore index 76283319f..d01469a36 100644 --- a/.gitignore +++ b/.gitignore @@ -167,5 +167,3 @@ tmp output.wav metagpt/roles/idea_agent.py .aider* -/tests/metagpt/actions/check_data.py -/config/config.yaml diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index bd46ae79a..cd2104c4b 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -274,29 +274,13 @@ class MLEngineer(Role): if __name__ == "__main__": - requirement = "Run data analysis on sklearn Iris dataset, include a plot" - # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" - # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" - # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" - - # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." - - # data_path = f"{DATA_PATH}/titanic" - # requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - # requirement = f"Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" - # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" - # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." save_dir = "" # save_dir = DATA_PATH / "save" / "2023-12-14_16-58-03" - def load_history(save_dir: str = save_dir): """ Load history from the specified save directory. From 4953929025e669aacc7aa2852d717df168e76655 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 21:44:18 +0800 Subject: [PATCH 166/637] add --- metagpt/actions/write_analysis_code.py | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index dda6c66cd..abfecbbc1 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -193,16 +193,16 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name = ML_MODULE_MAP[task_type] - prompt = TOOL_USAGE_PROMPT.format( - user_requirement=plan.goal, - history_code=code_context, - current_task=plan.current_task.instruction, - column_info=column_info, - special_prompt=special_prompt, - code_steps=code_steps, - module_name=module_name, - tool_catalog=tool_catalog, - ) + prompt = TOOL_USAGE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + special_prompt=special_prompt, + code_steps=code_steps, + module_name=module_name, + tool_catalog=tool_catalog, + ) From cf6577334c7bb72089ff25a2e4d6707300f05267 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 14 Dec 2023 21:46:19 +0800 Subject: [PATCH 167/637] update --- metagpt/actions/write_analysis_code.py | 103 ++++++++++++------------- 1 file changed, 49 insertions(+), 54 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index abfecbbc1..6970fb4f0 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -24,9 +24,9 @@ from metagpt.utils.common import create_func_config, remove_comments class BaseWriteAnalysisCode(Action): - DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt - REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" - + DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt + # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" + def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG # 全部转成list @@ -45,23 +45,23 @@ class BaseWriteAnalysisCode(Action): messages.append(p.to_dict()) elif isinstance(p.content, dict) and "code" in p.content: messages.append(p.content["code"]) - + # 添加默认的提示词 if ( - default_system_msg not in messages[0]["content"] - and messages[0]["role"] != "system" + default_system_msg not in messages[0]["content"] + and messages[0]["role"] != "system" ): messages.insert(0, {"role": "system", "content": default_system_msg}) elif ( - default_system_msg not in messages[0]["content"] - and messages[0]["role"] == "system" + default_system_msg not in messages[0]["content"] + and messages[0]["role"] == "system" ): messages[0] = { "role": "system", "content": messages[0]["content"] + default_system_msg, } return messages - + async def run( self, context: List[Message], plan: Plan = None, code_steps: str = "" ) -> str: @@ -79,19 +79,18 @@ class BaseWriteAnalysisCode(Action): class WriteCodeByGenerate(BaseWriteAnalysisCode): """Write code fully by generation""" - + def __init__(self, name: str = "", context=None, llm=None) -> str: super().__init__(name, context, llm) - + async def run( - self, - context: [List[Message]], - plan: Plan = None, - code_steps: str = "", - system_msg: str = None, - **kwargs, + self, + context: [List[Message]], + plan: Plan = None, + system_msg: str = None, + **kwargs, ) -> str: - context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) + # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) prompt = self.process_msg(context, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) return code_content["code"] @@ -99,16 +98,15 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - + def __init__(self, name: str = "", context=None, llm=None, schema_path=None): super().__init__(name, context, llm) self.schema_path = schema_path self.available_tools = {} - + if self.schema_path is not None: self._load_tools(schema_path) - logger.info(f"available_tools: {len(self.available_tools)}") - + def _load_tools(self, schema_path): """Load tools from yaml file""" yml_files = schema_path.glob("*.yml") @@ -116,7 +114,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module = yml_file.stem with open(yml_file, "r", encoding="utf-8") as f: self.available_tools[module] = yaml.safe_load(f) - + def _parse_recommend_tools(self, module: str, recommend_tools: list) -> dict: """ Parses and validates a list of recommended tools, and retrieves their schema from registry. @@ -133,15 +131,15 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): for tool in recommend_tools: if tool in available_tools: valid_tools.append(tool) - + tool_catalog = {tool: self.available_tools[module][tool] for tool in valid_tools} return tool_catalog - + async def _tool_recommendation( - self, - task: str, - code_steps: str, - available_tools: dict, + self, + task: str, + code_steps: str, + available_tools: dict, ) -> list: """ Recommend tools for the specified task. @@ -163,26 +161,26 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): rsp = await self.llm.aask_code(prompt, **tool_config) recommend_tools = rsp["recommend_tools"] return recommend_tools - + async def run( - self, - context: List[Message], - plan: Plan = None, - code_steps: str = "", - column_info: str = "", - **kwargs, + self, + context: List[Message], + plan: Plan = None, + column_info: str = "", + **kwargs, ) -> Tuple[List[Message], str]: task_type = plan.current_task.task_type available_tools = self.available_tools.get(task_type, {}) special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") - + code_steps = plan.current_task.code_steps + finished_tasks = plan.get_finished_tasks() code_context = [remove_comments(task.code) for task in finished_tasks] code_context = "\n\n".join(code_context) - + if len(available_tools) > 0: available_tools = {k: v["description"] for k, v in available_tools.items()} - + recommend_tools = await self._tool_recommendation( plan.current_task.instruction, code_steps, @@ -190,22 +188,19 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): ) tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) logger.info(f"Recommended tools: \n{recommend_tools}") - + module_name = ML_MODULE_MAP[task_type] - - prompt = TOOL_USAGE_PROMPT.format( - user_requirement=plan.goal, - history_code=code_context, - current_task=plan.current_task.instruction, - column_info=column_info, - special_prompt=special_prompt, - code_steps=code_steps, - module_name=module_name, - tool_catalog=tool_catalog, - ) - - + prompt = TOOL_USAGE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + special_prompt=special_prompt, + code_steps=code_steps, + module_name=module_name, + tool_catalog=tool_catalog, + ) else: prompt = GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, @@ -215,7 +210,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): special_prompt=special_prompt, code_steps=code_steps, ) - + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) context = [Message(content=prompt, role="user")] From 6a527f214a1ebe944823aadbc9f1bcfbaa3e6287 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 15 Dec 2023 00:35:44 +0800 Subject: [PATCH 168/637] update: use utils/save_code_file --- metagpt/roles/ml_engineer.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index d9a5027af..f7538ae2e 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -295,8 +295,6 @@ if __name__ == "__main__": requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." save_dir = "" - - # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" def load_history(save_dir: str = save_dir): From a42c4144a11058f3d44c2f8f154437de449cd506 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 15 Dec 2023 00:37:01 +0800 Subject: [PATCH 169/637] iterative step to code --- metagpt/actions/write_analysis_code.py | 46 ++++++++++++++++++++------ 1 file changed, 36 insertions(+), 10 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 6970fb4f0..0d548b806 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -191,16 +191,42 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name = ML_MODULE_MAP[task_type] - prompt = TOOL_USAGE_PROMPT.format( - user_requirement=plan.goal, - history_code=code_context, - current_task=plan.current_task.instruction, - column_info=column_info, - special_prompt=special_prompt, - code_steps=code_steps, - module_name=module_name, - tool_catalog=tool_catalog, - ) + # prompt = TOOL_USAGE_PROMPT.format( + # user_requirement=plan.goal, + # history_code=code_context, + # current_task=plan.current_task.instruction, + # column_info=column_info, + # special_prompt=special_prompt, + # code_steps=code_steps, + # module_name=module_name, + # tool_catalog=tool_catalog, + # ) + code_steps_ = eval(code_steps) + print(code_steps_) + + new_code = "" + tool_context = "" + for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): + prompt = TOOL_USAGE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + special_prompt=special_prompt, + code_steps=step_instruction, + module_name=module_name, + tool_catalog=tool_catalog, + ) + + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + + rsp = await self.llm.aask_code(prompt, **tool_config) + logger.info(f"rsp is: {rsp}") + new_code = new_code + "\n\n" + rsp["code"] + code_context = code_context + "\n\n" + new_code + tool_context = tool_context + "\n\n" + prompt + context = [Message(content=tool_context, role="user")] + return context, new_code else: prompt = GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, From 6957c2df65f514e25e35f7ad3e89b9edf3a89041 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 15 Dec 2023 00:41:13 +0800 Subject: [PATCH 170/637] revert to default --- metagpt/actions/write_analysis_code.py | 72 +++++++++++++------------- 1 file changed, 36 insertions(+), 36 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 0d548b806..29e5397e3 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -191,42 +191,42 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name = ML_MODULE_MAP[task_type] - # prompt = TOOL_USAGE_PROMPT.format( - # user_requirement=plan.goal, - # history_code=code_context, - # current_task=plan.current_task.instruction, - # column_info=column_info, - # special_prompt=special_prompt, - # code_steps=code_steps, - # module_name=module_name, - # tool_catalog=tool_catalog, - # ) - code_steps_ = eval(code_steps) - print(code_steps_) - - new_code = "" - tool_context = "" - for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): - prompt = TOOL_USAGE_PROMPT.format( - user_requirement=plan.goal, - history_code=code_context, - current_task=plan.current_task.instruction, - column_info=column_info, - special_prompt=special_prompt, - code_steps=step_instruction, - module_name=module_name, - tool_catalog=tool_catalog, - ) - - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - - rsp = await self.llm.aask_code(prompt, **tool_config) - logger.info(f"rsp is: {rsp}") - new_code = new_code + "\n\n" + rsp["code"] - code_context = code_context + "\n\n" + new_code - tool_context = tool_context + "\n\n" + prompt - context = [Message(content=tool_context, role="user")] - return context, new_code + prompt = TOOL_USAGE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + special_prompt=special_prompt, + code_steps=code_steps, + module_name=module_name, + tool_catalog=tool_catalog, + ) + # code_steps_ = eval(code_steps) + # print(code_steps_) + # + # new_code = "" + # tool_context = "" + # for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): + # prompt = TOOL_USAGE_PROMPT.format( + # user_requirement=plan.goal, + # history_code=code_context, + # current_task=plan.current_task.instruction, + # column_info=column_info, + # special_prompt=special_prompt, + # code_steps=step_instruction, + # module_name=module_name, + # tool_catalog=tool_catalog, + # ) + # + # tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + # + # rsp = await self.llm.aask_code(prompt, **tool_config) + # logger.info(f"rsp is: {rsp}") + # new_code = new_code + "\n\n" + rsp["code"] + # code_context = code_context + "\n\n" + new_code + # tool_context = tool_context + "\n\n" + prompt + # context = [Message(content=tool_context, role="user")] + # return context, new_code else: prompt = GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, From 2fe9f2b9cfed79677c11b16e34c3944d09b68df2 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 15 Dec 2023 10:06:46 +0800 Subject: [PATCH 171/637] remove old comments --- .../tools/functions/libs/data_preprocess.py | 50 ------------------- 1 file changed, 50 deletions(-) diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index ec3580889..8c70462ee 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -151,53 +151,3 @@ def get_column_info(df: pd.DataFrame) -> dict: columns=["Column_name", "Data_type", "NaN_Frequency(%)", "N_unique"], ) return samples.to_dict(orient='list') -# -# -# if __name__ == '__main__': -# def run(): -# V = { -# 'a': [-1, 2, 3, 6, 5, 4], -# 'b': [1.1, 2.2, 3.3, 6.6, 5.5, 4.4], -# 'c': ['aa', 'bb', 'cc', 'dd', 'ee', 'ff'], -# 'd': [1, None, 3, None, 5, 4], -# 'e': [1.1, np.NAN, 3.3, None, 5.5, 4.4], -# 'f': ['aa', np.NAN, 'cc', None, '', 'ff'], -# -# } -# -# df = pd.DataFrame(V) -# print(df.dtypes) -# -# numeric_features = ['a', 'b', 'd', 'e'] -# numeric_features_wo_miss = ['a', 'b', ] -# categorial_features = ['c', 'f'] -# -# df_ = fill_missing_value(df.copy(), numeric_features) -# print(df_) -# df_ = fill_missing_value(df.copy(), categorial_features, strategy='constant', fill_value='hehe') -# print(df_) -# -# df_ = fill_missing_value(df.copy(), numeric_features, strategy='constant', fill_value=999) -# print(df_) -# -# # df_ = label_encode(df.copy(), numeric_features + categorial_features, ) -# # print(df_) -# -# df_ = split_bins(df.copy(), numeric_features_wo_miss, strategy='quantile') -# print(df_) -# -# df_ = min_max_scale(df.copy(), numeric_features, ) -# print(df_) -# -# df_ = standard_scale(df.copy(), numeric_features, ) -# print(df_) -# -# df_ = log_transform(df.copy(), numeric_features, ) -# print(df_) -# -# df_ = max_abs_scale(df.copy(), numeric_features, ) -# print(df_) -# -# df_ = robust_scale(df.copy(), numeric_features, ) -# print(df_) -# run() \ No newline at end of file From 9ea745553c964c5559083ead545c8dac805ea12d Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 15 Dec 2023 10:22:50 +0800 Subject: [PATCH 172/637] update: iterative step to generate code --- metagpt/actions/write_analysis_code.py | 74 +++++++++++++------------- 1 file changed, 38 insertions(+), 36 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 29e5397e3..cce36d8c9 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -191,42 +191,44 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name = ML_MODULE_MAP[task_type] - prompt = TOOL_USAGE_PROMPT.format( - user_requirement=plan.goal, - history_code=code_context, - current_task=plan.current_task.instruction, - column_info=column_info, - special_prompt=special_prompt, - code_steps=code_steps, - module_name=module_name, - tool_catalog=tool_catalog, - ) - # code_steps_ = eval(code_steps) - # print(code_steps_) - # - # new_code = "" - # tool_context = "" - # for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): - # prompt = TOOL_USAGE_PROMPT.format( - # user_requirement=plan.goal, - # history_code=code_context, - # current_task=plan.current_task.instruction, - # column_info=column_info, - # special_prompt=special_prompt, - # code_steps=step_instruction, - # module_name=module_name, - # tool_catalog=tool_catalog, - # ) - # - # tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - # - # rsp = await self.llm.aask_code(prompt, **tool_config) - # logger.info(f"rsp is: {rsp}") - # new_code = new_code + "\n\n" + rsp["code"] - # code_context = code_context + "\n\n" + new_code - # tool_context = tool_context + "\n\n" + prompt - # context = [Message(content=tool_context, role="user")] - # return context, new_code + # prompt = TOOL_USAGE_PROMPT.format( + # user_requirement=plan.goal, + # history_code=code_context, + # current_task=plan.current_task.instruction, + # column_info=column_info, + # special_prompt=special_prompt, + # code_steps=code_steps, + # module_name=module_name, + # tool_catalog=tool_catalog, + # ) + + code_steps_ = eval(code_steps) + print(code_steps_) + + new_code = "" + tool_context = "" + for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): + prompt = TOOL_USAGE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + special_prompt=special_prompt, + code_steps=step_instruction, + module_name=module_name, + tool_catalog=tool_catalog, + ) + + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + + rsp = await self.llm.aask_code(prompt, **tool_config) + logger.info(f"rsp is: {rsp}") + new_code = new_code + "\n\n" + rsp["code"] + code_context = code_context + "\n\n" + new_code + tool_context = tool_context + "\n\n" + prompt + context = [Message(content=tool_context, role="user")] + return context, new_code + else: prompt = GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, From a723068f9f685241e32199c01be93a3758885d68 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 15 Dec 2023 10:26:29 +0800 Subject: [PATCH 173/637] add --- config/config.yaml | 97 ++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 97 insertions(+) create mode 100644 config/config.yaml diff --git a/config/config.yaml b/config/config.yaml new file mode 100644 index 000000000..17605307a --- /dev/null +++ b/config/config.yaml @@ -0,0 +1,97 @@ +# DO NOT MODIFY THIS FILE, create a new key.yaml, define OPENAI_API_KEY. +# The configuration of key.yaml has a higher priority and will not enter git + +#### if OpenAI +## The official OPENAI_API_BASE is https://api.openai.com/v1 +## If the official OPENAI_API_BASE is not available, we recommend using the [openai-forward](https://github.com/beidongjiedeguang/openai-forward). +## Or, you can configure OPENAI_PROXY to access official OPENAI_API_BASE. +OPENAI_API_BASE: "https://api.openai.com/v1" +#OPENAI_PROXY: "http://127.0.0.1:8118" +#OPENAI_API_KEY: "YOUR_API_KEY" # set the value to sk-xxx if you host the openai interface for open llm model +OPENAI_API_MODEL: "gpt-4" +MAX_TOKENS: 1500 +RPM: 10 + +#### if Spark +#SPARK_APPID : "YOUR_APPID" +#SPARK_API_SECRET : "YOUR_APISecret" +#SPARK_API_KEY : "YOUR_APIKey" +#DOMAIN : "generalv2" +#SPARK_URL : "ws://spark-api.xf-yun.com/v2.1/chat" + +#### if Anthropic +#Anthropic_API_KEY: "YOUR_API_KEY" + +#### if AZURE, check https://github.com/openai/openai-cookbook/blob/main/examples/azure/chat.ipynb +#### You can use ENGINE or DEPLOYMENT mode +#OPENAI_API_TYPE: "azure" +#OPENAI_API_BASE: "YOUR_AZURE_ENDPOINT" +#OPENAI_API_KEY: "YOUR_AZURE_API_KEY" +#OPENAI_API_VERSION: "YOUR_AZURE_API_VERSION" +#DEPLOYMENT_NAME: "YOUR_DEPLOYMENT_NAME" +#DEPLOYMENT_ID: "YOUR_DEPLOYMENT_ID" + +#### if zhipuai from `https://open.bigmodel.cn`. You can set here or export API_KEY="YOUR_API_KEY" +# ZHIPUAI_API_KEY: "YOUR_API_KEY" + +#### for Search + +## Supported values: serpapi/google/serper/ddg +#SEARCH_ENGINE: serpapi + +## Visit https://serpapi.com/ to get key. +#SERPAPI_API_KEY: "YOUR_API_KEY" + +## Visit https://console.cloud.google.com/apis/credentials to get key. +#GOOGLE_API_KEY: "YOUR_API_KEY" +## Visit https://programmablesearchengine.google.com/controlpanel/create to get id. +#GOOGLE_CSE_ID: "YOUR_CSE_ID" + +## Visit https://serper.dev/ to get key. +#SERPER_API_KEY: "YOUR_API_KEY" + +#### for web access + +## Supported values: playwright/selenium +#WEB_BROWSER_ENGINE: playwright + +## Supported values: chromium/firefox/webkit, visit https://playwright.dev/python/docs/api/class-browsertype +##PLAYWRIGHT_BROWSER_TYPE: chromium + +## Supported values: chrome/firefox/edge/ie, visit https://www.selenium.dev/documentation/webdriver/browsers/ +# SELENIUM_BROWSER_TYPE: chrome + +#### for TTS + +#AZURE_TTS_SUBSCRIPTION_KEY: "YOUR_API_KEY" +#AZURE_TTS_REGION: "eastus" + +#### for Stable Diffusion +## Use SD service, based on https://github.com/AUTOMATIC1111/stable-diffusion-webui +SD_URL: "YOUR_SD_URL" +SD_T2I_API: "/sdapi/v1/txt2img" + +#### for Execution +#LONG_TERM_MEMORY: false + +#### for Mermaid CLI +## If you installed mmdc (Mermaid CLI) only for metagpt then enable the following configuration. +#PUPPETEER_CONFIG: "./config/puppeteer-config.json" +#MMDC: "./node_modules/.bin/mmdc" + + +### for calc_usage +# CALC_USAGE: false + +### for Research +MODEL_FOR_RESEARCHER_SUMMARY: gpt-3.5-turbo +MODEL_FOR_RESEARCHER_REPORT: gpt-3.5-turbo-16k + +### choose the engine for mermaid conversion, +# default is nodejs, you can change it to playwright,pyppeteer or ink +# MERMAID_ENGINE: nodejs + +### browser path for pyppeteer engine, support Chrome, Chromium,MS Edge +#PYPPETEER_EXECUTABLE_PATH: "/usr/bin/google-chrome-stable" + +PROMPT_FORMAT: json #json or markdown \ No newline at end of file From 51ef51d516ee7674a56a5a35348f9e01d4b561ee Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 15 Dec 2023 10:27:28 +0800 Subject: [PATCH 174/637] revert --- metagpt/actions/write_analysis_code.py | 72 +++++++++++++------------- 1 file changed, 36 insertions(+), 36 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index cce36d8c9..34b605ea9 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -191,43 +191,43 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name = ML_MODULE_MAP[task_type] - # prompt = TOOL_USAGE_PROMPT.format( - # user_requirement=plan.goal, - # history_code=code_context, - # current_task=plan.current_task.instruction, - # column_info=column_info, - # special_prompt=special_prompt, - # code_steps=code_steps, - # module_name=module_name, - # tool_catalog=tool_catalog, - # ) + prompt = TOOL_USAGE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + special_prompt=special_prompt, + code_steps=code_steps, + module_name=module_name, + tool_catalog=tool_catalog, + ) - code_steps_ = eval(code_steps) - print(code_steps_) - - new_code = "" - tool_context = "" - for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): - prompt = TOOL_USAGE_PROMPT.format( - user_requirement=plan.goal, - history_code=code_context, - current_task=plan.current_task.instruction, - column_info=column_info, - special_prompt=special_prompt, - code_steps=step_instruction, - module_name=module_name, - tool_catalog=tool_catalog, - ) - - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - - rsp = await self.llm.aask_code(prompt, **tool_config) - logger.info(f"rsp is: {rsp}") - new_code = new_code + "\n\n" + rsp["code"] - code_context = code_context + "\n\n" + new_code - tool_context = tool_context + "\n\n" + prompt - context = [Message(content=tool_context, role="user")] - return context, new_code + # code_steps_ = eval(code_steps) + # print(code_steps_) + # + # new_code = "" + # tool_context = "" + # for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): + # prompt = TOOL_USAGE_PROMPT.format( + # user_requirement=plan.goal, + # history_code=code_context, + # current_task=plan.current_task.instruction, + # column_info=column_info, + # special_prompt=special_prompt, + # code_steps=step_instruction, + # module_name=module_name, + # tool_catalog=tool_catalog, + # ) + # + # tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + # + # rsp = await self.llm.aask_code(prompt, **tool_config) + # logger.info(f"rsp is: {rsp}") + # new_code = new_code + "\n\n" + rsp["code"] + # code_context = code_context + "\n\n" + new_code + # tool_context = tool_context + "\n\n" + prompt + # context = [Message(content=tool_context, role="user")] + # return context, new_code else: prompt = GENERATE_CODE_PROMPT.format( From 27b59a67daa91318f48615dea0e8bef722592d1e Mon Sep 17 00:00:00 2001 From: lidanyang Date: Mon, 18 Dec 2023 10:33:17 +0800 Subject: [PATCH 175/637] recover code --- .gitignore | 2 -- 1 file changed, 2 deletions(-) diff --git a/.gitignore b/.gitignore index f2ccde1d1..2f1250b93 100644 --- a/.gitignore +++ b/.gitignore @@ -148,8 +148,6 @@ allure-results .DS_Store .vscode -# Config -config/config.yaml log.txt docs/scripts/set_env.sh From d6566019b0c71e376b6aa27b85a9e54ee96e88ab Mon Sep 17 00:00:00 2001 From: lidanyang Date: Mon, 18 Dec 2023 10:34:38 +0800 Subject: [PATCH 176/637] recover code --- .gitignore | 1 - 1 file changed, 1 deletion(-) diff --git a/.gitignore b/.gitignore index 2f1250b93..9b679d48a 100644 --- a/.gitignore +++ b/.gitignore @@ -148,7 +148,6 @@ allure-results .DS_Store .vscode - log.txt docs/scripts/set_env.sh key.yaml From e67c679b1c13e6572a1934e2fdbb343ded8f81b9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 18 Dec 2023 15:55:05 +0800 Subject: [PATCH 177/637] update DEFAULT_SYSTEM_MSG. --- metagpt/actions/make_tools.py | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index 74037e900..590598cc3 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -13,11 +13,12 @@ from metagpt.actions.write_analysis_code import WriteCodeByGenerate class MakeTools(WriteCodeByGenerate): DEFAULT_SYSTEM_MSG = """Please Create a very General Function Code startswith `def` from any codes you got.\n **Notice: - 1. Reflect on whether it meets the requirements of a general function. + 1. Your code must contain a general function start with `def`. 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. 3. Use Google style for function annotations. 4. Write example code after `if __name__ == '__main__':`by using old varibales in old code, - and make sure it could be execute in the user's machine.** + and make sure it could be execute in the user's machine. + 5. Do not have missing package references.** """ def __init__(self, name: str = '', context: list[Message] = None, llm: LLM = None, workspace: str = None): @@ -50,11 +51,21 @@ class MakeTools(WriteCodeByGenerate): logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") saved_path.write_text(tool_code, encoding='utf-8') - @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) + # @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) async def run(self, code_message: List[Message | Dict], **kwargs) -> str: msgs = self.process_msg(code_message) logger.info(f"Ask: {msgs[-1]}") tool_code = await self.llm.aask_code(msgs, **kwargs) + max_tries, current_try = 3, 1 + func_name = self.parse_function_name(tool_code['code']) + while current_try < max_tries and func_name is None: + logger.warning(f"No function name found in code: \n{tool_code['code']}\n we will retry make tools.") + msgs.append({'role': 'assistant', 'content': 'We need a general function in above code,but not found function.'}) + tool_code = await self.llm.aask_code(msgs, **kwargs) + current_try += 1 + func_name = self.parse_function_name(tool_code['code']) + if func_name is not None: + break logger.info(f"Respond: Got {tool_code} from llm.") self.save(tool_code['code']) return tool_code["code"] From ea84fd34cd79153566e24a72247147c5509b2eef Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 18 Dec 2023 15:56:38 +0800 Subject: [PATCH 178/637] chore --- metagpt/actions/make_tools.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py index 590598cc3..c23e19edb 100644 --- a/metagpt/actions/make_tools.py +++ b/metagpt/actions/make_tools.py @@ -51,7 +51,7 @@ class MakeTools(WriteCodeByGenerate): logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") saved_path.write_text(tool_code, encoding='utf-8') - # @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) + @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) async def run(self, code_message: List[Message | Dict], **kwargs) -> str: msgs = self.process_msg(code_message) logger.info(f"Ask: {msgs[-1]}") From b5833397a4a12a46f41d02e6f2b44edadd48c3b7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 18 Dec 2023 20:18:20 +0800 Subject: [PATCH 179/637] feat: convert functions docstring schema to yaml --- metagpt/tools/functions/libs/udf/__init__.py | 77 +++++++++++++++++--- tests/metagpt/tools/functions/test_udf.py | 49 ++++++++++++- 2 files changed, 114 insertions(+), 12 deletions(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index 5bad9a3a4..0cada9545 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -1,9 +1,12 @@ import ast import os +import re +import yaml import inspect import importlib from pathlib import Path from typing import Dict, List +from metagpt.logs import logger def extract_function_signatures(file_path): @@ -12,6 +15,7 @@ def extract_function_signatures(file_path): tree = ast.parse(source_code) function_signatures = [] + function_returns = [] for node in ast.walk(tree): if isinstance(node, ast.FunctionDef): # 只提取用户自定义函数,排除内置函数 @@ -30,29 +34,84 @@ def extract_function_signatures(file_path): 'udf_path': f'from metagpt.tools.functions.libs.udf.{module_name} import {function_name}', 'udf_doc': inspect.getdoc(getattr(module, function_name))} function_signatures.append(function_schema) - - return function_signatures + # 获取函数返回变量名 + source_lines, _ = inspect.getsourcelines(getattr(module, function_name)) + for line in source_lines: + if line.strip().startswith("return "): + function_returns.append({ + 'udf_name': function_name, + 'udf_returns': [var.strip() for var in line.strip()[len("return "):].split(',')] + }) + break + return function_signatures, function_returns def get_function_signatures_in_folder(folder_path): python_files = [f for f in os.listdir(folder_path) if f.endswith('.py')] all_function_signatures = [] + all_function_returns = [] for file_name in python_files: file_path = os.path.join(folder_path, file_name) - function_signatures = extract_function_signatures(file_path) + function_signatures, function_returns = extract_function_signatures(file_path) all_function_signatures.extend(function_signatures) + all_function_returns.extend(function_returns) + return all_function_signatures, all_function_returns - return all_function_signatures + +# TODO: Create Tools Yaml Style Schema +def docstring_to_yaml(docstring: str, return_vars: List[str] = None): + logger.debug(f"\n\nFunction Docstring: \n{'-'*60}\n {docstring} \n\nFunction Returns: \n{'-'*60}\n{return_vars}\n") + if docstring is None: + return {} + # 匹配简介部分 + description_match = re.search(r'^(.*?)(?:Args:|Returns:|Raises:|$)', docstring, re.DOTALL) + description = description_match.group(1).strip() if description_match else "" + + # 匹配Args部分 + args_match = re.search(r'Args:\s*(.*?)(?:Returns:|Raises:|$)', docstring, re.DOTALL) + _args = args_match.group(1).strip() if args_match else "" + variable_pattern = re.compile(r'(\w+)\s*\((.*?)\):\s*(.*)') + params = variable_pattern.findall(_args) + if not params: + err_msg = f"No Args found in docstring as following, Please make sure it is google style\ + : \n\n{'-'*60}\n{docstring}\n{'-'*60}\n\n." + logger.error(err_msg) + raise ValueError(err_msg) + # 匹配Returns部分 + returns_match = re.search(r'Returns:\s*(.*?)(?:Raises:|$)', docstring, re.DOTALL) + returns = returns_match.group(1).strip() if returns_match else "" + return_pattern = re.compile(r'^(.*)\s*:\s*(.*)$') + # 添加返回值变量名 + return_vars = return_vars if isinstance(return_vars, list) else [return_vars] + returns = [(r, *r_desc) for r_desc, r in zip(return_pattern.findall(returns), return_vars)] + # 构建YAML字典 + yaml_data = { + 'description': description.strip('.').strip(), + 'parameters': { + 'properties': {param[0]: {'type': param[1], 'description': param[2]} for param in params}, + 'required': [param[0] for param in params] + }, + 'returns': {ret[0]: {'type': ret[1], 'description': ret[2]} for ret in returns} + } + return yaml_data + + +def extract_function_schema_yaml_in_folder(folder_path: str): + function_signatures, function_returns = get_function_signatures_in_folder(folder_path) + function_schema_yaml_data = {} + for func_docstring, func_returns in zip(function_signatures, function_returns): + if func_docstring['udf_doc']: + fun_yaml_data = docstring_to_yaml(func_docstring['udf_doc'], func_returns['udf_returns']) + fun_yaml_data.update({'type': 'function'}) + function_schema_yaml_data.update({func_returns['udf_name']: fun_yaml_data}) + return yaml.dump(function_schema_yaml_data, default_flow_style=False) folder_path = str(Path(__file__).parent.absolute()) -function_signatures = get_function_signatures_in_folder(folder_path) +function_signatures, function_returns = get_function_signatures_in_folder(folder_path) UDFS = [func for func in function_signatures if not func['udf_name'].startswith(('extract_function_signatures', 'get_function_signatures_in_folder'))] - -# TODO: Create Yaml style UDFS Schema -def udfs2yaml(udfs: List[Dict]) -> Dict: - pass +UDFS_YAML = extract_function_schema_yaml_in_folder(folder_path) diff --git a/tests/metagpt/tools/functions/test_udf.py b/tests/metagpt/tools/functions/test_udf.py index b0c921180..89897e548 100644 --- a/tests/metagpt/tools/functions/test_udf.py +++ b/tests/metagpt/tools/functions/test_udf.py @@ -1,9 +1,52 @@ -from metagpt.tools.functions.libs.udf import UDFS +import pytest +import yaml + +from metagpt.tools.functions.libs.udf import UDFS, docstring_to_yaml, UDFS_YAML from metagpt.logs import logger def test_udfs(): assert len(UDFS) > 0 - assert 'name' in UDFS[0] - assert 'doc' in UDFS[0] + assert 'udf_name' in UDFS[0] + assert 'udf_doc' in UDFS[0] logger.info(UDFS) + + +def test_docstring2yaml(): + docstring = """Calculate the duration in hours between two datetime columns. + + Args: + dataframe (pd.DataFrame): The dataframe containing the datetime columns. + + Returns: + pd.DataFrame: The dataframe with an additional column 'duration_hour' added. + """ + + yaml_result = docstring_to_yaml(docstring, return_vars='dataframe') + assert 'parameters' in yaml_result + assert 'properties' in yaml_result['parameters'] + assert 'dataframe' in yaml_result['parameters']['properties'] + + +def test_docstring2yaml_error(): + docstring = """Calculate the duration in hours between two datetime columns. + args: + dataframe (pd.DataFrame): The dataframe containing the datetime columns. + returns: + pd.DataFrame: The dataframe with an additional column 'duration_hour' added. + """ + with pytest.raises(ValueError) as exc_info: + docstring_to_yaml(docstring, return_vars='dataframe') + assert "No Args found" in exc_info + + +def test_UDFS_YAML(): + assert len(UDFS_YAML) > 0 + logger.info(f"\n\n{UDFS_YAML}") + function_schema = yaml.load(UDFS_YAML, Loader=yaml.FullLoader) + assert 'description' in function_schema[list(function_schema.keys())[0]] + assert 'type' in function_schema[list(function_schema.keys())[0]] + assert 'parameters' in function_schema[list(function_schema.keys())[0]] + assert 'properties' in function_schema[list(function_schema.keys())[0]]['parameters'] + assert 'required' in function_schema[list(function_schema.keys())[0]]['parameters'] + assert 'returns' in function_schema[list(function_schema.keys())[0]] From 1a2b4f1b3b08c8f046a179864ab5e6d5f57086df Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 18 Dec 2023 20:40:11 +0800 Subject: [PATCH 180/637] update make_tools. --- metagpt/actions/make_tools.py | 71 ------------------------ tests/metagpt/actions/test_make_tools.py | 2 +- 2 files changed, 1 insertion(+), 72 deletions(-) delete mode 100644 metagpt/actions/make_tools.py diff --git a/metagpt/actions/make_tools.py b/metagpt/actions/make_tools.py deleted file mode 100644 index c23e19edb..000000000 --- a/metagpt/actions/make_tools.py +++ /dev/null @@ -1,71 +0,0 @@ -from typing import List, Dict -from pathlib import Path -import re - -from tenacity import retry, stop_after_attempt, wait_fixed - -from metagpt.llm import LLM -from metagpt.logs import logger -from metagpt.schema import Message -from metagpt.actions.write_analysis_code import WriteCodeByGenerate - - -class MakeTools(WriteCodeByGenerate): - DEFAULT_SYSTEM_MSG = """Please Create a very General Function Code startswith `def` from any codes you got.\n - **Notice: - 1. Your code must contain a general function start with `def`. - 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. - 3. Use Google style for function annotations. - 4. Write example code after `if __name__ == '__main__':`by using old varibales in old code, - and make sure it could be execute in the user's machine. - 5. Do not have missing package references.** - """ - - def __init__(self, name: str = '', context: list[Message] = None, llm: LLM = None, workspace: str = None): - """ - :param str name: name, defaults to '' - :param list[Message] context: context, defaults to None - :param LLM llm: llm, defaults to None - :param str workspace: tools code saved file path dir, defaults to None - """ - super().__init__(name, context, llm) - self.workspace = workspace or str(Path(__file__).parents[1].joinpath("./tools/functions/libs/udf")) - self.file_suffix: str = '.py' - - def parse_function_name(self, function_code: str) -> str: - # 定义正则表达式模式 - pattern = r'\bdef\s+([a-zA-Z_]\w*)\s*\(' - # 在代码中搜索匹配的模式 - match = re.search(pattern, function_code) - # 如果找到匹配项,则返回匹配的函数名;否则返回None - if match: - return match.group(1) - else: - return None - - def save(self, tool_code: str) -> None: - func_name = self.parse_function_name(tool_code) - if func_name is None: - raise ValueError(f"No function name found in {tool_code}") - saved_path = Path(self.workspace).joinpath(func_name+self.file_suffix) - logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") - saved_path.write_text(tool_code, encoding='utf-8') - - @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) - async def run(self, code_message: List[Message | Dict], **kwargs) -> str: - msgs = self.process_msg(code_message) - logger.info(f"Ask: {msgs[-1]}") - tool_code = await self.llm.aask_code(msgs, **kwargs) - max_tries, current_try = 3, 1 - func_name = self.parse_function_name(tool_code['code']) - while current_try < max_tries and func_name is None: - logger.warning(f"No function name found in code: \n{tool_code['code']}\n we will retry make tools.") - msgs.append({'role': 'assistant', 'content': 'We need a general function in above code,but not found function.'}) - tool_code = await self.llm.aask_code(msgs, **kwargs) - current_try += 1 - func_name = self.parse_function_name(tool_code['code']) - if func_name is not None: - break - logger.info(f"Respond: Got {tool_code} from llm.") - self.save(tool_code['code']) - return tool_code["code"] diff --git a/tests/metagpt/actions/test_make_tools.py b/tests/metagpt/actions/test_make_tools.py index 264599439..cf7986b82 100644 --- a/tests/metagpt/actions/test_make_tools.py +++ b/tests/metagpt/actions/test_make_tools.py @@ -1,7 +1,7 @@ import pytest from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.make_tools import MakeTools +from metagpt.actions.write_analysis_code import MakeTools from metagpt.logs import logger From b18b1c366ead8cc4e2b950145d56d4885b1e6060 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 18 Dec 2023 21:57:45 +0800 Subject: [PATCH 181/637] update UDFS. --- metagpt/tools/functions/libs/udf/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index 0cada9545..8c74bbbe3 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -112,6 +112,6 @@ folder_path = str(Path(__file__).parent.absolute()) function_signatures, function_returns = get_function_signatures_in_folder(folder_path) UDFS = [func for func in function_signatures - if not func['udf_name'].startswith(('extract_function_signatures', 'get_function_signatures_in_folder'))] + if not func['udf_name'].startswith(('extract_function_signatures', 'get_function_signatures_in_folder', 'docstring_to_yaml'))] UDFS_YAML = extract_function_schema_yaml_in_folder(folder_path) From a71b75a8a928744f8bf1742e56fa51e56365314e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 18 Dec 2023 22:10:28 +0800 Subject: [PATCH 182/637] feat: MakeTools, WriteCodeWithUDFs. --- metagpt/actions/write_analysis_code.py | 104 +++++++++++++++++++++++-- 1 file changed, 96 insertions(+), 8 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 663f76b7b..c41e0fc5a 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -5,6 +5,10 @@ @File : write_code_v2.py """ from typing import Dict, List, Union, Tuple +from tenacity import retry, stop_after_attempt, wait_fixed +from pathlib import Path +import re +import json from metagpt.actions import Action from metagpt.llm import LLM @@ -86,7 +90,6 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): self, context: [List[Message]], plan: Plan = None, - code_steps: str = "", system_msg: str = None, **kwargs, ) -> str: @@ -206,25 +209,110 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): return rsp["code"] +class MakeTools(WriteCodeByGenerate): + DEFAULT_SYSTEM_MSG = """Please Create a very General Function Code startswith `def` from any codes you got.\n + **Notice: + 1. Your code must contain a general function start with `def`. + 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. + 3. Use Google style for function annotations. + 4. Write example code after `if __name__ == '__main__':`by using old varibales in old code, + and make sure it could be execute in the user's machine. + 5. Do not have missing package references.** + """ + + def __init__(self, name: str = '', context: list[Message] = None, llm: LLM = None, workspace: str = None): + """ + :param str name: name, defaults to '' + :param list[Message] context: context, defaults to None + :param LLM llm: llm, defaults to None + :param str workspace: tools code saved file path dir, defaults to None + """ + super().__init__(name, context, llm) + self.workspace = workspace or str(Path(__file__).parents[1].joinpath("./tools/functions/libs/udf")) + self.file_suffix: str = '.py' + + def parse_function_name(self, function_code: str) -> str: + # 定义正则表达式模式 + pattern = r'\bdef\s+([a-zA-Z_]\w*)\s*\(' + # 在代码中搜索匹配的模式 + match = re.search(pattern, function_code) + # 如果找到匹配项,则返回匹配的函数名;否则返回None + if match: + return match.group(1) + else: + return None + + def save(self, tool_code: str) -> None: + func_name = self.parse_function_name(tool_code) + if func_name is None: + raise ValueError(f"No function name found in {tool_code}") + saved_path = Path(self.workspace).joinpath(func_name+self.file_suffix) + logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") + saved_path.write_text(tool_code, encoding='utf-8') + + @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) + async def run(self, code_message: List[Message | Dict], **kwargs) -> str: + msgs = self.process_msg(code_message) + logger.info(f"\n\nAsk to Make tools:\n{'-'*60}\n {msgs[-1]}") + tool_code = await self.llm.aask_code(msgs, **kwargs) + max_tries, current_try = 3, 1 + func_name = self.parse_function_name(tool_code['code']) + while current_try < max_tries and func_name is None: + logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") + logger.warning(f"No function name found in code, we will retry make tools. \n\n{tool_code['code']}\n") + msgs.append({'role': 'assistant', 'content': 'We need a general function in above code,but not found function.'}) + tool_code = await self.llm.aask_code(msgs, **kwargs) + current_try += 1 + func_name = self.parse_function_name(tool_code['code']) + if func_name is not None: + break + self.save(tool_code['code']) + return tool_code["code"] + + class WriteCodeWithUDFs(WriteCodeByGenerate): """Write code with user defined function.""" from metagpt.tools.functions.libs.udf import UDFS - DEFAULT_SYSTEM_MSG = f"""Please remember these functions, you will use these functions to write code:\n - {UDFS}, **Notice: 1. if no right udf for user requirement, please send `No udf found`** + UDFS_DEFAULT_SYSTEM_MSG = f"""Please remember these functions, you will use these functions to write code:\n + {UDFS}, **Notice: 1. if no udf meets user requirement, please send `No udf found`. 2.Only use function code provied to you. + 3. Dont generate code from scratch.** """ async def aask_code_and_text(self, context: List[Dict], **kwargs) -> Tuple[str]: rsp = await self.llm.acompletion(context, **kwargs) rsp_content = self.llm.get_choice_text(rsp) code = CodeParser.parse_code(None, rsp_content) - if code.startswith('No udf found') or rsp_content.startswith('No udf found'): + if 'No udf found' in code or 'No udf found' in rsp_content: rsp_content = 'No udf found' code = 'No udf found' return code, rsp_content - async def run(self, context: List[Message], plan: Plan = None, task_guide: str = "", **kwargs) -> str: - prompt = self.process_msg(context) - logger.info(prompt[-1]) - code, _ = await self.aask_code_and_text(prompt, **kwargs) + async def run(self, context: List[Message], plan: Plan = None, **kwargs) -> str: + from metagpt.tools.functions.libs.udf import UDFS + if len(UDFS) > 0: + # Write code from user defined function. + prompt = self.process_msg(context, self.UDFS_DEFAULT_SYSTEM_MSG) + logger.info(prompt[-1]) + try: + logger.info("Local user defined function as following:") + logger.info(json.dumps(UDFS, indent=4, ensure_ascii=False)) + except Exception: + from pprint import pprint + pprint(UDFS) + logger.info('Writing code from user defined function by LLM...') + code, _ = await self.aask_code_and_text(prompt, **kwargs) + logger.info(f"Writing code from user defined function: \n{'-'*50}\n {code}") + if code != 'No udf found': + return code + logger.warning("No udf found, we will write code from scratch by LLM.") + # Writing code from scratch. + logger.warning("Writing code from scratch by LLM.") + code = await super().run(context, plan, self.DEFAULT_SYSTEM_MSG, **kwargs) + logger.info(f"Code Writing code from scratch by LLM is :\n{'-'*60}\n {code}") + # Make tools for above code. + logger.info("Make tools for above code.") + make_tools = MakeTools() + tool_code = await make_tools.run(code) + make_tools.save(tool_code) return code From 79787e8129119b9b4a848a54c26e4215225b9798 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 18 Dec 2023 22:15:32 +0800 Subject: [PATCH 183/637] feat: add make tools. --- metagpt/roles/ml_engineer.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index e7fe38ff4..b039c61e7 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -9,7 +9,7 @@ from metagpt.schema import Message, Plan from metagpt.memory import Memory from metagpt.logs import logger from metagpt.actions.write_plan import WritePlan, update_plan_from_rsp, precheck_update_plan_from_rsp -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools, MakeTools from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis, Reflect, ReviewConst from metagpt.actions.execute_code import ExecutePyCode from metagpt.roles.kaggle_manager import DownloadData, SubmitResult @@ -126,6 +126,10 @@ class MLEngineer(Role): context=context, plan=self.plan, code_steps=code_steps, temperature=0.0 ) cause_by = WriteCodeByGenerate + # make and save tools. + make_tools = MakeTools() + tool_code = await make_tools.run(code) + make_tools.save(tool_code) else: code = await WriteCodeWithTools().run( context=context, plan=self.plan, code_steps=code_steps, data_desc="" From 52052c82447fcb7d92108723b52274f8788f52c3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 18 Dec 2023 22:30:14 +0800 Subject: [PATCH 184/637] update make tools. --- metagpt/actions/write_analysis_code.py | 62 ++++++++++++++++++++++++++ 1 file changed, 62 insertions(+) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 839184cdc..4194bafc9 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -220,3 +220,65 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): rsp = await self.llm.aask_code(prompt, **tool_config) context = [Message(content=prompt, role="user")] return context, rsp["code"] + + +class MakeTools(WriteCodeByGenerate): + DEFAULT_SYSTEM_MSG = """Please Create a very General Function Code startswith `def` from any codes you got.\n + **Notice: + 1. Your code must contain a general function start with `def`. + 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. + 3. Use Google style for function annotations. + 4. Write example code after `if __name__ == '__main__':`by using old varibales in old code, + and make sure it could be execute in the user's machine. + 5. Dont have missing package references.** + """ + + def __init__(self, name: str = '', context: list[Message] = None, llm: LLM = None, workspace: str = None): + """ + :param str name: name, defaults to '' + :param list[Message] context: context, defaults to None + :param LLM llm: llm, defaults to None + :param str workspace: tools code saved file path dir, defaults to None + """ + super().__init__(name, context, llm) + self.workspace = workspace or str(Path(__file__).parents[1].joinpath("./tools/functions/libs/udf")) + self.file_suffix: str = '.py' + + def parse_function_name(self, function_code: str) -> str: + # 定义正则表达式模式 + pattern = r'\bdef\s+([a-zA-Z_]\w*)\s*\(' + # 在代码中搜索匹配的模式 + match = re.search(pattern, function_code) + # 如果找到匹配项,则返回匹配的函数名;否则返回None + if match: + return match.group(1) + else: + return None + + def save(self, tool_code: str) -> None: + func_name = self.parse_function_name(tool_code) + if func_name is None: + raise ValueError(f"No function name found in {tool_code}") + saved_path = Path(self.workspace).joinpath(func_name+self.file_suffix) + logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") + saved_path.write_text(tool_code, encoding='utf-8') + + @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) + async def run(self, code_message: List[Message | Dict], **kwargs) -> str: + msgs = self.process_msg(code_message) + logger.info(f"\n\nAsk to Make tools:\n{'-'*60}\n {msgs[-1]}") + tool_code = await self.llm.aask_code(msgs, **kwargs) + max_tries, current_try = 3, 1 + func_name = self.parse_function_name(tool_code['code']) + while current_try < max_tries and func_name is None: + logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") + logger.warning(f"No function name found in code, we will retry make tools. \n\n{tool_code['code']}\n") + msgs.append({'role': 'assistant', 'content': 'We need a general function in above code,but not found function.'}) + tool_code = await self.llm.aask_code(msgs, **kwargs) + current_try += 1 + func_name = self.parse_function_name(tool_code['code']) + if func_name is not None: + break + logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") + self.save(tool_code['code']) + return tool_code["code"] From 87821fc6cca7181e32a4d0e740cff531a3cb7cd5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 18 Dec 2023 22:33:58 +0800 Subject: [PATCH 185/637] update make tools. --- metagpt/roles/ml_engineer.py | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 28ff9fb3d..1361c566f 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -8,7 +8,7 @@ from metagpt.actions import Action from metagpt.actions.debug_code import DebugCode from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis, Reflect, ReviewConst -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools, MakeTools from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.actions.write_plan import WritePlan from metagpt.actions.write_plan import update_plan_from_rsp, precheck_update_plan_from_rsp @@ -48,6 +48,7 @@ class MLEngineer(Role): self.plan = Plan(goal=goal) self.use_tools = False + self.make_tools = True self.use_code_steps = False self.execute_code = ExecutePyCode() self.auto_run = auto_run @@ -173,10 +174,11 @@ class MLEngineer(Role): ) debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] cause_by = WriteCodeByGenerate - # make and save tools. - make_tools = MakeTools() - tool_code = await make_tools.run(code) - make_tools.save(tool_code) + if self.make_tools: + # make and save tools. + make_tools = MakeTools() + tool_code = await make_tools.run(code) + make_tools.save(tool_code) else: logger.info("Write code with tools") schema_path = PROJECT_ROOT / "metagpt/tools/functions/schemas" From 4cb2028c7240f8be607a9b9f57cdfb47bd197117 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 10:24:57 +0800 Subject: [PATCH 186/637] update for make tools test. --- metagpt/roles/ml_engineer.py | 25 ++++++++++++++++--------- 1 file changed, 16 insertions(+), 9 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 1361c566f..75c403226 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -48,7 +48,8 @@ class MLEngineer(Role): self.plan = Plan(goal=goal) self.use_tools = False - self.make_tools = True + self.make_udfs = False + self.use_udfs = False self.use_code_steps = False self.execute_code = ExecutePyCode() self.auto_run = auto_run @@ -168,14 +169,19 @@ class MLEngineer(Role): logger.info(f"new code \n{code}") cause_by = DebugCode elif not self.use_tools or self.plan.current_task.task_type == "other": - logger.info("Write code with pure generation") - code = await WriteCodeByGenerate().run( - context=context, plan=self.plan, temperature=0.0 - ) - debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] - cause_by = WriteCodeByGenerate - if self.make_tools: - # make and save tools. + if self.use_udfs: + # use user-defined function tools. + pass + else: + logger.info("Write code with pure generation") + code = await WriteCodeByGenerate().run( + context=context, plan=self.plan, temperature=0.0 + ) + debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] + cause_by = WriteCodeByGenerate + + if self.make_udfs: + # make and save user-defined function tools. make_tools = MakeTools() tool_code = await make_tools.run(code) make_tools.save(tool_code) @@ -291,6 +297,7 @@ if __name__ == "__main__": async def main(requirement: str = requirement, auto_run: bool = True): role = MLEngineer(goal=requirement, auto_run=auto_run) + role.make_udfs = True await role.run(requirement) fire.Fire(main) From d9c814420b5e31430e7143d4b430404c4ce8f63c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 11:21:51 +0800 Subject: [PATCH 187/637] fix: no args error. --- metagpt/tools/functions/libs/udf/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index 8c74bbbe3..5596cd37a 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -77,7 +77,7 @@ def docstring_to_yaml(docstring: str, return_vars: List[str] = None): err_msg = f"No Args found in docstring as following, Please make sure it is google style\ : \n\n{'-'*60}\n{docstring}\n{'-'*60}\n\n." logger.error(err_msg) - raise ValueError(err_msg) + params = (('', '', ''),) # 匹配Returns部分 returns_match = re.search(r'Returns:\s*(.*?)(?:Raises:|$)', docstring, re.DOTALL) returns = returns_match.group(1).strip() if returns_match else "" From c1a3a12c9250a582f7348067a615c52c85fd6c2b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 11:27:26 +0800 Subject: [PATCH 188/637] update udf test for function schema. --- tests/metagpt/tools/functions/test_udf.py | 20 ++++++++------------ 1 file changed, 8 insertions(+), 12 deletions(-) diff --git a/tests/metagpt/tools/functions/test_udf.py b/tests/metagpt/tools/functions/test_udf.py index 89897e548..111ec532a 100644 --- a/tests/metagpt/tools/functions/test_udf.py +++ b/tests/metagpt/tools/functions/test_udf.py @@ -28,18 +28,6 @@ def test_docstring2yaml(): assert 'dataframe' in yaml_result['parameters']['properties'] -def test_docstring2yaml_error(): - docstring = """Calculate the duration in hours between two datetime columns. - args: - dataframe (pd.DataFrame): The dataframe containing the datetime columns. - returns: - pd.DataFrame: The dataframe with an additional column 'duration_hour' added. - """ - with pytest.raises(ValueError) as exc_info: - docstring_to_yaml(docstring, return_vars='dataframe') - assert "No Args found" in exc_info - - def test_UDFS_YAML(): assert len(UDFS_YAML) > 0 logger.info(f"\n\n{UDFS_YAML}") @@ -50,3 +38,11 @@ def test_UDFS_YAML(): assert 'properties' in function_schema[list(function_schema.keys())[0]]['parameters'] assert 'required' in function_schema[list(function_schema.keys())[0]]['parameters'] assert 'returns' in function_schema[list(function_schema.keys())[0]] + # 指定要保存的文件路径 + file_path = './tests/data/function_schema.yaml' + + # 使用 PyYAML 将字典保存为 YAML 文件 + with open(file_path, 'w') as file: + yaml.dump(function_schema, file, default_flow_style=False) + + print(f'Data has been saved to {file_path}') From 4de104ef8f3bff4a486e058354c9038a378f025b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 11:32:25 +0800 Subject: [PATCH 189/637] update parameters for None. --- metagpt/tools/functions/libs/udf/__init__.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index 5596cd37a..3c4e72d8b 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -77,7 +77,7 @@ def docstring_to_yaml(docstring: str, return_vars: List[str] = None): err_msg = f"No Args found in docstring as following, Please make sure it is google style\ : \n\n{'-'*60}\n{docstring}\n{'-'*60}\n\n." logger.error(err_msg) - params = (('', '', ''),) + params = ((None, None, None),) # 匹配Returns部分 returns_match = re.search(r'Returns:\s*(.*?)(?:Raises:|$)', docstring, re.DOTALL) returns = returns_match.group(1).strip() if returns_match else "" @@ -89,8 +89,8 @@ def docstring_to_yaml(docstring: str, return_vars: List[str] = None): yaml_data = { 'description': description.strip('.').strip(), 'parameters': { - 'properties': {param[0]: {'type': param[1], 'description': param[2]} for param in params}, - 'required': [param[0] for param in params] + 'properties': {param[0]: {'type': param[1], 'description': param[2]} for param in params if param[0] is not None}, + 'required': [param[0] for param in params if param[0] is not None] }, 'returns': {ret[0]: {'type': ret[1], 'description': ret[2]} for ret in returns} } From 0daf7ea4e3bfd8af5de11788d4fa5e295b98cf5e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 11:34:06 +0800 Subject: [PATCH 190/637] chore. --- metagpt/tools/functions/libs/udf/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index 3c4e72d8b..add03f376 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -76,7 +76,7 @@ def docstring_to_yaml(docstring: str, return_vars: List[str] = None): if not params: err_msg = f"No Args found in docstring as following, Please make sure it is google style\ : \n\n{'-'*60}\n{docstring}\n{'-'*60}\n\n." - logger.error(err_msg) + logger.warning(err_msg) params = ((None, None, None),) # 匹配Returns部分 returns_match = re.search(r'Returns:\s*(.*?)(?:Raises:|$)', docstring, re.DOTALL) From 5db006334219a5d3d858d6c2ff9ab27461746765 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 19 Dec 2023 12:57:45 +0800 Subject: [PATCH 191/637] Add ml_engineer_simple.py for ablation experiments --- metagpt/roles/ml_engineer_simple.py | 148 ++++++++++++++++++++++++++++ 1 file changed, 148 insertions(+) create mode 100644 metagpt/roles/ml_engineer_simple.py diff --git a/metagpt/roles/ml_engineer_simple.py b/metagpt/roles/ml_engineer_simple.py new file mode 100644 index 000000000..e66770211 --- /dev/null +++ b/metagpt/roles/ml_engineer_simple.py @@ -0,0 +1,148 @@ +import re +from typing import List +import json +from datetime import datetime + +import fire + +from metagpt.roles import Role +from metagpt.schema import Message +from metagpt.memory import Memory +from metagpt.logs import logger +from metagpt.actions.write_analysis_code import WriteCodeByGenerate +from metagpt.actions.ml_da_action import AskReview, ReviewConst +from metagpt.actions.execute_code import ExecutePyCode +from metagpt.roles.kaggle_manager import DownloadData +from metagpt.utils.save_code import save_code_file + +STRUCTURAL_CONTEXT_SIMPLE = """ +## User Requirement +{user_requirement} +## Data Description +{data_desc} +""" + +JUDGE_PROMPT_TEMPLATE = """ +# User Requirement +{user_requirement} +----- +# Context +{context} +----- +# State +Output "Ture" or "False". Judging from the code perspective, whether the user's needs have been completely fulfilled. +===== +# Finally output State, Thought and Next Action separately in one sentence +State: +Thought: +Next Action: +""" + + +class MLEngineerSimple(Role): + def __init__( + self, name="ABC", profile="MLEngineerSimple", goal="", auto_run: bool = False + ): + super().__init__(name=name, profile=profile, goal=goal) + self._set_react_mode(react_mode="react") + self._watch([DownloadData]) + self._init_actions([WriteCodeByGenerate, ExecutePyCode]) + + self.goal = goal + self.data_desc = "" + self.use_tools = False + self.use_code_steps = False + self.execute_code = ExecutePyCode() + self.auto_run = auto_run + + # memory for working on each task, discarded each time a task is done + self.working_memory = Memory() + + async def _act(self): + memories = self.get_memories() + if memories: + latest_event = memories[-1].cause_by + if latest_event == DownloadData: + self.data_desc = memories[-1].content + + await self._act_no_plan() + + # save code using datetime.now or keywords related to the goal of your project (plan.goal). + project_record = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") + save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") + + async def _act_no_plan(self, max_retry: int = 20): + counter = 0 + state = False + while not state and counter < max_retry: + context = self.get_useful_memories() + print(f"memories数量:{len(context)}") + # print("===\n" +str(context) + "\n===") + code = await WriteCodeByGenerate().run( + context=context, temperature=0.0 + ) + cause_by = WriteCodeByGenerate + self.working_memory.add( + Message(content=code, role="assistant", cause_by=cause_by) + ) + + result, success = await self.execute_code.run(code) + print(result) + self.working_memory.add( + Message(content=result, role="user", cause_by=ExecutePyCode) + ) + + if "!pip" in code: + success = False + + counter += 1 + + if not success and counter >= max_retry: + logger.info("coding failed!") + review, _ = await self._ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) + if ReviewConst.CHANGE_WORD[0] in review: + counter = 0 # redo the task again with help of human suggestions + + completed_plan_memory = self.get_useful_memories() # completed plan as a outcome + self._rc.memory.add(completed_plan_memory[0]) # add to persistent memory + prompt = JUDGE_PROMPT_TEMPLATE.format(user_requirement=self.goal, context=completed_plan_memory) + rsp = await self._llm.aask(prompt) + self.working_memory.add( + Message(content=rsp, role="system") + ) + + matches = re.findall(r'\b(True|False)\b', rsp) + state = False if 'False' in matches else True + + async def _ask_review(self, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): + auto_run = auto_run or self.auto_run + if not auto_run: + context = self.get_useful_memories() + review, confirmed = await AskReview().run(context=context[-5:], trigger=trigger) + if not confirmed: + self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) + return review, confirmed + return "", True + + def get_useful_memories(self) -> List[Message]: + """find useful memories only to reduce context length and improve performance""" + user_requirement = self.goal + context = STRUCTURAL_CONTEXT_SIMPLE.format( + user_requirement=user_requirement, data_desc=self.data_desc + ) + context_msg = [Message(content=context, role="user")] + + return context_msg + self.get_working_memories() + + def get_working_memories(self, num=6) -> List[Message]: + return self.working_memory.get(num) # 默认为6 + + +if __name__ == "__main__": + requirement = "Run data analysis on sklearn Iris dataset, include a plot" + + async def main(requirement: str = requirement, auto_run: bool = True): + role = MLEngineerSimple(goal=requirement, auto_run=auto_run) + await role.run(requirement) + + fire.Fire(main) From 7ddca9e99564e6d102d4d8b443effbbddedb774c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 14:36:33 +0800 Subject: [PATCH 192/637] update MakeTools DEFAULT_SYSTEM_MSG. --- metagpt/actions/write_analysis_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 4194bafc9..0a1d74263 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -223,7 +223,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): class MakeTools(WriteCodeByGenerate): - DEFAULT_SYSTEM_MSG = """Please Create a very General Function Code startswith `def` from any codes you got.\n + DEFAULT_SYSTEM_MSG = """Convert any codes provied for you to a very General Function Code startswith `def`.\n **Notice: 1. Your code must contain a general function start with `def`. 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. From fdf16f55352102d002223af6ee4d054622be0e79 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 14:38:08 +0800 Subject: [PATCH 193/637] add code_prompt for make tools. --- metagpt/roles/ml_engineer.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 75c403226..96e21c8c8 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -180,10 +180,12 @@ class MLEngineer(Role): debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] cause_by = WriteCodeByGenerate - if self.make_udfs: + if self.make_udfs and len(code.split('\n')) > 2: # make and save user-defined function tools. make_tools = MakeTools() - tool_code = await make_tools.run(code) + code_prompt = f"The following code is about {self.plan.current_task.instruction},\ + convert it to be a General Function, {code}" + tool_code = await make_tools.run(code_prompt) make_tools.save(tool_code) else: logger.info("Write code with tools") From a4ba5660b82a528ae876c30336623e9f33afdf24 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 16:30:38 +0800 Subject: [PATCH 194/637] convert UDFS_YAML to dict. --- metagpt/tools/functions/libs/udf/__init__.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index add03f376..ad36b2817 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -114,4 +114,5 @@ function_signatures, function_returns = get_function_signatures_in_folder(folder UDFS = [func for func in function_signatures if not func['udf_name'].startswith(('extract_function_signatures', 'get_function_signatures_in_folder', 'docstring_to_yaml'))] -UDFS_YAML = extract_function_schema_yaml_in_folder(folder_path) +UDFS_YAML_STR: str = extract_function_schema_yaml_in_folder(folder_path) +UDFS_YAML: dict = yaml.load(UDFS_YAML_STR, Loader=yaml.FullLoader) From 3bb445b925ba5901bd0e5d9e4e1339c3c60c13dd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 16:50:37 +0800 Subject: [PATCH 195/637] fix: no returns function tools. --- metagpt/tools/functions/libs/udf/__init__.py | 14 ++++++++++---- tests/metagpt/tools/functions/test_udf.py | 5 +++-- 2 files changed, 13 insertions(+), 6 deletions(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index ad36b2817..245288de2 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -43,11 +43,18 @@ def extract_function_signatures(file_path): 'udf_returns': [var.strip() for var in line.strip()[len("return "):].split(',')] }) break + + # 没有返回值的函数 + if not function_returns or function_returns[-1]['udf_name'] != function_name: + function_returns.append({ + 'udf_name': function_name, + 'udf_returns': [None] + }) return function_signatures, function_returns def get_function_signatures_in_folder(folder_path): - python_files = [f for f in os.listdir(folder_path) if f.endswith('.py')] + python_files = [f for f in os.listdir(folder_path) if f.endswith('.py') and f != '__init__.py'] all_function_signatures = [] all_function_returns = [] @@ -59,7 +66,7 @@ def get_function_signatures_in_folder(folder_path): return all_function_signatures, all_function_returns -# TODO: Create Tools Yaml Style Schema +# Create Tools Yaml Style Schema def docstring_to_yaml(docstring: str, return_vars: List[str] = None): logger.debug(f"\n\nFunction Docstring: \n{'-'*60}\n {docstring} \n\nFunction Returns: \n{'-'*60}\n{return_vars}\n") if docstring is None: @@ -111,8 +118,7 @@ def extract_function_schema_yaml_in_folder(folder_path: str): folder_path = str(Path(__file__).parent.absolute()) function_signatures, function_returns = get_function_signatures_in_folder(folder_path) -UDFS = [func for func in function_signatures - if not func['udf_name'].startswith(('extract_function_signatures', 'get_function_signatures_in_folder', 'docstring_to_yaml'))] +UDFS = [func for func in function_signatures] UDFS_YAML_STR: str = extract_function_schema_yaml_in_folder(folder_path) UDFS_YAML: dict = yaml.load(UDFS_YAML_STR, Loader=yaml.FullLoader) diff --git a/tests/metagpt/tools/functions/test_udf.py b/tests/metagpt/tools/functions/test_udf.py index 111ec532a..b4060ad13 100644 --- a/tests/metagpt/tools/functions/test_udf.py +++ b/tests/metagpt/tools/functions/test_udf.py @@ -1,5 +1,6 @@ import pytest import yaml +import json from metagpt.tools.functions.libs.udf import UDFS, docstring_to_yaml, UDFS_YAML from metagpt.logs import logger @@ -30,8 +31,8 @@ def test_docstring2yaml(): def test_UDFS_YAML(): assert len(UDFS_YAML) > 0 - logger.info(f"\n\n{UDFS_YAML}") - function_schema = yaml.load(UDFS_YAML, Loader=yaml.FullLoader) + logger.info(f"\n\n{json.dumps(UDFS_YAML, indent=2, ensure_ascii=False)}") + function_schema = UDFS_YAML assert 'description' in function_schema[list(function_schema.keys())[0]] assert 'type' in function_schema[list(function_schema.keys())[0]] assert 'parameters' in function_schema[list(function_schema.keys())[0]] From 6895e74d3ef0e44ce04a4c2195b96da1d7920edb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 17:01:55 +0800 Subject: [PATCH 196/637] update parse No Args function. --- metagpt/tools/functions/libs/udf/__init__.py | 3 --- 1 file changed, 3 deletions(-) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index 245288de2..b74ae2ab9 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -81,9 +81,6 @@ def docstring_to_yaml(docstring: str, return_vars: List[str] = None): variable_pattern = re.compile(r'(\w+)\s*\((.*?)\):\s*(.*)') params = variable_pattern.findall(_args) if not params: - err_msg = f"No Args found in docstring as following, Please make sure it is google style\ - : \n\n{'-'*60}\n{docstring}\n{'-'*60}\n\n." - logger.warning(err_msg) params = ((None, None, None),) # 匹配Returns部分 returns_match = re.search(r'Returns:\s*(.*?)(?:Raises:|$)', docstring, re.DOTALL) From 52b8ba84d32d6d42b6dde75b772d2dd68195c9ff Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 17:57:16 +0800 Subject: [PATCH 197/637] update globals with function tools. --- metagpt/tools/functions/libs/udf/__init__.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index b74ae2ab9..5d9c35b27 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -29,6 +29,8 @@ def extract_function_signatures(file_path): # 导入函数 module_name = Path(file_path).parts[-1][:-len(Path(file_path).suffix)] module = importlib.import_module(f"metagpt.tools.functions.libs.udf.{module_name}") + # 将函数导入到当前命名空间 + globals().update({function_name: getattr(module, function_name)}) # 获取函数注释和函数路径 function_schema = {'udf_name': function_signature, 'udf_path': f'from metagpt.tools.functions.libs.udf.{module_name} import {function_name}', From cb31ede9c11d1ca7514f75f9abfbb4c5266043b5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 17:58:22 +0800 Subject: [PATCH 198/637] add udf in ML_MODULE_MAP. --- metagpt/prompts/ml_engineer.py | 1 + 1 file changed, 1 insertion(+) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 33eb9c40c..cca9649b3 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -301,6 +301,7 @@ ML_SPECIFIC_PROMPT = { ML_MODULE_MAP = { "data_preprocess": "metagpt.tools.functions.libs.data_preprocess", "feature_engineering": "metagpt.tools.functions.libs.feature_engineering", + "udf": "metagpt.tools.functions.libs.udf", } STRUCTURAL_CONTEXT = """ From c7335419ce32b567fc9cc17b9c70d67656bad0e6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 18:20:04 +0800 Subject: [PATCH 199/637] fix: BaseWriteAnalysisCode now do not install packages or check packages first. --- metagpt/actions/write_analysis_code.py | 22 +++++++++++++--------- 1 file changed, 13 insertions(+), 9 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 0a1d74263..bc069414f 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -29,7 +29,7 @@ from metagpt.utils.common import create_func_config, remove_comments class BaseWriteAnalysisCode(Action): - DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt + DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): @@ -112,13 +112,17 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): if self.schema_path is not None: self._load_tools(schema_path) - def _load_tools(self, schema_path): + def _load_tools(self, schema_path, schema_module=None): """Load tools from yaml file""" - yml_files = schema_path.glob("*.yml") - for yml_file in yml_files: - module = yml_file.stem - with open(yml_file, "r", encoding="utf-8") as f: - self.available_tools[module] = yaml.safe_load(f) + if isinstance(schema_path, dict): + schema_module = schema_module or 'udf' + self.available_tools.update({schema_module: schema_path}) + else: + yml_files = schema_path.glob("*.yml") + for yml_file in yml_files: + module = yml_file.stem + with open(yml_file, "r", encoding="utf-8") as f: + self.available_tools[module] = yaml.safe_load(f) def _parse_recommend_tools(self, module: str, recommend_tools: list) -> dict: """ @@ -174,7 +178,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): column_info: str = "", **kwargs, ) -> Tuple[List[Message], str]: - task_type = plan.current_task.task_type + task_type = plan.current_task.task_type or 'udf' available_tools = self.available_tools.get(task_type, {}) special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") code_steps = plan.current_task.code_steps @@ -227,7 +231,7 @@ class MakeTools(WriteCodeByGenerate): **Notice: 1. Your code must contain a general function start with `def`. 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. - 3. Use Google style for function annotations. + 3. Must use Google style for function docstring, and your code must have function docstring. 4. Write example code after `if __name__ == '__main__':`by using old varibales in old code, and make sure it could be execute in the user's machine. 5. Dont have missing package references.** From 6ed432205bf78831ade0911824aa40914e9a601a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 19 Dec 2023 18:23:21 +0800 Subject: [PATCH 200/637] feat: add use_udfs with WriteCodeWithTools. --- metagpt/roles/ml_engineer.py | 21 +++++++++++++++++---- 1 file changed, 17 insertions(+), 4 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 96e21c8c8..fa9acadbc 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -48,7 +48,7 @@ class MLEngineer(Role): self.plan = Plan(goal=goal) self.use_tools = False - self.make_udfs = False + self.make_udfs = False # user-defined functions self.use_udfs = False self.use_code_steps = False self.execute_code = ExecutePyCode() @@ -171,7 +171,17 @@ class MLEngineer(Role): elif not self.use_tools or self.plan.current_task.task_type == "other": if self.use_udfs: # use user-defined function tools. - pass + from metagpt.tools.functions.libs.udf import UDFS_YAML + logger.warning("Writing code with user-defined function tools...") + logger.info(f"Local user defined function as following:\ + \n{json.dumps(list(UDFS_YAML.keys()), indent=2, ensure_ascii=False)}") + tool_context, code = await WriteCodeWithTools(schema_path=UDFS_YAML).run( + context=context, + plan=self.plan, + column_info=self.data_desc.get("column_info", ""), + ) + debug_context = tool_context + cause_by = WriteCodeWithTools else: logger.info("Write code with pure generation") code = await WriteCodeByGenerate().run( @@ -180,8 +190,10 @@ class MLEngineer(Role): debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] cause_by = WriteCodeByGenerate - if self.make_udfs and len(code.split('\n')) > 2: + if self.make_udfs and len(code.split('\n')) > 4: # make and save user-defined function tools. + logger.warning(f"Making tools for task_id {self.plan.current_task_id}: \ + `{self.plan.current_task.instruction}` \n code {code}") make_tools = MakeTools() code_prompt = f"The following code is about {self.plan.current_task.instruction},\ convert it to be a General Function, {code}" @@ -299,7 +311,8 @@ if __name__ == "__main__": async def main(requirement: str = requirement, auto_run: bool = True): role = MLEngineer(goal=requirement, auto_run=auto_run) - role.make_udfs = True + role.make_udfs = False + role.use_udfs = True await role.run(requirement) fire.Fire(main) From 0f3c0c21e5996f5f28cd26e98fdb3b65da249df8 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 19 Dec 2023 18:50:49 +0800 Subject: [PATCH 201/637] update JUDGE_PROMPT_TEMPLATE in ml_engineer_simple.py --- metagpt/roles/ml_engineer_simple.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/metagpt/roles/ml_engineer_simple.py b/metagpt/roles/ml_engineer_simple.py index e66770211..cc7d8fc97 100644 --- a/metagpt/roles/ml_engineer_simple.py +++ b/metagpt/roles/ml_engineer_simple.py @@ -32,10 +32,10 @@ JUDGE_PROMPT_TEMPLATE = """ # State Output "Ture" or "False". Judging from the code perspective, whether the user's needs have been completely fulfilled. ===== -# Finally output State, Thought and Next Action separately in one sentence +# Output State("Ture" or "False") firstly, then output Thought and Next Steps for the code requirements based on the context respectively in one sentence State: Thought: -Next Action: +Next Steps: """ @@ -132,10 +132,10 @@ class MLEngineerSimple(Role): ) context_msg = [Message(content=context, role="user")] - return context_msg + self.get_working_memories() + return context_msg + self.get_working_memories(6) - def get_working_memories(self, num=6) -> List[Message]: - return self.working_memory.get(num) # 默认为6 + def get_working_memories(self, num=0) -> List[Message]: + return self.working_memory.get(num) # 默认为6 if __name__ == "__main__": From 8afac012b49df5ffb26dc031345c685c748e8797 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 20 Dec 2023 09:57:19 +0800 Subject: [PATCH 202/637] set the plan.current_task.task_type to udf when use udfs. --- metagpt/roles/ml_engineer.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index fa9acadbc..3c1853fd5 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -175,6 +175,8 @@ class MLEngineer(Role): logger.warning("Writing code with user-defined function tools...") logger.info(f"Local user defined function as following:\ \n{json.dumps(list(UDFS_YAML.keys()), indent=2, ensure_ascii=False)}") + # set task_type to `udf` + self.plan.current_task.task_type = 'udf' tool_context, code = await WriteCodeWithTools(schema_path=UDFS_YAML).run( context=context, plan=self.plan, @@ -184,6 +186,7 @@ class MLEngineer(Role): cause_by = WriteCodeWithTools else: logger.info("Write code with pure generation") + # TODO: 添加基于current_task.instruction-code_path的k-v缓存 code = await WriteCodeByGenerate().run( context=context, plan=self.plan, temperature=0.0 ) From 19b0120c15c3ad5cce82256f2cdb374df4507f72 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 20 Dec 2023 09:58:31 +0800 Subject: [PATCH 203/637] restore task_type value. --- metagpt/actions/write_analysis_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index bc069414f..88f22684d 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -178,7 +178,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): column_info: str = "", **kwargs, ) -> Tuple[List[Message], str]: - task_type = plan.current_task.task_type or 'udf' + task_type = plan.current_task.task_type available_tools = self.available_tools.get(task_type, {}) special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") code_steps = plan.current_task.code_steps From a0d2f9b6caaf4ecb5cbc2152a02cedc84060de03 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 20 Dec 2023 11:14:59 +0800 Subject: [PATCH 204/637] update: rm async, mv to utils --- metagpt/roles/ml_engineer.py | 48 ++--------------------------- metagpt/utils/recovery_util.py | 56 ++++++++++++++++++++++++++++++++++ 2 files changed, 59 insertions(+), 45 deletions(-) create mode 100644 metagpt/utils/recovery_util.py diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index f7538ae2e..16ffe69db 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -3,8 +3,7 @@ import json from datetime import datetime import fire -import nbformat -from pathlib import Path + from metagpt.actions import Action from metagpt.actions.debug_code import DebugCode @@ -27,7 +26,7 @@ from metagpt.roles.kaggle_manager import DownloadData, SubmitResult from metagpt.schema import Message, Plan from metagpt.utils.common import remove_comments, create_func_config from metagpt.utils.save_code import save_code_file - +from metagpt.utils.recovery_util import save_history, load_history class UpdateDataColumns(Action): async def run(self, plan: Plan = None) -> dict: @@ -297,49 +296,8 @@ if __name__ == "__main__": save_dir = "" # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" - def load_history(save_dir: str = save_dir): - """ - Load history from the specified save directory. - - Args: - save_dir (str): The directory from which to load the history. - - Returns: - Tuple: A tuple containing the loaded plan and notebook. - """ - - plan_path = Path(save_dir) / "plan.json" - nb_path = Path(save_dir) / "history_nb" / "code.ipynb" - plan = json.load(open(plan_path, "r", encoding="utf-8")) - nb = nbformat.read(open(nb_path, "r", encoding="utf-8"), as_version=nbformat.NO_CONVERT) - return plan, nb - async def save_history(role: Role = MLEngineer, save_dir: str = save_dir): - """ - Save history to the specified directory. - - Args: - role (Role): The role containing the plan and execute_code attributes. - save_dir (str): The directory to save the history. - - Returns: - Path: The path to the saved history directory. - """ - record_time = datetime.now().strftime('%Y-%m-%d_%H-%M-%S') - save_path = DATA_PATH / "output" / f"{record_time}" - - # overwrite exist trajectory - save_path.mkdir(parents=True, exist_ok=True) - - plan = role.plan.dict() - - with open(save_path / "plan.json", "w", encoding="utf-8") as plan_file: - json.dump(plan, plan_file, indent=4, ensure_ascii=False) - - save_code_file(name=Path(record_time) / "history_nb", code_context=role.execute_code.nb, file_format="ipynb") - return save_path - async def main(requirement: str = requirement, auto_run: bool = True, save_dir: str = save_dir): """ @@ -368,7 +326,7 @@ if __name__ == "__main__": await role.run(requirement) except Exception as e: - save_path = await save_history(role, save_dir) + save_path = save_history(role, save_dir) logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") diff --git a/metagpt/utils/recovery_util.py b/metagpt/utils/recovery_util.py new file mode 100644 index 000000000..ef4f0aca7 --- /dev/null +++ b/metagpt/utils/recovery_util.py @@ -0,0 +1,56 @@ +# -*- coding: utf-8 -*- +# @Date : 12/20/2023 11:07 AM +# @Author : stellahong (stellahong@fuzhi.ai) +# @Desc : +import nbformat +from pathlib import Path +import json +from datetime import datetime + +from metagpt.roles.role import Role +from metagpt.roles.ml_engineer import MLEngineer +from metagpt.const import DATA_PATH +from metagpt.utils.save_code import save_code_file + +def load_history(save_dir: str = ""): + """ + Load history from the specified save directory. + + Args: + save_dir (str): The directory from which to load the history. + + Returns: + Tuple: A tuple containing the loaded plan and notebook. + """ + + plan_path = Path(save_dir) / "plan.json" + nb_path = Path(save_dir) / "history_nb" / "code.ipynb" + plan = json.load(open(plan_path, "r", encoding="utf-8")) + nb = nbformat.read(open(nb_path, "r", encoding="utf-8"), as_version=nbformat.NO_CONVERT) + return plan, nb + + +def save_history(role: Role = MLEngineer, save_dir: str = ""): + """ + Save history to the specified directory. + + Args: + role (Role): The role containing the plan and execute_code attributes. + save_dir (str): The directory to save the history. + + Returns: + Path: The path to the saved history directory. + """ + record_time = datetime.now().strftime('%Y-%m-%d_%H-%M-%S') + save_path = DATA_PATH / "output" / f"{record_time}" + + # overwrite exist trajectory + save_path.mkdir(parents=True, exist_ok=True) + + plan = role.plan.dict() + + with open(save_path / "plan.json", "w", encoding="utf-8") as plan_file: + json.dump(plan, plan_file, indent=4, ensure_ascii=False) + + save_code_file(name=Path(record_time) / "history_nb", code_context=role.execute_code.nb, file_format="ipynb") + return save_path \ No newline at end of file From 0c42d55d64b1aa155a584c3feb26641bff5ae067 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 20 Dec 2023 11:16:44 +0800 Subject: [PATCH 205/637] rm comments --- metagpt/actions/write_analysis_code.py | 28 +------------------------- 1 file changed, 1 insertion(+), 27 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 34b605ea9..ecbb68122 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -202,33 +202,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): tool_catalog=tool_catalog, ) - # code_steps_ = eval(code_steps) - # print(code_steps_) - # - # new_code = "" - # tool_context = "" - # for idx, (step_id, step_instruction) in enumerate(code_steps_.items()): - # prompt = TOOL_USAGE_PROMPT.format( - # user_requirement=plan.goal, - # history_code=code_context, - # current_task=plan.current_task.instruction, - # column_info=column_info, - # special_prompt=special_prompt, - # code_steps=step_instruction, - # module_name=module_name, - # tool_catalog=tool_catalog, - # ) - # - # tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - # - # rsp = await self.llm.aask_code(prompt, **tool_config) - # logger.info(f"rsp is: {rsp}") - # new_code = new_code + "\n\n" + rsp["code"] - # code_context = code_context + "\n\n" + new_code - # tool_context = tool_context + "\n\n" + prompt - # context = [Message(content=tool_context, role="user")] - # return context, new_code - + else: prompt = GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, From 913538639ddcf5c129c1681b8734631d0eb4034e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 20 Dec 2023 12:11:42 +0800 Subject: [PATCH 206/637] feat: --- metagpt/roles/ml_engineer.py | 53 +++++++++++++++++++++++++++++------- 1 file changed, 43 insertions(+), 10 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 3c1853fd5..052b99ad5 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -21,6 +21,7 @@ from metagpt.prompts.ml_engineer import ( PRINT_DATA_COLUMNS ) from metagpt.roles import Role +from metagpt.roles.role import RoleContext from metagpt.roles.kaggle_manager import DownloadData, SubmitResult from metagpt.schema import Message, Plan from metagpt.utils.common import remove_comments, create_func_config @@ -192,16 +193,6 @@ class MLEngineer(Role): ) debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] cause_by = WriteCodeByGenerate - - if self.make_udfs and len(code.split('\n')) > 4: - # make and save user-defined function tools. - logger.warning(f"Making tools for task_id {self.plan.current_task_id}: \ - `{self.plan.current_task.instruction}` \n code {code}") - make_tools = MakeTools() - code_prompt = f"The following code is about {self.plan.current_task.instruction},\ - convert it to be a General Function, {code}" - tool_code = await make_tools.run(code_prompt) - make_tools.save(tool_code) else: logger.info("Write code with tools") schema_path = PROJECT_ROOT / "metagpt/tools/functions/schemas" @@ -219,6 +210,9 @@ class MLEngineer(Role): result, success = await self.execute_code.run(code) print(result) + # make tools for successful code and long code. + if success and self.make_udfs and len(code.split('\n')) > 4: + await self.make_tools(code=code) self.working_memory.add( Message(content=result, role="user", cause_by=ExecutePyCode) ) @@ -304,6 +298,39 @@ class MLEngineer(Role): def get_working_memories(self) -> List[Message]: return self.working_memory.get() + def reset(self): + """Restart role with the same goal.""" + self.plan = Plan(goal=self.plan.goal) + self.execute_code = ExecutePyCode() + + async def make_tools(self, code: str): + """Make user-defined functions(udfs, aka tools) for pure generation code. + + Args: + code (str): pure generation code by class WriteCodeByGenerate. + """ + logger.warning(f"Making tools for task_id {self.plan.current_task_id}: \ + `{self.plan.current_task.instruction}` \n code: \n {code}") + make_tools = MakeTools() + code_prompt = f"The following code is about {self.plan.current_task.instruction},\ + convert it to be a General Function, {code}" + tool_code = await make_tools.run(code_prompt) + # check tool_code by execute_code + logger.info(f"Checking task_id {self.plan.current_task_id} tool code by executor...") + _, success = await self.execute_code.run(tool_code) + make_tool_retries, make_tool_current_retry = 3, 1 + while not success: + tool_code = await make_tools.run(code_prompt) + _, success = await self.execute_code.run(tool_code) + if make_tool_current_retry > make_tool_retries: + logger.error(f"We have tried the maximum number of attempts {make_tool_retries}\ + and still have not created tools for task_id {self.plan.current_task_id} successfully,\ + we will skip it.") + break + # save successful tool code in udf + if success: + make_tools.save(tool_code) + if __name__ == "__main__": requirement = "Run data analysis on sklearn Iris dataset, include a plot" @@ -314,6 +341,12 @@ if __name__ == "__main__": async def main(requirement: str = requirement, auto_run: bool = True): role = MLEngineer(goal=requirement, auto_run=auto_run) + # make udfs + role.make_udfs = True + role.use_udfs = False + await role.run(requirement) + # use udfs + role.reset() role.make_udfs = False role.use_udfs = True await role.run(requirement) From 7b8c15b5df5cdb3a622a51945053f02bbc3dc25b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 20 Dec 2023 12:15:51 +0800 Subject: [PATCH 207/637] feat: add make_tools and feat function. --- metagpt/roles/ml_engineer.py | 1 - 1 file changed, 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 052b99ad5..b908d9ef8 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -21,7 +21,6 @@ from metagpt.prompts.ml_engineer import ( PRINT_DATA_COLUMNS ) from metagpt.roles import Role -from metagpt.roles.role import RoleContext from metagpt.roles.kaggle_manager import DownloadData, SubmitResult from metagpt.schema import Message, Plan from metagpt.utils.common import remove_comments, create_func_config From 48ef61c6e42c65aa38a5c4466c24191912198c4e Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 20 Dec 2023 14:46:29 +0800 Subject: [PATCH 208/637] change format --- metagpt/roles/ml_engineer.py | 12 +++++------- 1 file changed, 5 insertions(+), 7 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 16ffe69db..33b570d1a 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -4,7 +4,6 @@ from datetime import datetime import fire - from metagpt.actions import Action from metagpt.actions.debug_code import DebugCode from metagpt.actions.execute_code import ExecutePyCode @@ -28,6 +27,7 @@ from metagpt.utils.common import remove_comments, create_func_config from metagpt.utils.save_code import save_code_file from metagpt.utils.recovery_util import save_history, load_history + class UpdateDataColumns(Action): async def run(self, plan: Plan = None) -> dict: finished_tasks = plan.get_finished_tasks() @@ -41,7 +41,7 @@ class UpdateDataColumns(Action): class MLEngineer(Role): def __init__( - self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False + self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") @@ -104,8 +104,7 @@ class MLEngineer(Role): task.code = task.code + "\n\n" + new_code confirmed_and_more = (ReviewConst.CONTINUE_WORD[0] in review.lower() - and review.lower() not in ReviewConst.CONTINUE_WORD[ - 0]) # "confirm, ... (more content, such as changing downstream tasks)" + and review.lower() not in ReviewConst.CONTINUE_WORD[0]) # "confirm, ... (more content, such as changing downstream tasks)" if confirmed_and_more: self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) await self._update_plan(review) @@ -294,11 +293,10 @@ if __name__ == "__main__": requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." save_dir = "" + + # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" - - - async def main(requirement: str = requirement, auto_run: bool = True, save_dir: str = save_dir): """ The main function to run the MLEngineer with optional history loading. From 72e550b148927ea7c58b989a5f80fde79dfc713e Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 20 Dec 2023 15:39:18 +0800 Subject: [PATCH 209/637] minor update: move action, fix circular import, add entry parameters --- metagpt/actions/ml_da_action.py | 18 +++++++++++++-- metagpt/roles/ml_engineer.py | 40 ++++++++------------------------- metagpt/utils/recovery_util.py | 3 +-- 3 files changed, 26 insertions(+), 35 deletions(-) diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py index 5e4580b17..b6270f12f 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_da_action.py @@ -3,9 +3,12 @@ from typing import Dict, List, Union from metagpt.actions import Action from metagpt.schema import Message, Plan -from metagpt.utils.common import CodeParser +from metagpt.utils.common import CodeParser, remove_comments, create_func_config from metagpt.logs import logger - +from metagpt.prompts.ml_engineer import ( + UPDATE_DATA_COLUMNS, + PRINT_DATA_COLUMNS +) class ReviewConst: TASK_REVIEW_TRIGGER = "task" @@ -114,3 +117,14 @@ class Reflect(Action): rsp = CodeParser.parse_code(block=None, text=rsp_json) reflection = json.loads(rsp)["reflection"] return reflection + + +class UpdateDataColumns(Action): + async def run(self, plan: Plan = None) -> dict: + finished_tasks = plan.get_finished_tasks() + code_context = [remove_comments(task.code) for task in finished_tasks] + code_context = "\n\n".join(code_context) + prompt = UPDATE_DATA_COLUMNS.format(history_code=code_context) + tool_config = create_func_config(PRINT_DATA_COLUMNS) + rsp = await self.llm.aask_code(prompt, **tool_config) + return rsp diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 33b570d1a..73aba1fe8 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -4,10 +4,9 @@ from datetime import datetime import fire -from metagpt.actions import Action from metagpt.actions.debug_code import DebugCode from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis, Reflect, ReviewConst +from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis, Reflect, ReviewConst, UpdateDataColumns from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.actions.write_plan import WritePlan @@ -16,42 +15,26 @@ from metagpt.const import DATA_PATH, PROJECT_ROOT from metagpt.logs import logger from metagpt.memory import Memory from metagpt.prompts.ml_engineer import STRUCTURAL_CONTEXT -from metagpt.prompts.ml_engineer import ( - UPDATE_DATA_COLUMNS, - PRINT_DATA_COLUMNS -) from metagpt.roles import Role from metagpt.roles.kaggle_manager import DownloadData, SubmitResult from metagpt.schema import Message, Plan -from metagpt.utils.common import remove_comments, create_func_config from metagpt.utils.save_code import save_code_file from metagpt.utils.recovery_util import save_history, load_history -class UpdateDataColumns(Action): - async def run(self, plan: Plan = None) -> dict: - finished_tasks = plan.get_finished_tasks() - code_context = [remove_comments(task.code) for task in finished_tasks] - code_context = "\n\n".join(code_context) - prompt = UPDATE_DATA_COLUMNS.format(history_code=code_context) - tool_config = create_func_config(PRINT_DATA_COLUMNS) - rsp = await self.llm.aask_code(prompt, **tool_config) - return rsp - - class MLEngineer(Role): def __init__( - self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False + self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False, use_tools=False, use_code_steps=False, ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act") self._watch([DownloadData, SubmitResult]) self.plan = Plan(goal=goal) - self.use_tools = True - self.use_code_steps = True self.execute_code = ExecutePyCode() self.auto_run = auto_run + self.use_tools = use_tools + self.use_code_steps = use_code_steps self.data_desc = {} # memory for working on each task, discarded each time a task is done @@ -277,7 +260,6 @@ if __name__ == "__main__": # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" - # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." # data_path = f"{DATA_PATH}/titanic" @@ -291,13 +273,10 @@ if __name__ == "__main__": data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - save_dir = "" - - # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" - - async def main(requirement: str = requirement, auto_run: bool = True, save_dir: str = save_dir): + + async def main(requirement: str = requirement, auto_run: bool = True, use_tools: bool = False, use_code_steps: bool = False, save_dir: str = ""): """ The main function to run the MLEngineer with optional history loading. @@ -312,13 +291,13 @@ if __name__ == "__main__": if save_dir: logger.info("Resuming from history trajectory") plan, nb = load_history(save_dir) - role = MLEngineer(goal=requirement, auto_run=auto_run) + role = MLEngineer(goal=requirement, auto_run=auto_run, use_tools=use_tools, use_code_steps=use_code_steps) role.plan = Plan(**plan) role.execute_code = ExecutePyCode(nb) else: logger.info("Run from scratch") - role = MLEngineer(goal=requirement, auto_run=auto_run) + role = MLEngineer(goal=requirement, auto_run=auto_run, use_tools=use_tools, use_code_steps=use_code_steps) try: await role.run(requirement) @@ -327,6 +306,5 @@ if __name__ == "__main__": save_path = save_history(role, save_dir) logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") - - + fire.Fire(main) diff --git a/metagpt/utils/recovery_util.py b/metagpt/utils/recovery_util.py index ef4f0aca7..afe7fc021 100644 --- a/metagpt/utils/recovery_util.py +++ b/metagpt/utils/recovery_util.py @@ -8,7 +8,6 @@ import json from datetime import datetime from metagpt.roles.role import Role -from metagpt.roles.ml_engineer import MLEngineer from metagpt.const import DATA_PATH from metagpt.utils.save_code import save_code_file @@ -30,7 +29,7 @@ def load_history(save_dir: str = ""): return plan, nb -def save_history(role: Role = MLEngineer, save_dir: str = ""): +def save_history(role: Role, save_dir: str = ""): """ Save history to the specified directory. From 99945e3493797b117ba022a974912ceeffb8fda4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 20 Dec 2023 17:57:07 +0800 Subject: [PATCH 210/637] update default_system_msg in BaseWriteAnalysisCode. --- metagpt/actions/write_analysis_code.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 88f22684d..924677605 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -33,7 +33,7 @@ class BaseWriteAnalysisCode(Action): # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): - default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG + default_system_msg = system_msg or "" # 全部转成list if not isinstance(prompt, list): prompt = [prompt] @@ -231,7 +231,7 @@ class MakeTools(WriteCodeByGenerate): **Notice: 1. Your code must contain a general function start with `def`. 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. - 3. Must use Google style for function docstring, and your code must have function docstring. + 3. Must use Google style for function docstring, and your docstring must be consistent with the code,without missing anything. 4. Write example code after `if __name__ == '__main__':`by using old varibales in old code, and make sure it could be execute in the user's machine. 5. Dont have missing package references.** From aa5c42ff8b99023bc05df075f5c15c486ebd3f2f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 20 Dec 2023 18:12:15 +0800 Subject: [PATCH 211/637] use self.DEFAULT_SYSTEM_MSG in process_msg. --- metagpt/actions/write_analysis_code.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 924677605..e50c069f0 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -96,7 +96,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): **kwargs, ) -> str: # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) - prompt = self.process_msg(context, system_msg) + prompt = self.process_msg(context, system_msg or self.DEFAULT_SYSTEM_MSG) code_content = await self.llm.aask_code(prompt, **kwargs) return code_content["code"] @@ -269,7 +269,7 @@ class MakeTools(WriteCodeByGenerate): @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) async def run(self, code_message: List[Message | Dict], **kwargs) -> str: - msgs = self.process_msg(code_message) + msgs = self.process_msg(code_message, self.DEFAULT_SYSTEM_MSG) logger.info(f"\n\nAsk to Make tools:\n{'-'*60}\n {msgs[-1]}") tool_code = await self.llm.aask_code(msgs, **kwargs) max_tries, current_try = 3, 1 From 1145641cdcbb94b3506c820ea10adc31e35d61aa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 20 Dec 2023 18:16:32 +0800 Subject: [PATCH 212/637] update --- .../actions/test_write_analysis_code.py | 22 +------------------ 1 file changed, 1 insertion(+), 21 deletions(-) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index 68ca129cc..1a568cdcd 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -1,7 +1,7 @@ import asyncio import pytest -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools, WriteCodeWithUDFs +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.execute_code import ExecutePyCode from metagpt.schema import Message, Plan, Task from metagpt.logs import logger @@ -304,23 +304,3 @@ async def test_write_code_reuse_code_long_for_wine(): success_rate = sum(success) / trials_num logger.info(f"success rate: {success_rate :.2f}") assert success_rate >= 0.8 - - -@pytest.mark.asyncio -async def test_write_code_with_udfs(): - wudf = WriteCodeWithUDFs() - ep = ExecutePyCode() - rsp = await wudf.run("Get Apple stock data for the past 90 days.") - logger.info(rsp) - assert 'metagpt' in rsp - output, output_type = await ep.run(rsp) - assert output_type is True - logger.info(output) - - -@pytest.mark.asyncio -async def test_write_code_with_udfs_no_udf_found(): - wudf = WriteCodeWithUDFs() - rsp = await wudf.run("Identify if there is a dog in the picture.") - logger.info(rsp) - assert 'No udf found' in rsp From 5af4f6b4c524e62dd43ff6f6f6e80062f8427ead Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 20 Dec 2023 19:56:26 +0800 Subject: [PATCH 213/637] add new test for aask_code about write code by steps. --- tests/metagpt/provider/test_openai.py | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index 2b0af37b5..98a3670f1 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -78,3 +78,17 @@ def test_ask_code_list_str(): assert "language" in rsp assert "code" in rsp assert len(rsp["code"]) > 0 + + +@pytest.mark.asyncio +async def test_ask_code_steps2(): + llm = OpenAIGPTAPI() + msg = ["step by setp 生成代码: Step 1. 先生成随机数组a, Step 2. 求a中最大值, Step 3. 绘制数据a的直方图"] + rsp = await llm.aask_code(msg) # -> {'language': 'python', 'code': 'max_value = max(a)\nmax_value'} + print(rsp) + assert "language" in rsp + assert "code" in rsp + assert len(rsp["code"]) > 0 + assert "Step 1" in rsp["code"] + assert "Step 2" in rsp["code"] + assert "Step 3" in rsp["code"] From a39cc30164140588c3b4a938618cfe22893d1438 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 21 Dec 2023 10:11:07 +0800 Subject: [PATCH 214/637] add test for ml_engineer. --- tests/metagpt/roles/test_daml.py | 36 ++++++++++++++++++++++++++++++++ 1 file changed, 36 insertions(+) create mode 100644 tests/metagpt/roles/test_daml.py diff --git a/tests/metagpt/roles/test_daml.py b/tests/metagpt/roles/test_daml.py new file mode 100644 index 000000000..672a3daed --- /dev/null +++ b/tests/metagpt/roles/test_daml.py @@ -0,0 +1,36 @@ +import pytest +from tqdm import tqdm + +from metagpt.logs import logger +from metagpt.roles.ml_engineer import MLEngineer + + +async def make_use_tools(requirement: str, auto_run: bool = True): + """make and use tools for requirement.""" + role = MLEngineer(goal=requirement, auto_run=auto_run) + # make udfs + role.make_udfs = True + role.use_udfs = False + await role.run(requirement) + # use udfs + role.reset() + role.make_udfs = False + role.use_udfs = True + await role.run(requirement) + + +@pytest.mark.asyncio +async def test_make_use_tools(): + requirements = ["Run data analysis on sklearn Iris dataset, include a plot", + "Run data analysis on sklearn Diabetes dataset, include a plot", + "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy", + "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy", + "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: tests/data/titanic.csv"] + success = 0 + for requirement in tqdm(requirements, total=len(requirements)): + try: + await make_use_tools(requirement) + success += 1 + except Exception as e: + logger.error(f"Found Error in {requirement}, {e}") + logger.info(f"success: {round(success/len(requirements), 1)*100}%") From c43e2bed6b916096f117f72db393903694a7c090 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 21 Dec 2023 10:14:25 +0800 Subject: [PATCH 215/637] update condition for DebugCode. --- metagpt/roles/ml_engineer.py | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index b908d9ef8..9fa12b41d 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -99,7 +99,7 @@ class MLEngineer(Role): self.plan.finish_current_task() self.working_memory.clear() - if self.use_tools: + if self.use_tools or self.use_udfs: success, new_code = await self._update_data_columns() if success: task.code = task.code + "\n\n" + new_code @@ -159,7 +159,8 @@ class MLEngineer(Role): # print(context) # print("*" * 10) # breakpoint() - if counter > 0 and self.use_tools: + if counter > 0 and (self.use_tools or self.use_udfs): + logger.warning('We got a bug code, now start to debug...') code = await DebugCode().run( plan=self.plan.current_task.instruction, code=code, @@ -168,11 +169,11 @@ class MLEngineer(Role): ) logger.info(f"new code \n{code}") cause_by = DebugCode - elif not self.use_tools or self.plan.current_task.task_type == "other": + elif not self.use_tools or self.plan.current_task.task_type in ("other", "udf"): if self.use_udfs: # use user-defined function tools. from metagpt.tools.functions.libs.udf import UDFS_YAML - logger.warning("Writing code with user-defined function tools...") + logger.warning("Writing code with user-defined function tools by WriteCodeWithTools.") logger.info(f"Local user defined function as following:\ \n{json.dumps(list(UDFS_YAML.keys()), indent=2, ensure_ascii=False)}") # set task_type to `udf` @@ -211,6 +212,7 @@ class MLEngineer(Role): print(result) # make tools for successful code and long code. if success and self.make_udfs and len(code.split('\n')) > 4: + logger.info('Execute code successfully. Now start to make tools ...') await self.make_tools(code=code) self.working_memory.add( Message(content=result, role="user", cause_by=ExecutePyCode) From 1160f075360aecf53b6604bcdbc0cc98d4913f87 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 21 Dec 2023 11:03:54 +0800 Subject: [PATCH 216/637] update reset. --- metagpt/roles/ml_engineer.py | 87 +++++++++++++++++++----------------- 1 file changed, 46 insertions(+), 41 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 5d514a18f..3e656304b 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -25,7 +25,7 @@ from metagpt.roles.kaggle_manager import DownloadData, SubmitResult from metagpt.schema import Message, Plan from metagpt.utils.common import remove_comments, create_func_config from metagpt.utils.save_code import save_code_file -from metagpt.utils.recovery_util import save_history, load_history +# from metagpt.utils.recovery_util import save_history, load_history class UpdateDataColumns(Action): @@ -297,6 +297,7 @@ class MLEngineer(Role): """Restart role with the same goal.""" self.plan = Plan(goal=self.plan.goal) self.execute_code = ExecutePyCode() + self.working_memory = Memory() async def make_tools(self, code: str): """Make user-defined functions(udfs, aka tools) for pure generation code. @@ -328,23 +329,27 @@ class MLEngineer(Role): if __name__ == "__main__": - # requirement = "Run data analysis on sklearn Iris dataset, include a plot" + requirement = "Run data analysis on sklearn Iris dataset, include a plot" # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" - # async def main(requirement: str = requirement, auto_run: bool = True): - # role = MLEngineer(goal=requirement, auto_run=auto_run) - # # make udfs - # role.make_udfs = True - # role.use_udfs = False - # await role.run(requirement) - # # use udfs - # role.reset() - # role.make_udfs = False - # role.use_udfs = True - # await role.run(requirement) + async def main(requirement: str = requirement, auto_run: bool = True): + role = MLEngineer(goal=requirement, auto_run=auto_run) + # make udfs + role.use_tools = False + role.use_code_steps = False + role.make_udfs = True + role.use_udfs = False + await role.run(requirement) + # use udfs + role.reset() + role.make_udfs = False + role.use_udfs = True + role.use_code_steps = False + role.use_tools = False + await role.run(requirement) # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." @@ -358,44 +363,44 @@ if __name__ == "__main__": # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." - data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" - requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" + # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - save_dir = "" + # save_dir = "" - # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" + # # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" - async def main(requirement: str = requirement, auto_run: bool = True, save_dir: str = save_dir): - """ - The main function to run the MLEngineer with optional history loading. + # async def main(requirement: str = requirement, auto_run: bool = True, save_dir: str = save_dir): + # """ + # The main function to run the MLEngineer with optional history loading. - Args: - requirement (str): The requirement for the MLEngineer. - auto_run (bool): Whether to auto-run the MLEngineer. - save_dir (str): The directory from which to load the history or to save the new history. + # Args: + # requirement (str): The requirement for the MLEngineer. + # auto_run (bool): Whether to auto-run the MLEngineer. + # save_dir (str): The directory from which to load the history or to save the new history. - Raises: - Exception: If an error occurs during execution, log the error and save the history. - """ - if save_dir: - logger.info("Resuming from history trajectory") - plan, nb = load_history(save_dir) - role = MLEngineer(goal=requirement, auto_run=auto_run) - role.plan = Plan(**plan) - role.execute_code = ExecutePyCode(nb) + # Raises: + # Exception: If an error occurs during execution, log the error and save the history. + # """ + # if save_dir: + # logger.info("Resuming from history trajectory") + # plan, nb = load_history(save_dir) + # role = MLEngineer(goal=requirement, auto_run=auto_run) + # role.plan = Plan(**plan) + # role.execute_code = ExecutePyCode(nb) - else: - logger.info("Run from scratch") - role = MLEngineer(goal=requirement, auto_run=auto_run) + # else: + # logger.info("Run from scratch") + # role = MLEngineer(goal=requirement, auto_run=auto_run) - try: - await role.run(requirement) - except Exception as e: + # try: + # await role.run(requirement) + # except Exception as e: - save_path = save_history(role, save_dir) + # save_path = save_history(role, save_dir) - logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") + # logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") fire.Fire(main) From 82dce58e4e3c646b3cb2190c8db9a854bc297969 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 21 Dec 2023 13:33:42 +0800 Subject: [PATCH 217/637] update DEFAULT_SYSTEM_MSG. --- metagpt/actions/write_analysis_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 02aba0e62..d457ea75b 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -29,7 +29,7 @@ from metagpt.utils.common import create_func_config, remove_comments class BaseWriteAnalysisCode(Action): - DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt + DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): From e8f5ce0f0a64c222af06b59588707798d3444a6b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 21 Dec 2023 13:34:31 +0800 Subject: [PATCH 218/637] update use_udfs. --- metagpt/roles/ml_engineer.py | 43 ++++++++++++++++++------------------ 1 file changed, 21 insertions(+), 22 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 7e5cc8caf..092229ec9 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -148,7 +148,16 @@ class MLEngineer(Role): ) logger.info(f"new code \n{code}") cause_by = DebugCode - elif not self.use_tools or self.plan.current_task.task_type in ("other", "udf"): + elif not self.use_tools or self.plan.current_task.task_type == 'other': + logger.info("Write code with pure generation") + # TODO: 添加基于current_task.instruction-code_path的k-v缓存 + code = await WriteCodeByGenerate().run( + context=context, plan=self.plan, temperature=0.0 + ) + debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] + cause_by = WriteCodeByGenerate + else: + logger.info("Write code with tools") if self.use_udfs: # use user-defined function tools. from metagpt.tools.functions.libs.udf import UDFS_YAML @@ -165,24 +174,14 @@ class MLEngineer(Role): debug_context = tool_context cause_by = WriteCodeWithTools else: - logger.info("Write code with pure generation") - # TODO: 添加基于current_task.instruction-code_path的k-v缓存 - code = await WriteCodeByGenerate().run( - context=context, plan=self.plan, temperature=0.0 + schema_path = PROJECT_ROOT / "metagpt/tools/functions/schemas" + tool_context, code = await WriteCodeWithTools(schema_path=schema_path).run( + context=context, + plan=self.plan, + column_info=self.data_desc.get("column_info", ""), ) - debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] - cause_by = WriteCodeByGenerate - else: - logger.info("Write code with tools") - schema_path = PROJECT_ROOT / "metagpt/tools/functions/schemas" - tool_context, code = await WriteCodeWithTools(schema_path=schema_path).run( - context=context, - plan=self.plan, - column_info=self.data_desc.get("column_info", ""), - ) - debug_context = tool_context - cause_by = WriteCodeWithTools - + debug_context = tool_context + cause_by = WriteCodeWithTools self.working_memory.add( Message(content=code, role="assistant", cause_by=cause_by) ) @@ -346,10 +345,10 @@ if __name__ == "__main__": # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." - data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" - requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - save_dir = "" - # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" + # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" + # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + # save_dir = "" + # # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" async def main(requirement: str = requirement, auto_run: bool = True, use_tools: bool = False, use_code_steps: bool = False, save_dir: str = ""): """ From 94b352cf2375296567bb1033efee85855f64e724 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 21 Dec 2023 16:43:23 +0800 Subject: [PATCH 219/637] update MakeTools DEFAULT_SYSTEM_MSG. --- metagpt/actions/write_analysis_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index d457ea75b..099934c5a 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -236,7 +236,7 @@ class MakeTools(WriteCodeByGenerate): 3. Must use Google style for function docstring, and your docstring must be consistent with the code,without missing anything. 4. Write example code after `if __name__ == '__main__':`by using old varibales in old code, and make sure it could be execute in the user's machine. - 5. Dont have missing package references.** + 5. Only use the imported packages** """ def __init__(self, name: str = '', context: list[Message] = None, llm: LLM = None, workspace: str = None): From 6d36511249fbbcd4fc595f9a9c11861cac94c8d2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 21 Dec 2023 16:58:02 +0800 Subject: [PATCH 220/637] update make tools: code -> remove_comments(code). --- metagpt/roles/ml_engineer.py | 17 +++++++++-------- 1 file changed, 9 insertions(+), 8 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 092229ec9..f44d42554 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -20,6 +20,7 @@ from metagpt.roles.kaggle_manager import DownloadData, SubmitResult from metagpt.schema import Message, Plan from metagpt.utils.save_code import save_code_file from metagpt.utils.recovery_util import save_history, load_history +from metagpt.utils.common import remove_comments class MLEngineer(Role): @@ -189,7 +190,7 @@ class MLEngineer(Role): result, success = await self.execute_code.run(code) print(result) # make tools for successful code and long code. - if success and self.make_udfs and len(code.split('\n')) > 4: + if success and self.make_udfs and len(remove_comments(code).split('\n')) > 4: logger.info('Execute code successfully. Now start to make tools ...') await self.make_tools(code=code) self.working_memory.add( @@ -326,12 +327,12 @@ if __name__ == "__main__": role.use_udfs = False await role.run(requirement) # use udfs - role.reset() - role.make_udfs = False - role.use_udfs = True - role.use_code_steps = False - role.use_tools = False - await role.run(requirement) + # role.reset() + # role.make_udfs = False + # role.use_udfs = True + # role.use_code_steps = False + # role.use_tools = False + # await role.run(requirement) # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." @@ -381,4 +382,4 @@ if __name__ == "__main__": logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") - fire.Fire(main) + fire.Fire(run_udfs) From 01fe23be4508a4791e8096cd0824d276f4359098 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 21 Dec 2023 17:05:14 +0800 Subject: [PATCH 221/637] update ml ops --- .../tools/functions/libs/data_preprocess.py | 42 ++++-- .../functions/libs/feature_engineering.py | 131 ++++++++++++++---- .../functions/schemas/data_preprocess.yml | 2 +- .../functions/schemas/feature_engineering.yml | 125 ++++++++++++++++- 4 files changed, 257 insertions(+), 43 deletions(-) diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index 8c70462ee..f1665b405 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -1,3 +1,5 @@ +import json + import numpy as np import pandas as pd from sklearn.impute import SimpleImputer @@ -20,10 +22,14 @@ class FillMissingValue(MLProcess): self.si = None def fit(self, df: pd.DataFrame): + if len(self.features) == 0: + return self.si = SimpleImputer(strategy=self.strategy, fill_value=self.fill_value) self.si.fit(df[self.features]) def transform(self, df: pd.DataFrame): + if len(self.features) == 0: + return df df[self.features] = self.si.transform(df[self.features]) return df @@ -122,11 +128,15 @@ class LabelEncode(MLProcess): self.le_encoders = [] def fit(self, df: pd.DataFrame): + if len(self.features) == 0: + return for col in self.features: le = LabelEncoder().fit(df[col].astype(str).unique().tolist() + ['unknown']) self.le_encoders.append(le) def transform(self, df: pd.DataFrame): + if len(self.features) == 0: + return df for i in range(len(self.features)): data_list = df[self.features[i]].astype(str).tolist() for unique_item in np.unique(df[self.features[i]].astype(str)): @@ -137,17 +147,23 @@ class LabelEncode(MLProcess): def get_column_info(df: pd.DataFrame) -> dict: - data = [] - for i in df.columns: - nan_freq = float("%.2g" % (df[i].isna().mean() * 100)) - n_unique = df[i].nunique() - data_type = str(df[i].dtype).replace("dtype('", "").replace("')", "") - if data_type == "O": - data_type = "object" - data.append([i, data_type, nan_freq, n_unique]) + column_info = { + "Category": [], + "Numeric": [], + "Datetime": [], + "Others": [], + } + for col in df.columns: + data_type = str(df[col].dtype).replace("dtype('", "").replace("')", "") + if data_type.startswith("object"): + column_info["Category"].append(col) + elif data_type.startswith("int") or data_type.startswith("float"): + column_info["Numeric"].append(col) + elif data_type.startswith("datetime"): + column_info["Datetime"].append(col) + else: + column_info["Others"].append(col) - samples = pd.DataFrame( - data, - columns=["Column_name", "Data_type", "NaN_Frequency(%)", "N_unique"], - ) - return samples.to_dict(orient='list') + if len(json.dumps(column_info)) > 2000: + column_info['Numeric'] = column_info['Numeric'][0:5] + ['Too many cols, omission here...'] + return column_info diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index 1ec2b9675..df36752b9 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -6,12 +6,12 @@ # @Desc : Feature Engineering Tools import itertools +import lightgbm as lgb import numpy as np import pandas as pd -from dateutil.relativedelta import relativedelta from joblib import Parallel, delayed -from pandas.api.types import is_numeric_dtype from pandas.core.dtypes.common import is_object_dtype +from sklearn.feature_selection import VarianceThreshold from sklearn.model_selection import KFold from sklearn.preprocessing import PolynomialFeatures, KBinsDiscretizer @@ -19,15 +19,27 @@ from metagpt.tools.functions.libs.base import MLProcess class PolynomialExpansion(MLProcess): - def __init__(self, cols: list, degree: int = 2): + def __init__(self, cols: list, degree: int = 2, label_col: str = None): self.cols = cols self.degree = degree + self.label_col = label_col + if self.label_col in self.cols: + self.cols.remove(self.label_col) self.poly = PolynomialFeatures(degree=degree, include_bias=False) def fit(self, df: pd.DataFrame): + if len(self.cols) == 0: + return + if len(self.cols) > 10: + corr = df[self.cols + [self.label_col]].corr() + corr = corr[self.label_col].abs().sort_values(ascending=False) + self.cols = corr.index.tolist()[1:11] + self.poly.fit(df[self.cols].fillna(0)) def transform(self, df: pd.DataFrame) -> pd.DataFrame: + if len(self.cols) == 0: + return df ts_data = self.poly.transform(df[self.cols].fillna(0)) column_name = self.poly.get_feature_names_out(self.cols) ts_data = pd.DataFrame(ts_data, index=df.index, columns=column_name) @@ -158,27 +170,35 @@ class SplitBins(MLProcess): df[self.cols] = self.encoder.transform(df[self.cols].fillna(0)) return df -# @registry.register("feature_engineering", ExtractTimeComps) -# def extract_time_comps(df, time_col, time_comps): -# time_s = pd.to_datetime(df[time_col], errors="coerce") -# time_comps_df = pd.DataFrame() -# -# if "year" in time_comps: -# time_comps_df["year"] = time_s.dt.year -# if "month" in time_comps: -# time_comps_df["month"] = time_s.dt.month -# if "day" in time_comps: -# time_comps_df["day"] = time_s.dt.day -# if "hour" in time_comps: -# time_comps_df["hour"] = time_s.dt.hour -# if "dayofweek" in time_comps: -# time_comps_df["dayofweek"] = time_s.dt.dayofweek + 1 -# if "is_weekend" in time_comps: -# time_comps_df["is_weekend"] = time_s.dt.dayofweek.isin([5, 6]).astype(int) -# df = pd.concat([df, time_comps_df], axis=1) -# return df -# -# + +class ExtractTimeComps(MLProcess): + def __init__(self, time_col: str, time_comps: list): + self.time_col = time_col + self.time_comps = time_comps + + def fit(self, df: pd.DataFrame): + pass + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + time_s = pd.to_datetime(df[self.time_col], errors="coerce") + time_comps_df = pd.DataFrame() + + if "year" in self.time_comps: + time_comps_df["year"] = time_s.dt.year + if "month" in self.time_comps: + time_comps_df["month"] = time_s.dt.month + if "day" in self.time_comps: + time_comps_df["day"] = time_s.dt.day + if "hour" in self.time_comps: + time_comps_df["hour"] = time_s.dt.hour + if "dayofweek" in self.time_comps: + time_comps_df["dayofweek"] = time_s.dt.dayofweek + 1 + if "is_weekend" in self.time_comps: + time_comps_df["is_weekend"] = time_s.dt.dayofweek.isin([5, 6]).astype(int) + df = pd.concat([df, time_comps_df], axis=1) + return df + + # @registry.register("feature_engineering", FeShiftByTime) # def fe_shift_by_time(df, time_col, group_col, shift_col, periods, freq): # df[time_col] = pd.to_datetime(df[time_col]) @@ -290,3 +310,66 @@ class GeneralSelection(MLProcess): def transform(self, df: pd.DataFrame) -> pd.DataFrame: df = df[self.feats + [self.label_col]] return df + + +class TreeBasedSelection(MLProcess): + def __init__(self, label_col: str, task_type: str): + self.label_col = label_col + self.task_type = task_type + self.feats = None + + def fit(self, df: pd.DataFrame): + params = { + 'boosting_type': 'gbdt', + 'objective': 'binary', + 'learning_rate': 0.1, + 'num_leaves': 31, + } + + if self.task_type == "cls": + params["objective"] = "binary" + params["metric"] = "auc" + elif self.task_type == "mcls": + params["objective"] = "multiclass" + params["num_class"] = df[self.label_col].nunique() + params["metric"] = "auc_mu" + elif self.task_type == "reg": + params["objective"] = "regression" + params["metric"] = "rmse" + + num_cols = df.select_dtypes(include=np.number).columns.tolist() + cols = [f for f in num_cols if f not in [self.label_col]] + + dtrain = lgb.Dataset(df[cols], df[self.label_col]) + model = lgb.train(params, dtrain, num_boost_round=100) + df_imp = pd.DataFrame({'feature_name': dtrain.feature_name, + 'importance': model.feature_importance("gain")}) + + df_imp.sort_values("importance", ascending=False, inplace=True) + df_imp = df_imp[df_imp["importance"] > 0] + self.feats = df_imp['feature_name'].tolist() + self.feats.append(self.label_col) + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + df = df[self.feats] + return df + + +class VarianceBasedSelection(MLProcess): + def __init__(self, label_col: str, threshold: float = 0): + self.label_col = label_col + self.threshold = threshold + self.feats = None + self.selector = VarianceThreshold(threshold=self.threshold) + + def fit(self, df: pd.DataFrame): + num_cols = df.select_dtypes(include=np.number).columns.tolist() + cols = [f for f in num_cols if f not in [self.label_col]] + + self.selector.fit(df[cols]) + self.feats = df[cols].columns[self.selector.get_support(indices=True)].tolist() + self.feats.append(self.label_col) + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + df = df[self.feats] + return df diff --git a/metagpt/tools/functions/schemas/data_preprocess.yml b/metagpt/tools/functions/schemas/data_preprocess.yml index 95b0124cc..4de697abd 100644 --- a/metagpt/tools/functions/schemas/data_preprocess.yml +++ b/metagpt/tools/functions/schemas/data_preprocess.yml @@ -11,7 +11,7 @@ FillMissingValue: description: "columns to be processed" strategy: type: str - description: "the imputation strategy" + description: "the imputation strategy, notice mean/median can only be used for numeric features" default: mean enum: - mean diff --git a/metagpt/tools/functions/schemas/feature_engineering.yml b/metagpt/tools/functions/schemas/feature_engineering.yml index 3ba9e863b..62e6ad5b3 100644 --- a/metagpt/tools/functions/schemas/feature_engineering.yml +++ b/metagpt/tools/functions/schemas/feature_engineering.yml @@ -1,6 +1,6 @@ PolynomialExpansion: type: class - description: "Add polynomial and interaction features from selected numeric columns, excluding the bias column." + description: "Add polynomial and interaction features from selected numeric columns to input DataFrame." methods: __init__: description: "Initialize self." @@ -9,12 +9,16 @@ PolynomialExpansion: cols: type: list description: "Columns for polynomial expansion." + label_col: + type: str + description: "Label column name." degree: type: int description: "The degree of the polynomial features." default: 2 required: - cols + - label_col fit: description: "Fit the PolynomialExpansion model." parameters: @@ -36,14 +40,14 @@ PolynomialExpansion: returns: df: type: DataFrame - description: "The transformed DataFrame." + description: "The transformed DataFrame without duplicated columns." fit_transform: description: "Fit and transform the input DataFrame." parameters: properties: df: type: DataFrame - description: "The input DataFrame." + description: "The input DataFrame without duplicated columns." required: - df returns: @@ -224,7 +228,7 @@ CatCross: properties: cols: type: list - description: "Columns to be pairwise crossed." + description: "Columns to be pairwise crossed, at least 2 columns." max_cat_num: type: int description: "Maximum unique categories per crossed feature." @@ -430,4 +434,115 @@ GeneralSelection: returns: df: type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file + description: "The transformed DataFrame." + + +TreeBasedSelection: + type: class + description: "Select features based on tree-based model and remove features with low importance." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + label_col: + type: str + description: "Label column name." + task_type: + type: str + description: "Task type, 'cls' for classification, 'mcls' for multi-class classification, 'reg' for regression." + enum: + - cls + - mcls + - reg + required: + - label_col + - task_type + fit: + description: "Fit the TreeBasedSelection model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." + +VarianceBasedSelection: + type: class + description: "Select features based on variance and remove features with low variance." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + label_col: + type: str + description: "Label column name." + threshold: + type: float + description: "Threshold for variance." + default: 0.0 + required: + - label_col + fit: + description: "Fit the VarianceBasedSelection model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." \ No newline at end of file From bb7f4c33105e0a020c8249fb8477c0b3365b1fb4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 21 Dec 2023 17:16:33 +0800 Subject: [PATCH 222/637] update code prompt for make tools. --- metagpt/actions/write_analysis_code.py | 13 +++++++++++-- metagpt/roles/ml_engineer.py | 7 +++---- 2 files changed, 14 insertions(+), 6 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 099934c5a..c9acb32b9 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -270,9 +270,18 @@ class MakeTools(WriteCodeByGenerate): saved_path.write_text(tool_code, encoding='utf-8') @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) - async def run(self, code_message: List[Message | Dict], **kwargs) -> str: - msgs = self.process_msg(code_message, self.DEFAULT_SYSTEM_MSG) + async def run(self, code: str, code_desc: str = None, **kwargs) -> str: + # 拼接code prompt + code_prompt = f"The following code is about {code_desc}, convert it to be a General Function, {code}" + msgs = self.process_msg(code_prompt, self.DEFAULT_SYSTEM_MSG) logger.info(f"\n\nAsk to Make tools:\n{'-'*60}\n {msgs[-1]}") + + # 更新kwargs + if 'code' in kwargs: + kwargs.pop('code') + if 'code_desc' in kwargs: + kwargs.pop('code_desc') + tool_code = await self.llm.aask_code(msgs, **kwargs) max_tries, current_try = 3, 1 func_name = self.parse_function_name(tool_code['code']) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index f44d42554..db2dfeeff 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -291,15 +291,14 @@ class MLEngineer(Role): logger.warning(f"Making tools for task_id {self.plan.current_task_id}: \ `{self.plan.current_task.instruction}` \n code: \n {code}") make_tools = MakeTools() - code_prompt = f"The following code is about {self.plan.current_task.instruction},\ - convert it to be a General Function, {code}" - tool_code = await make_tools.run(code_prompt) + tool_code = await make_tools.run(code, self.plan.current_task.instruction) # check tool_code by execute_code logger.info(f"Checking task_id {self.plan.current_task_id} tool code by executor...") _, success = await self.execute_code.run(tool_code) make_tool_retries, make_tool_current_retry = 3, 1 while not success: - tool_code = await make_tools.run(code_prompt) + # tool_code = await make_tools.run(code_prompt) + tool_code = await make_tools.run(code) _, success = await self.execute_code.run(tool_code) if make_tool_current_retry > make_tool_retries: logger.error(f"We have tried the maximum number of attempts {make_tool_retries}\ From e0903fe51f3838e57f05f3f4976b027f3366ef7d Mon Sep 17 00:00:00 2001 From: lidanyang Date: Thu, 21 Dec 2023 17:45:50 +0800 Subject: [PATCH 223/637] refine ml prompt --- metagpt/prompts/ml_engineer.py | 26 ++++++++++++++++---------- metagpt/roles/ml_engineer.py | 7 ++++--- 2 files changed, 20 insertions(+), 13 deletions(-) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 33eb9c40c..ff446281c 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -6,7 +6,7 @@ # @Desc : UPDATE_DATA_COLUMNS = """ # Background -Keep dataset column information updated to reflect changes in training or testing datasets, aiding in informed decision-making during data analysis. +Keep dataset column information updated before model train. ## Done Tasks ```python {history_code} @@ -18,15 +18,13 @@ Update and print the dataset's column information only if the train or test data from metagpt.tools.functions.libs.data_preprocess import get_column_info column_info = get_column_info(df) -print("df_column_info") +print("column_info") print(column_info) ```end # Constraints: - Use the DataFrame variable from 'Done Tasks' in place of df. - Import `get_column_info` only if it's not already imported. -- Skip update if no changes in training/testing data, except for initial data load. -- No need to update info if only model evaluation is performed. """ GEN_DATA_DESC_PROMPT = """ @@ -185,7 +183,7 @@ ojb_cols = train.select_dtypes(include='object').columns.tolist() for col in obj_cols: encoder = LabelEncoder() - train[col] = encoder.fit_transform(train[col]) + train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown']) test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown') test[col] = encoder.transform(test[col]) @@ -241,6 +239,8 @@ from metagpt.tools.functions.libs.data_preprocess import FillMissingValue train_processed = train.copy() test_processed = test.copy() num_cols = train_processed.select_dtypes(include='number').columns.tolist() +if 'label' in num_cols: + num_cols.remove('label') fill_missing_value = FillMissingValue(features=num_cols, strategy='mean') fill_missing_value.fit(train_processed) train_processed = fill_missing_value.transform(train_processed) @@ -266,23 +266,29 @@ The current task is about data preprocessing, please note the following: - Monitor data types per column, applying appropriate methods. - Ensure operations are on existing dataset columns. - Avoid writing processed data to files. +- Avoid any change to label column, such as standardization, etc. - Prefer alternatives to one-hot encoding for categorical data. -- Only encode necessary categorical columns to allow for potential feature-specific engineering tasks later. +- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later. +- Each step do data preprocessing to train, must do same for test separately at the same time. """ FEATURE_ENGINEERING_PROMPT = """ The current task is about feature engineering. when performing it, please adhere to the following principles: -- Ensure operations are on existing dataset columns and consider the data type (numerical, categorical, etc.) and application scenario (classification, regression tasks, etc.). -- Create impactful features based on real-world knowledge and column info. -- Generate as diverse features as possible to improve the model's performance. +- Generate as diverse features as possible to improve the model's performance step-by-step. - If potential impactful features are not included in 'Code Steps', add new steps to generate them. +- Avoid creating redundant or excessively numerous features in one step. +- Exclude ID columns from feature generation and remove them. +- Each step do feature engineering to train, must do same for test separately at the same time. +- Avoid using the label column to create features, except for cat encoding. +- Use the data from previous task result if exist, do not mock or reload data yourself. """ MODEL_TRAIN_PROMPT = """ The current task is about training a model, please ensure high performance: - Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as lightGBM, XGBoost, CatBoost, etc. -- Before training, first check not is_numeric_dtype columns and use label encoding to convert them to numeric columns. +- If non-numeric columns exist, perform label encode together with all steps. - Use the data from previous task result directly, do not mock or reload data yourself. +- Set suitable hyperparameters for the model, make metrics as high as possible. """ MODEL_EVALUATE_PROMPT = """ diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 73aba1fe8..8ad7f43c9 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -80,8 +80,8 @@ class MLEngineer(Role): task.result = result self.plan.finish_current_task() self.working_memory.clear() - - if self.use_tools: + + if self.use_tools and task.task_type not in ['model_train', 'model_evaluate']: success, new_code = await self._update_data_columns() if success: task.code = task.code + "\n\n" + new_code @@ -120,6 +120,7 @@ class MLEngineer(Role): if is_update: result, success = await self.execute_code.run(code) if success: + print(result) self.data_desc["column_info"] = result return success, code @@ -269,7 +270,7 @@ if __name__ == "__main__": # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" - # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." + # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report AUC Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." From 7806013dcebf611d26581d170c4e7c2fb7ee673a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Fri, 22 Dec 2023 14:07:26 +0800 Subject: [PATCH 224/637] update: use WriteCodeByGenerate conditions. --- metagpt/roles/ml_engineer.py | 15 ++++++++------- 1 file changed, 8 insertions(+), 7 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index db2dfeeff..c2df4bb79 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -149,7 +149,8 @@ class MLEngineer(Role): ) logger.info(f"new code \n{code}") cause_by = DebugCode - elif not self.use_tools or self.plan.current_task.task_type == 'other': + elif (not self.use_tools and not self.use_udfs) or ( + self.plan.current_task.task_type == 'other' and not self.use_udfs): logger.info("Write code with pure generation") # TODO: 添加基于current_task.instruction-code_path的k-v缓存 code = await WriteCodeByGenerate().run( @@ -326,12 +327,12 @@ if __name__ == "__main__": role.use_udfs = False await role.run(requirement) # use udfs - # role.reset() - # role.make_udfs = False - # role.use_udfs = True - # role.use_code_steps = False - # role.use_tools = False - # await role.run(requirement) + role.reset() + role.make_udfs = False + role.use_udfs = True + role.use_code_steps = False + role.use_tools = False + await role.run(requirement) # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." From 3d3c4d56d759673a65a0273b310473c50e4e6ca9 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 21 Nov 2023 14:38:19 +0800 Subject: [PATCH 225/637] add increment development function --- metagpt/actions/__init__.py | 1 + metagpt/actions/refine.py | 10 + metagpt/actions/refine_design_api.py | 231 ++++++++++++++++ metagpt/actions/refine_prd.py | 260 +++++++++++++++++++ metagpt/actions/refine_project_management.py | 208 +++++++++++++++ 5 files changed, 710 insertions(+) create mode 100644 metagpt/actions/refine.py create mode 100644 metagpt/actions/refine_design_api.py create mode 100644 metagpt/actions/refine_prd.py create mode 100644 metagpt/actions/refine_project_management.py diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index c34c72ed2..99a4175f6 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -14,6 +14,7 @@ from metagpt.actions.debug_error import DebugError from metagpt.actions.design_api import WriteDesign from metagpt.actions.design_api_review import DesignReview from metagpt.actions.project_management import AssignTasks, WriteTasks +from metagpt.actions.refine import Refine from metagpt.actions.research import CollectLinks, WebBrowseAndSummarize, ConductResearch from metagpt.actions.run_code import RunCode from metagpt.actions.search_and_summarize import SearchAndSummarize diff --git a/metagpt/actions/refine.py b/metagpt/actions/refine.py new file mode 100644 index 000000000..beea40fc8 --- /dev/null +++ b/metagpt/actions/refine.py @@ -0,0 +1,10 @@ +from metagpt.actions import Action + + +# 增量开发动作的基类 +class Refine(Action): + def __init__(self, name="Refine", context=None, llm=None): + super().__init__(name, context, llm) + + def run(self, *args, **kwargs): + raise NotImplementedError diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py new file mode 100644 index 000000000..591db175a --- /dev/null +++ b/metagpt/actions/refine_design_api.py @@ -0,0 +1,231 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +import shutil +from pathlib import Path +from typing import List + +from metagpt.actions import Action, ActionOutput +from metagpt.config import CONFIG +from metagpt.const import WORKSPACE_ROOT +from metagpt.logs import logger +from metagpt.utils.common import CodeParser +from metagpt.utils.get_template import get_template +from metagpt.utils.json_to_markdown import json_to_markdown +from metagpt.utils.mermaid import mermaid_to_file + +templates = { + "json": { + "PROMPT_TEMPLATE": """ +# Context +{context} + +## Legacy Design +{legacy} + +## Format example +{format_example} +----- +Role: You are an architect; the goal is to design a SOTA PEP8-compliant python system; make the best use of good open source tools +Requirement: Fill in the following missing information based on the context, each section name is a key in json +Max Output: 8192 chars or 2048 tokens. Try to use them up. + +## Difference Description: Provided as Python list[str], the list of differences between the new design and the legacy design + +## Implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. + +## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores + +## File list: Provided as Python list[str], the list of ONLY REQUIRED files needed to write the program(LESS IS MORE!). Only need relative paths, comply with PEP8 standards. ALWAYS write a main.py or app.py here + +## Data structures and interface definitions: Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. + +## Program call flow: Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. + +## Anything UNCLEAR: Provide as Plain text. Make clear here. + +output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, +and only output the json inside this tag, nothing else +""", + "FORMAT_EXAMPLE": """ +[CONTENT] +{ + "Difference Description": ["The ..."], + "Implementation approach": "We will ...", + "Python package name": "snake_game", + "File list": ["main.py"], + "Data structures and interface definitions": ' + classDiagram + class Game{ + +int score + } + ... + Game "1" -- "1" Food: has + ', + "Program call flow": ' + sequenceDiagram + participant M as Main + ... + G->>M: end game + ', + "Anything UNCLEAR": "The requirement is clear to me." +} +[/CONTENT] +""", + }, + "markdown": { + "PROMPT_TEMPLATE": """ +# Context +{context} + +## Legacy Design +{legacy} + +## Format example +{format_example} +----- +Role: You are an architect; the goal is to design a SOTA PEP8-compliant python system; make the best use of good open source tools +Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately +Max Output: 8192 chars or 2048 tokens. Try to use them up. +Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. + +## Difference Description: Provided as Python list[str], the list of differences between the new design and the legacy design + +## Implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. + +## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores + +## File list: Provided as Python list[str], the list of ONLY REQUIRED files needed to write the program(LESS IS MORE!). Only need relative paths, comply with PEP8 standards. ALWAYS write a main.py or app.py here + +## Data structures and interface definitions: Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. + +## Program call flow: Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. + +## Anything UNCLEAR: Provide as Plain text. Make clear here. + +""", + "FORMAT_EXAMPLE": """ +--- +## Difference Description +```python +[ + "The ...", +] +``` + +## Implementation approach +We will ... + +## Python package name +```python +"snake_game" +``` + +## File list +```python +[ + "main.py", +] +``` + +## Data structures and interface definitions +```mermaid +classDiagram + class Game{ + +int score + } + ... + Game "1" -- "1" Food: has +``` + +## Program call flow +```mermaid +sequenceDiagram + participant M as Main + ... + G->>M: end game +``` + +## Anything UNCLEAR +The requirement is clear to me. +--- +""", + }, +} + +OUTPUT_MAPPING = { + "Difference Description": (List[str], ...), + "Implementation approach": (str, ...), + "Python package name": (str, ...), + "File list": (List[str], ...), + "Data structures and interface definitions": (str, ...), + "Program call flow": (str, ...), + "Anything UNCLEAR": (str, ...), +} + + +class RefineDesign(Action): + def __init__(self, name, context=None, llm=None): + super().__init__(name, context, llm) + self.desc = ( + "Based on the PRD, think about the system design, and design the corresponding APIs, " + "data structures, library tables, processes, and paths. Please provide your design, feedback " + "clearly and in detail." + ) + + def recreate_workspace(self, workspace: Path): + try: + shutil.rmtree(workspace) + except FileNotFoundError: + pass # Folder does not exist, but we don't care + workspace.mkdir(parents=True, exist_ok=True) + + async def _save_prd(self, docs_path, resources_path, context): + prd_file = docs_path / "prd.md" + if context[-1].instruct_content and context[-1].instruct_content.dict()["Competitive Quadrant Chart"]: + quadrant_chart = context[-1].instruct_content.dict()["Competitive Quadrant Chart"] + await mermaid_to_file(quadrant_chart, resources_path / "competitive_analysis") + + if context[-1].instruct_content: + logger.info(f"Saving PRD to {prd_file}") + prd_file.write_text(json_to_markdown(context[-1].instruct_content.dict())) + + async def _save_system_design(self, docs_path, resources_path, system_design): + data_api_design = system_design.instruct_content.dict()[ + "Data structures and interface definitions" + ] # CodeParser.parse_code(block="Data structures and interface definitions", text=content) + seq_flow = system_design.instruct_content.dict()[ + "Program call flow" + ] # CodeParser.parse_code(block="Program call flow", text=content) + await mermaid_to_file(data_api_design, resources_path / "data_api_design") + await mermaid_to_file(seq_flow, resources_path / "seq_flow") + system_design_file = docs_path / "system_design.md" + logger.info(f"Saving System Designs to {system_design_file}") + system_design_file.write_text((json_to_markdown(system_design.instruct_content.dict()))) + + async def _save(self, context, system_design): + if isinstance(system_design, ActionOutput): + ws_name = system_design.instruct_content.dict()["Python package name"] + else: + ws_name = CodeParser.parse_str(block="Python package name", text=system_design) + workspace = WORKSPACE_ROOT / ws_name + self.recreate_workspace(workspace) + docs_path = workspace / "docs" + resources_path = workspace / "resources" + docs_path.mkdir(parents=True, exist_ok=True) + resources_path.mkdir(parents=True, exist_ok=True) + await self._save_prd(docs_path, resources_path, context) + await self._save_system_design(docs_path, resources_path, system_design) + + async def run(self, context, legacy, format=CONFIG.prompt_format): + prompt_template, format_example = get_template(templates, format) + prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) + # system_design = await self._aask(prompt) + system_design = await self._aask_v1(prompt, "system_design", OUTPUT_MAPPING, format=format) + # fix Python package name, we can't system_design.instruct_content.python_package_name = "xxx" since "Python package name" contain space, have to use setattr + setattr( + system_design.instruct_content, + "Python package name", + system_design.instruct_content.dict()["Python package name"].strip().strip("'").strip('"'), + ) + await self._save(context, system_design) + return system_design diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py new file mode 100644 index 000000000..1e2709c6b --- /dev/null +++ b/metagpt/actions/refine_prd.py @@ -0,0 +1,260 @@ +from typing import List + +from metagpt.actions import Refine, ActionOutput, SearchAndSummarize +from metagpt.config import CONFIG +from metagpt.logs import logger +from metagpt.utils.get_template import get_template + +increment_template = { + "json": { + "PROMPT_TEMPLATE": """ +# Context +## User's New Requirements +{new_requirements} + +## Difference Description +{difference_description} + +## Legacy PRD +{legacy} + +## Search Information +{search_information} + +## mermaid quadrantChart code syntax example. DONT USE QUOTO IN CODE DUE TO INVALID SYNTAX. Replace the with REAL COMPETITOR NAME +```mermaid +quadrantChart + title Reach and engagement of campaigns + x-axis Low Reach --> High Reach + y-axis Low Engagement --> High Engagement + quadrant-1 We should expand + quadrant-2 Need to promote + quadrant-3 Re-evaluate + quadrant-4 May be improved + "Campaign: A": [0.3, 0.6] + "Campaign B": [0.45, 0.23] + "Campaign C": [0.57, 0.69] + "Campaign D": [0.78, 0.34] + "Campaign E": [0.40, 0.34] + "Campaign F": [0.35, 0.78] + "Our Target Product": [0.5, 0.6] +``` + +## Format example +{format_example} +----- +Role: You are a professional product manager; the goal is to design a concise, usable, efficient product based on the new requirements, the difference description and Legacy PRD. +Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design + +## New Requirements: Provide as Plain text and place the new requirements here + +## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple + +## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple + +## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less + +## Competitive Analysis: Provided as Python list[str], up to 7 competitive product analyses, consider as similar competitors as possible + +## Competitive Quadrant Chart: Use mermaid quadrantChart code syntax. up to 14 competitive products. Translation: Distribute these competitor scores evenly between 0 and 1, trying to conform to a normal distribution centered around 0.5 as much as possible. + +## Requirement Analysis: Provide as Plain text. Be simple. LESS IS MORE. Make your requirements less dumb. Delete the parts unnessasery. + +## Requirement Pool: Provided as Python list[list[str], the parameters are requirement description, priority(P0/P1/P2), respectively, comply with PEP standards; no more than 5 requirements and consider to make its difficulty lower + +## UI Design draft: Provide as Plain text. Be simple. Describe the elements and functions, also provide a simple style description and layout description. +## Anything UNCLEAR: Provide as Plain text. Make clear here. + +output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, +and only output the json inside this tag, nothing else +""", + "FORMAT_EXAMPLE": """ +[CONTENT] +{ + "New Requirements": "", + "Difference Description": "", + "Search Information": "", + "Requirements": "", + "Product Goals": [], + "User Stories": [], + "Competitive Analysis": [], + "Competitive Quadrant Chart": "quadrantChart + title Reach and engagement of campaigns + x-axis Low Reach --> High Reach + y-axis Low Engagement --> High Engagement + quadrant-1 We should expand + quadrant-2 Need to promote + quadrant-3 Re-evaluate + quadrant-4 May be improved + Campaign A: [0.3, 0.6] + Campaign B: [0.45, 0.23] + Campaign C: [0.57, 0.69] + Campaign D: [0.78, 0.34] + Campaign E: [0.40, 0.34] + Campaign F: [0.35, 0.78]", + "Requirement Analysis": "", + "Requirement Pool": [["P0","P0 requirement"],["P1","P1 requirement"]], + "UI Design draft": "", + "Anything UNCLEAR": "", +} +[/CONTENT] +""", + }, + "markdown": { + "PROMPT_TEMPLATE": """ +# Context +You need to refine the requirements based on the new requirements and the existing requirements' output. +## User's New Requirements +{new_requirements} + +## Difference Description +{difference_description} + +## Legacy PRD +{legacy} + +## Search Information +{search_information} + +## mermaid quadrantChart code syntax example. DONT USE QUOTO IN CODE DUE TO INVALID SYNTAX. Replace the with REAL COMPETITOR NAME +```mermaid +quadrantChart + title Reach and engagement of campaigns + x-axis Low Reach --> High Reach + y-axis Low Engagement --> High Engagement + quadrant-1 We should expand + quadrant-2 Need to promote + quadrant-3 Re-evaluate + quadrant-4 May be improved + "Campaign: A": [0.3, 0.6] + "Campaign B": [0.45, 0.23] + "Campaign C": [0.57, 0.69] + "Campaign D": [0.78, 0.34] + "Campaign E": [0.40, 0.34] + "Campaign F": [0.35, 0.78] + "Our Target Product": [0.5, 0.6] +``` + +## Format example +{format_example} +----- +Role: You are a professional product manager; the goal is to design a concise, usable, efficient product +Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format. + +## New Requirements: Provide as Plain text and place the new requirements here + +## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple + +## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple + +## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less + +## Competitive Analysis: Provided as Python list[str], up to 7 competitive product analyses, consider as similar competitors as possible + +## Competitive Quadrant Chart: Use mermaid quadrantChart code syntax. up to 14 competitive products. Translation: Distribute these competitor scores evenly between 0 and 1, trying to conform to a normal distribution centered around 0.5 as much as possible. + +## Requirement Analysis: Provide as Plain text. Be simple. LESS IS MORE. Make your requirements less dumb. Delete the parts unnessasery. + +## Requirement Pool: Provided as Python list[list[str], the parameters are requirement description, priority(P0/P1/P2), respectively, comply with PEP standards; no more than 5 requirements and consider to make its difficulty lower + +## UI Design draft: Provide as Plain text. Be simple. Describe the elements and functions, also provide a simple style description and layout description. +## Anything UNCLEAR: Provide as Plain text. Make clear here. +""", + "FORMAT_EXAMPLE": """ +--- +## New Requirements +The boss ... + +## Difference Description +```python +[ + "...", +] + +## Product Goals +```python +[ + "Create a ...", +] +``` + +## User Stories +```python +[ + "As a user, ...", +] +``` + +## Competitive Analysis +```python +[ + "Python Snake Game: ...", +] +``` + +## Competitive Quadrant Chart +```mermaid +quadrantChart + title Reach and engagement of campaigns + ... + "Our Target Product": [0.6, 0.7] +``` + +## Requirement Analysis +The product should be a ... + +## Requirement Pool +```python +[ + ["End game ...", "P0"] +] +``` + +## UI Design draft +Give a basic function description, and a draft + +## Anything UNCLEAR +There are no unclear points. +--- +""", + }, +} + +INCREMENT_OUTPUT_MAPPING = { + "New Requirements": (str, ...), + # "Major Enhancements": (List[str], ...), + "Difference Description": (List[str], ...), + "Product Goals": (List[str], ...), + "User Stories": (List[str], ...), + "Competitive Analysis": (List[str], ...), + "Competitive Quadrant Chart": (str, ...), + "Requirement Analysis": (str, ...), + "Requirement Pool": (List[List[str]], ...), + "UI Design draft": (str, ...), + "Anything UNCLEAR": (str, ...), +} + + +# 对于产品经理,增量开发的动作是:RefinePDR,输出是结合新需求和已有需求输出的新的PDR +class RefinePRD(Refine): + + def __init__(self, name="RefinePRD", context=None, llm=None): + super().__init__(name, context, llm) + + async def run(self, new_requirements, difference_description, legacy, format=CONFIG.prompt_format, *args, **kwargs): + sas = SearchAndSummarize() + rsp = "" + info = f"### Search Results\n{sas.result}\n\n### Search Summary\n{rsp}" + if sas.result: + logger.info(sas.result) + logger.info(rsp) + + prompt_template, format_example = get_template(increment_template, format) + prompt = prompt_template.format( + new_requirements=new_requirements, difference_description=difference_description, legacy=legacy, search_information=info, + format_example=format_example + ) + logger.debug(prompt) + prd = await self._aask_v1(prompt, "prd", INCREMENT_OUTPUT_MAPPING, format=format) + return prd diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py new file mode 100644 index 000000000..354996f3e --- /dev/null +++ b/metagpt/actions/refine_project_management.py @@ -0,0 +1,208 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +from typing import List + +from metagpt.actions.action import Action +from metagpt.config import CONFIG +from metagpt.const import WORKSPACE_ROOT +from metagpt.utils.common import CodeParser +from metagpt.utils.get_template import get_template +from metagpt.utils.json_to_markdown import json_to_markdown + +templates = { + "json": { + "PROMPT_TEMPLATE": """ +# Context +{context} + +## Legacy Design +{legacy} + +## Format example +{format_example} +----- +Role: You are a project manager; the goal is to break down tasks according to PRD/technical design, give a task list, and analyze task dependencies to start with the prerequisite modules +Requirements: Based on the context, fill in the following missing information, each section name is a key in json. Here the granularity of the task is a file, if there are any missing files, you can supplement them +Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. + +## Required Python third-party packages: Provided in requirements.txt format + +## Required Other language third-party packages: Provided in requirements.txt format + +## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. + +## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first + +## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first + +## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. + +## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. + +## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. + +output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, +and only output the json inside this tag, nothing else +""", + "FORMAT_EXAMPLE": ''' +{ + "Required Python third-party packages": [ + "flask==1.1.2", + "bcrypt==3.2.0" + ], + "Required Other language third-party packages": [ + "No third-party ..." + ], + "Full API spec": """ + openapi: 3.0.0 + ... + description: A JSON object ... + """, + "Logic Analysis": [ + ["game.py","Contains..."] + ], + "Task list": [ + "game.py" + ], + "Shared Knowledge": """ + 'game.py' contains ... + """, + "Difference Description": """ + The ... + """, + "Anything UNCLEAR": "We need ... how to start." +} +''', + }, + "markdown": { + "PROMPT_TEMPLATE": """ +# Context +{context} + +## Legacy Design +{legacy} + +## Format example +{format_example} +----- +Role: You are a project manager; the goal is to break down tasks according to PRD/technical design, give a task list, and analyze task dependencies to start with the prerequisite modules +Requirements: Based on the context, fill in the following missing information, note that all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file, if there are any missing files, you can supplement them +Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. + +## Required Python third-party packages: Provided in requirements.txt format + +## Required Other language third-party packages: Provided in requirements.txt format + +## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. + +## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first + +## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first + +## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. + +## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. + +## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. + +""", + "FORMAT_EXAMPLE": ''' +--- +## Required Python third-party packages +```python +""" +flask==1.1.2 +bcrypt==3.2.0 +""" +``` + +## Required Other language third-party packages +```python +""" +No third-party ... +""" +``` + +## Full API spec +```python +""" +openapi: 3.0.0 +... +description: A JSON object ... +""" +``` + +## Logic Analysis +```python +[ + ["game.py", "Contains ..."], +] +``` + +## Task list +```python +[ + "game.py", +] +``` + +## Shared Knowledge +```python +""" +'game.py' contains ... +""" +``` + +## Difference Description +```python +""" +The ... +""" +``` + +## Anything UNCLEAR +We need ... how to start. +--- +''', + }, +} +OUTPUT_MAPPING = { + "Required Python third-party packages": (List[str], ...), + "Required Other language third-party packages": (List[str], ...), + "Full API spec": (str, ...), + "Logic Analysis": (List[List[str]], ...), + "Task list": (List[str], ...), + "Shared Knowledge": (str, ...), + "Difference Description": (str, ...), + "Anything UNCLEAR": (str, ...), +} + + +class RefineTasks(Action): + def __init__(self, name="CreateTasks", context=None, llm=None): + super().__init__(name, context, llm) + + def _save(self, context, rsp): + if context[-1].instruct_content: + ws_name = context[-1].instruct_content.dict()["Python package name"] + else: + ws_name = CodeParser.parse_str(block="Python package name", text=context[-1].content) + file_path = WORKSPACE_ROOT / ws_name / "docs/api_spec_and_tasks.md" + file_path.write_text(json_to_markdown(rsp.instruct_content.dict())) + + # Write requirements.txt + requirements_path = WORKSPACE_ROOT / ws_name / "requirements.txt" + requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Required Python third-party packages"))) + + async def run(self, context, legacy, format=CONFIG.prompt_format): + prompt_template, format_example = get_template(templates, format) + prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) + rsp = await self._aask_v1(prompt, "task", OUTPUT_MAPPING, format=format) + self._save(context, rsp) + return rsp + + +class AssignTasks(Action): + async def run(self, *args, **kwargs): + # Here you should implement the actual action + pass From 9aa745291767ea53c3796526375a082e47a6d141 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 24 Nov 2023 18:22:06 +0800 Subject: [PATCH 226/637] update increment development, add bug fix function --- metagpt/actions/refine_design_api.py | 25 +++++- metagpt/actions/refine_prd.py | 17 ++++- metagpt/actions/refine_project_management.py | 34 ++++----- metagpt/actions/write_code_refine.py | 80 ++++++++++++++++++++ 4 files changed, 134 insertions(+), 22 deletions(-) create mode 100644 metagpt/actions/write_code_refine.py diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py index 591db175a..f1c231525 100644 --- a/metagpt/actions/refine_design_api.py +++ b/metagpt/actions/refine_design_api.py @@ -1,5 +1,6 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- +import re import shutil from pathlib import Path from typing import List @@ -25,7 +26,7 @@ templates = { ## Format example {format_example} ----- -Role: You are an architect; the goal is to design a SOTA PEP8-compliant python system; make the best use of good open source tools +Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and the provided difference descriptions. Make the best use of good open source tools. Requirement: Fill in the following missing information based on the context, each section name is a key in json Max Output: 8192 chars or 2048 tokens. Try to use them up. @@ -83,7 +84,7 @@ and only output the json inside this tag, nothing else ## Format example {format_example} ----- -Role: You are an architect; the goal is to design a SOTA PEP8-compliant python system; make the best use of good open source tools +Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and the provided difference descriptions. Make the best use of good open source tools. Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately Max Output: 8192 chars or 2048 tokens. Try to use them up. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. @@ -179,6 +180,25 @@ class RefineDesign(Action): pass # Folder does not exist, but we don't care workspace.mkdir(parents=True, exist_ok=True) + def create_or_increment_workspace(self, workspace: Path): + # 如果工作空间已存在,添加数字以区分 + original_workspace = workspace + index = 1 + while workspace.exists(): + ws_name_match = re.match(r'^(.*)_([\d]+)$', original_workspace.name) + if ws_name_match: + base_name, existing_index = ws_name_match.groups() + index = int(existing_index) + index += 1 + workspace = original_workspace.parent / f"{base_name}_{index}" + else: + workspace = original_workspace.parent / f"{original_workspace.name}_{index}" + index += 1 + + # 创建工作空间,包括所有必要的父文件夹 + workspace.mkdir(parents=True, exist_ok=True) + return workspace + async def _save_prd(self, docs_path, resources_path, context): prd_file = docs_path / "prd.md" if context[-1].instruct_content and context[-1].instruct_content.dict()["Competitive Quadrant Chart"]: @@ -208,6 +228,7 @@ class RefineDesign(Action): else: ws_name = CodeParser.parse_str(block="Python package name", text=system_design) workspace = WORKSPACE_ROOT / ws_name + # workspace = self.create_or_increment_workspace(workspace) self.recreate_workspace(workspace) docs_path = workspace / "docs" resources_path = workspace / "resources" diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py index 1e2709c6b..81cb02af8 100644 --- a/metagpt/actions/refine_prd.py +++ b/metagpt/actions/refine_prd.py @@ -43,13 +43,15 @@ quadrantChart ## Format example {format_example} ----- -Role: You are a professional product manager; the goal is to design a concise, usable, efficient product based on the new requirements, the difference description and Legacy PRD. +Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design ## New Requirements: Provide as Plain text and place the new requirements here ## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple +## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple + ## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple ## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less @@ -72,9 +74,10 @@ and only output the json inside this tag, nothing else [CONTENT] { "New Requirements": "", - "Difference Description": "", + "Difference Description": [], "Search Information": "", "Requirements": "", + "Incremental Development Plan": [], "Product Goals": [], "User Stories": [], "Competitive Analysis": [], @@ -138,7 +141,7 @@ quadrantChart ## Format example {format_example} ----- -Role: You are a professional product manager; the goal is to design a concise, usable, efficient product +Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format. @@ -146,6 +149,8 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD W ## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple +## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple + ## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple ## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less @@ -172,6 +177,11 @@ The boss ... "...", ] +## Incremental Development Plan +[ + "It ...", +] + ## Product Goals ```python [ @@ -225,6 +235,7 @@ INCREMENT_OUTPUT_MAPPING = { "New Requirements": (str, ...), # "Major Enhancements": (List[str], ...), "Difference Description": (List[str], ...), + "Incremental Development Plan": (List[str], ...), "Product Goals": (List[str], ...), "User Stories": (List[str], ...), "Competitive Analysis": (List[str], ...), diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py index 354996f3e..7d9e118d8 100644 --- a/metagpt/actions/refine_project_management.py +++ b/metagpt/actions/refine_project_management.py @@ -21,7 +21,7 @@ templates = { ## Format example {format_example} ----- -Role: You are a project manager; the goal is to break down tasks according to PRD/technical design, give a task list, and analyze task dependencies to start with the prerequisite modules +Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a task list, and analyze task dependencies to start with the prerequisite modules. Requirements: Based on the context, fill in the following missing information, each section name is a key in json. Here the granularity of the task is a file, if there are any missing files, you can supplement them Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. @@ -31,14 +31,14 @@ Attention: Use '##' to split sections, not '#', and '## ' SHOULD W ## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. +## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. + ## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first ## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first ## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. -## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. - ## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, @@ -58,6 +58,9 @@ and only output the json inside this tag, nothing else ... description: A JSON object ... """, + "Difference Description": """ + The ... + """, "Logic Analysis": [ ["game.py","Contains..."] ], @@ -67,9 +70,6 @@ and only output the json inside this tag, nothing else "Shared Knowledge": """ 'game.py' contains ... """, - "Difference Description": """ - The ... - """, "Anything UNCLEAR": "We need ... how to start." } ''', @@ -85,7 +85,7 @@ and only output the json inside this tag, nothing else ## Format example {format_example} ----- -Role: You are a project manager; the goal is to break down tasks according to PRD/technical design, give a task list, and analyze task dependencies to start with the prerequisite modules +Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a task list, and analyze task dependencies to start with the prerequisite modules. Requirements: Based on the context, fill in the following missing information, note that all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file, if there are any missing files, you can supplement them Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. @@ -95,14 +95,14 @@ Attention: Use '##' to split sections, not '#', and '## ' SHOULD W ## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. +## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. + ## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first ## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first ## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. -## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. - ## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. """, @@ -132,6 +132,13 @@ description: A JSON object ... """ ``` +## Difference Description +```python +""" +The ... +""" +``` + ## Logic Analysis ```python [ @@ -153,13 +160,6 @@ description: A JSON object ... """ ``` -## Difference Description -```python -""" -The ... -""" -``` - ## Anything UNCLEAR We need ... how to start. --- @@ -170,10 +170,10 @@ OUTPUT_MAPPING = { "Required Python third-party packages": (List[str], ...), "Required Other language third-party packages": (List[str], ...), "Full API spec": (str, ...), + "Difference Description": (str, ...), "Logic Analysis": (List[List[str]], ...), "Task list": (List[str], ...), "Shared Knowledge": (str, ...), - "Difference Description": (str, ...), "Anything UNCLEAR": (str, ...), } diff --git a/metagpt/actions/write_code_refine.py b/metagpt/actions/write_code_refine.py new file mode 100644 index 000000000..527952aed --- /dev/null +++ b/metagpt/actions/write_code_refine.py @@ -0,0 +1,80 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +from metagpt.actions.action import Action +from metagpt.logs import logger +from metagpt.schema import Message +from metagpt.utils.common import CodeParser +from tenacity import retry, stop_after_attempt, wait_fixed + + +PROMPT_TEMPLATE = """ +NOTICE +Role: You are a professional software engineer, and your main task is to conduct incremental development, which includes reviewing existing code, providing modification suggestions, rewriting code, and optimizing the codebase. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". + +## Code Review: Based on the following context and legacy code, and following the checklist, provide key, clear, concise, and specific code modification suggestions, up to 5. +``` +1. Check 0: Is the code implemented as per the requirements? +2. Check 1: Are there any issues with the code logic? +3. Check 2: Does the existing code follow the "Data structures and interface definitions"? +4. Check 3: Is there a function in the code that is omitted or not fully implemented that needs to be implemented? +5. Check 4: Does the code have unnecessary or lack dependencies? +``` + +## Incremental Development: {filename} Based on the findings from the "Code Review," context, and the legacy code, conduct incremental development by rewriting, optimizing, and adding new code using triple quotes. +----- +# Context +{context} + +## Legacy Code +You are tasked with conducting incremental development in the existing code and creating a new code file, {filename}, based on the provided legacy code and above information. +``` +{code} +``` +----- + +## Format example +----- +{format_example} +----- + +""" + +FORMAT_EXAMPLE = """ + +## Code Review +1. The code ... +2. ... +3. ... +4. ... +5. ... + +## Incremental Development: {filename} +```python +## {filename} - Incremental Development +... +``` +""" + + + +class WriteCodeRefine(Action): + def __init__(self, name="WriteCodeRefine", context: list[Message] = None, llm=None): + super().__init__(name, context, llm) + + @retry(stop=stop_after_attempt(2), wait=wait_fixed(1)) + async def write_code(self, prompt): + code_rsp = await self._aask(prompt) + code = CodeParser.parse_code(block="", text=code_rsp) + return code + + async def run(self, context, code, filename): + format_example = FORMAT_EXAMPLE.format(filename=filename) + prompt = PROMPT_TEMPLATE.format(context=context, code=code, filename=filename, format_example=format_example) + logger.info(f'Code refine {filename}..') + code = await self.write_code(prompt) + # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) + # self._save(context, filename, code) + return code + \ No newline at end of file From 64ebf02dcec25a13bc3dbf65bb3f957026f939ac Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 29 Nov 2023 17:31:15 +0800 Subject: [PATCH 227/637] update increment development, add write_code_guide.py file --- metagpt/actions/refine_design_api.py | 73 +++---- metagpt/actions/refine_prd.py | 192 ++----------------- metagpt/actions/refine_project_management.py | 118 +++++------- metagpt/actions/write_code_guide.py | 67 +++++++ metagpt/actions/write_code_refine.py | 61 +++--- 5 files changed, 176 insertions(+), 335 deletions(-) create mode 100644 metagpt/actions/write_code_guide.py diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py index f1c231525..909fa6db9 100644 --- a/metagpt/actions/refine_design_api.py +++ b/metagpt/actions/refine_design_api.py @@ -3,7 +3,7 @@ import re import shutil from pathlib import Path -from typing import List +from typing import List, Union from metagpt.actions import Action, ActionOutput from metagpt.config import CONFIG @@ -26,34 +26,29 @@ templates = { ## Format example {format_example} ----- -Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and the provided difference descriptions. Make the best use of good open source tools. -Requirement: Fill in the following missing information based on the context, each section name is a key in json +Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and legacy design. Make the best use of good open source tools. +Requirement: Fill in the following missing information based on the context, each section name is a key in json. Output exactly as shown in the example, including single and double quotes. Max Output: 8192 chars or 2048 tokens. Try to use them up. -## Difference Description: Provided as Python list[str], the list of differences between the new design and the legacy design +## Difference Description: Provide as list, the foremost differences description for system design here based on the previous. -## Implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. +## Incremental implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. ## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores -## File list: Provided as Python list[str], the list of ONLY REQUIRED files needed to write the program(LESS IS MORE!). Only need relative paths, comply with PEP8 standards. ALWAYS write a main.py or app.py here +## Data structures and interface definitions: Use single quotes to wrap content. Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. -## Data structures and interface definitions: Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. +## Program call flow: Use single quotes to wrap content. Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. -## Program call flow: Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. - -## Anything UNCLEAR: Provide as Plain text. Make clear here. - -output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, -and only output the json inside this tag, nothing else +output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example. +Output exactly as shown in the example, including single and double quotes, and only output the json inside this tag, nothing else """, "FORMAT_EXAMPLE": """ [CONTENT] { "Difference Description": ["The ..."], - "Implementation approach": "We will ...", - "Python package name": "snake_game", - "File list": ["main.py"], + "Incremental implementation approach": "We will ...", + "Python package name": "new_name", "Data structures and interface definitions": ' classDiagram class Game{ @@ -67,8 +62,7 @@ and only output the json inside this tag, nothing else participant M as Main ... G->>M: end game - ', - "Anything UNCLEAR": "The requirement is clear to me." + ' } [/CONTENT] """, @@ -84,24 +78,20 @@ and only output the json inside this tag, nothing else ## Format example {format_example} ----- -Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and the provided difference descriptions. Make the best use of good open source tools. -Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately +Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and legacy design. Make the best use of good open source tools. +Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately. Output exactly as shown in the example, including single and double quotes. Max Output: 8192 chars or 2048 tokens. Try to use them up. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Difference Description: Provided as Python list[str], the list of differences between the new design and the legacy design +## Difference Description: Provide as list, the foremost differences description for system design here based on the previous. -## Implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. +## Incremental implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. ## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores -## File list: Provided as Python list[str], the list of ONLY REQUIRED files needed to write the program(LESS IS MORE!). Only need relative paths, comply with PEP8 standards. ALWAYS write a main.py or app.py here +## Data structures and interface definitions: Use single quotes to wrap content. Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. -## Data structures and interface definitions: Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. - -## Program call flow: Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. - -## Anything UNCLEAR: Provide as Plain text. Make clear here. +## Program call flow: Use single quotes to wrap content. Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. """, "FORMAT_EXAMPLE": """ @@ -113,19 +103,12 @@ Attention: Use '##' to split sections, not '#', and '## ' SHOULD W ] ``` -## Implementation approach +## Incremental implementation approach We will ... ## Python package name ```python -"snake_game" -``` - -## File list -```python -[ - "main.py", -] +"new_name" ``` ## Data structures and interface definitions @@ -145,22 +128,19 @@ sequenceDiagram ... G->>M: end game ``` - -## Anything UNCLEAR -The requirement is clear to me. --- """, }, } OUTPUT_MAPPING = { - "Difference Description": (List[str], ...), - "Implementation approach": (str, ...), + # "Incremental Requirements": (str, ...), + "Difference Description": (Union[List[str], str], ...), + "Incremental implementation approach": (str, ...), "Python package name": (str, ...), - "File list": (List[str], ...), + # "File list": (List[str], ...), "Data structures and interface definitions": (str, ...), - "Program call flow": (str, ...), - "Anything UNCLEAR": (str, ...), + "Program call flow": (str, ...) } @@ -201,9 +181,6 @@ class RefineDesign(Action): async def _save_prd(self, docs_path, resources_path, context): prd_file = docs_path / "prd.md" - if context[-1].instruct_content and context[-1].instruct_content.dict()["Competitive Quadrant Chart"]: - quadrant_chart = context[-1].instruct_content.dict()["Competitive Quadrant Chart"] - await mermaid_to_file(quadrant_chart, resources_path / "competitive_analysis") if context[-1].instruct_content: logger.info(f"Saving PRD to {prd_file}") diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py index 81cb02af8..0f01b9904 100644 --- a/metagpt/actions/refine_prd.py +++ b/metagpt/actions/refine_prd.py @@ -1,4 +1,4 @@ -from typing import List +from typing import List, Union from metagpt.actions import Refine, ActionOutput, SearchAndSummarize from metagpt.config import CONFIG @@ -9,96 +9,36 @@ increment_template = { "json": { "PROMPT_TEMPLATE": """ # Context -## User's New Requirements +## User's Incremental Requirements {new_requirements} -## Difference Description -{difference_description} - ## Legacy PRD {legacy} ## Search Information {search_information} -## mermaid quadrantChart code syntax example. DONT USE QUOTO IN CODE DUE TO INVALID SYNTAX. Replace the with REAL COMPETITOR NAME -```mermaid -quadrantChart - title Reach and engagement of campaigns - x-axis Low Reach --> High Reach - y-axis Low Engagement --> High Engagement - quadrant-1 We should expand - quadrant-2 Need to promote - quadrant-3 Re-evaluate - quadrant-4 May be improved - "Campaign: A": [0.3, 0.6] - "Campaign B": [0.45, 0.23] - "Campaign C": [0.57, 0.69] - "Campaign D": [0.78, 0.34] - "Campaign E": [0.40, 0.34] - "Campaign F": [0.35, 0.78] - "Our Target Product": [0.5, 0.6] -``` - ## Format example {format_example} ----- Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. -Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design +Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive designOnly output one json, nothing else. -## New Requirements: Provide as Plain text and place the new requirements here +## Incremental Requirements: Provide as str, the foremost incremental requirements for PRD here based on the previous. -## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple +## Difference Description: Provide as str, the foremost differences description for PRD here based on the previous. ## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple -## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple - -## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less - -## Competitive Analysis: Provided as Python list[str], up to 7 competitive product analyses, consider as similar competitors as possible - -## Competitive Quadrant Chart: Use mermaid quadrantChart code syntax. up to 14 competitive products. Translation: Distribute these competitor scores evenly between 0 and 1, trying to conform to a normal distribution centered around 0.5 as much as possible. - -## Requirement Analysis: Provide as Plain text. Be simple. LESS IS MORE. Make your requirements less dumb. Delete the parts unnessasery. - -## Requirement Pool: Provided as Python list[list[str], the parameters are requirement description, priority(P0/P1/P2), respectively, comply with PEP standards; no more than 5 requirements and consider to make its difficulty lower - -## UI Design draft: Provide as Plain text. Be simple. Describe the elements and functions, also provide a simple style description and layout description. -## Anything UNCLEAR: Provide as Plain text. Make clear here. - output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, and only output the json inside this tag, nothing else """, "FORMAT_EXAMPLE": """ [CONTENT] { - "New Requirements": "", + "Incremental Requirements": "", "Difference Description": [], - "Search Information": "", - "Requirements": "", "Incremental Development Plan": [], - "Product Goals": [], - "User Stories": [], - "Competitive Analysis": [], - "Competitive Quadrant Chart": "quadrantChart - title Reach and engagement of campaigns - x-axis Low Reach --> High Reach - y-axis Low Engagement --> High Engagement - quadrant-1 We should expand - quadrant-2 Need to promote - quadrant-3 Re-evaluate - quadrant-4 May be improved - Campaign A: [0.3, 0.6] - Campaign B: [0.45, 0.23] - Campaign C: [0.57, 0.69] - Campaign D: [0.78, 0.34] - Campaign E: [0.40, 0.34] - Campaign F: [0.35, 0.78]", - "Requirement Analysis": "", - "Requirement Pool": [["P0","P0 requirement"],["P1","P1 requirement"]], - "UI Design draft": "", - "Anything UNCLEAR": "", } [/CONTENT] """, @@ -106,154 +46,58 @@ and only output the json inside this tag, nothing else "markdown": { "PROMPT_TEMPLATE": """ # Context -You need to refine the requirements based on the new requirements and the existing requirements' output. -## User's New Requirements +You need to refine the requirements based on the Incremental Requirements and the existing requirements' output. +## User's Incremental Requirements {new_requirements} -## Difference Description -{difference_description} - ## Legacy PRD {legacy} ## Search Information {search_information} -## mermaid quadrantChart code syntax example. DONT USE QUOTO IN CODE DUE TO INVALID SYNTAX. Replace the with REAL COMPETITOR NAME -```mermaid -quadrantChart - title Reach and engagement of campaigns - x-axis Low Reach --> High Reach - y-axis Low Engagement --> High Engagement - quadrant-1 We should expand - quadrant-2 Need to promote - quadrant-3 Re-evaluate - quadrant-4 May be improved - "Campaign: A": [0.3, 0.6] - "Campaign B": [0.45, 0.23] - "Campaign C": [0.57, 0.69] - "Campaign D": [0.78, 0.34] - "Campaign E": [0.40, 0.34] - "Campaign F": [0.35, 0.78] - "Our Target Product": [0.5, 0.6] -``` - ## Format example {format_example} ----- Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format. +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format.Only output one json, nothing else. -## New Requirements: Provide as Plain text and place the new requirements here +## Incremental Requirements: Provide as str, the foremost incremental requirements for PRD here based on the previous. -## Difference Description: Provide as Python list[str], up to 5 clear, difference descriptions. If the requirement itself is simple, the difference description should also be simple +## Difference Description: Provide as str, the foremost differences description for PRD here based on the previous. ## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple - -## Product Goals: Provided as Python list[str], up to 3 clear, orthogonal product goals. If the requirement itself is simple, the goal should also be simple - -## User Stories: Provided as Python list[str], up to 5 scenario-based user stories, If the requirement itself is simple, the user stories should also be less - -## Competitive Analysis: Provided as Python list[str], up to 7 competitive product analyses, consider as similar competitors as possible - -## Competitive Quadrant Chart: Use mermaid quadrantChart code syntax. up to 14 competitive products. Translation: Distribute these competitor scores evenly between 0 and 1, trying to conform to a normal distribution centered around 0.5 as much as possible. - -## Requirement Analysis: Provide as Plain text. Be simple. LESS IS MORE. Make your requirements less dumb. Delete the parts unnessasery. - -## Requirement Pool: Provided as Python list[list[str], the parameters are requirement description, priority(P0/P1/P2), respectively, comply with PEP standards; no more than 5 requirements and consider to make its difficulty lower - -## UI Design draft: Provide as Plain text. Be simple. Describe the elements and functions, also provide a simple style description and layout description. -## Anything UNCLEAR: Provide as Plain text. Make clear here. """, "FORMAT_EXAMPLE": """ --- -## New Requirements +## Incremental Requirements The boss ... ## Difference Description -```python -[ - "...", -] +... ## Incremental Development Plan [ - "It ...", + "...", ] - -## Product Goals -```python -[ - "Create a ...", -] -``` - -## User Stories -```python -[ - "As a user, ...", -] -``` - -## Competitive Analysis -```python -[ - "Python Snake Game: ...", -] -``` - -## Competitive Quadrant Chart -```mermaid -quadrantChart - title Reach and engagement of campaigns - ... - "Our Target Product": [0.6, 0.7] -``` - -## Requirement Analysis -The product should be a ... - -## Requirement Pool -```python -[ - ["End game ...", "P0"] -] -``` - -## UI Design draft -Give a basic function description, and a draft - -## Anything UNCLEAR -There are no unclear points. ---- """, }, } INCREMENT_OUTPUT_MAPPING = { - "New Requirements": (str, ...), - # "Major Enhancements": (List[str], ...), - "Difference Description": (List[str], ...), + "Incremental Requirements": (str, ...), + "Difference Description": (Union[List[str], str], ...), "Incremental Development Plan": (List[str], ...), - "Product Goals": (List[str], ...), - "User Stories": (List[str], ...), - "Competitive Analysis": (List[str], ...), - "Competitive Quadrant Chart": (str, ...), - "Requirement Analysis": (str, ...), - "Requirement Pool": (List[List[str]], ...), - "UI Design draft": (str, ...), - "Anything UNCLEAR": (str, ...), } -# 对于产品经理,增量开发的动作是:RefinePDR,输出是结合新需求和已有需求输出的新的PDR class RefinePRD(Refine): def __init__(self, name="RefinePRD", context=None, llm=None): super().__init__(name, context, llm) - async def run(self, new_requirements, difference_description, legacy, format=CONFIG.prompt_format, *args, **kwargs): + async def run(self, new_requirements, legacy, format=CONFIG.prompt_format, *args, **kwargs): sas = SearchAndSummarize() rsp = "" info = f"### Search Results\n{sas.result}\n\n### Search Summary\n{rsp}" @@ -263,7 +107,7 @@ class RefinePRD(Refine): prompt_template, format_example = get_template(increment_template, format) prompt = prompt_template.format( - new_requirements=new_requirements, difference_description=difference_description, legacy=legacy, search_information=info, + new_requirements=new_requirements, legacy=legacy, search_information=info, format_example=format_example ) logger.debug(prompt) diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py index 7d9e118d8..dfeeb2db0 100644 --- a/metagpt/actions/refine_project_management.py +++ b/metagpt/actions/refine_project_management.py @@ -1,6 +1,6 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- -from typing import List +from typing import List, Union from metagpt.actions.action import Action from metagpt.config import CONFIG @@ -15,62 +15,50 @@ templates = { # Context {context} -## Legacy Design +## Legacy {legacy} ## Format example {format_example} ----- -Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a task list, and analyze task dependencies to start with the prerequisite modules. -Requirements: Based on the context, fill in the following missing information, each section name is a key in json. Here the granularity of the task is a file, if there are any missing files, you can supplement them +Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a Task list, and analyze task dependencies to start with the prerequisite modules. +Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Required Python third-party packages: Provided in requirements.txt format +## Difference Description: Provide as a python list, the foremost differences description for project management here based on the previous. -## Required Other language third-party packages: Provided in requirements.txt format +## Incremental Required Python third-party packages: Provided as a python list, the requirements.txt format -## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. +## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. -## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. +## Logic Analysis: Only files need to modified, Provided as a Python list[list[str]. If the file has no changes, the file will not be output. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first based on the previous. -## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first - -## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first - -## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. - -## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. +## Task list: Only files need to modified, provided as Python list[str]. If the file has no changes, the file will not be output. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, and only output the json inside this tag, nothing else """, "FORMAT_EXAMPLE": ''' { - "Required Python third-party packages": [ + "Incremental Requirements": "...", + "Difference Description": [ + "...", + ] + "Incremental Required Python third-party packages": [ "flask==1.1.2", "bcrypt==3.2.0" ], - "Required Other language third-party packages": [ - "No third-party ..." - ], "Full API spec": """ openapi: 3.0.0 ... description: A JSON object ... """, - "Difference Description": """ - The ... - """, "Logic Analysis": [ ["game.py","Contains..."] ], "Task list": [ "game.py" - ], - "Shared Knowledge": """ - 'game.py' contains ... - """, - "Anything UNCLEAR": "We need ... how to start." + ] } ''', }, @@ -79,48 +67,44 @@ and only output the json inside this tag, nothing else # Context {context} -## Legacy Design +## Legacy {legacy} ## Format example {format_example} ----- -Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a task list, and analyze task dependencies to start with the prerequisite modules. -Requirements: Based on the context, fill in the following missing information, note that all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file, if there are any missing files, you can supplement them +Role: You are a project manager; the goal is to perform incremental development based on the context and the legacy. Break down tasks according to PRD/technical design, provide a Task list need to modified files, and analyze task dependencies to start with the prerequisite modules. +Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Required Python third-party packages: Provided in requirements.txt format +## Difference Description: Provided as a python list, the foremost differences description for project management here based on the previous. -## Required Other language third-party packages: Provided in requirements.txt format +## Incremental Required Python third-party packages: Provided as a python list, the requirements.txt format -## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend. +## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. -## Difference Description: Please provide a detailed description of the differences between this project and its predecessors or similar projects that can include changes in technology, architecture. - -## Logic Analysis: Provided as a Python list[list[str]. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first - -## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first - -## Shared Knowledge: Anything that should be public like utils' functions, config's variables details that should make clear first. - -## Anything UNCLEAR: Provide as Plain text. Make clear here. For example, don't forget a main entry. don't forget to init 3rd party libs. +## Logic Analysis: Only files need to modified, Provided as a Python list[list[str]. If the file has no changes, the file will not be output. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first based on the previous. +## Task list: Only files need to modified, provided as Python list[str]. If the file has no changes, the file will not be output. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first """, "FORMAT_EXAMPLE": ''' --- -## Required Python third-party packages +## Incremental Requirements +... + +## Difference Description ```python -""" -flask==1.1.2 -bcrypt==3.2.0 -""" +[ + "The ...", +] ``` -## Required Other language third-party packages +## Incremental Required Python third-party packages ```python -""" -No third-party ... -""" +[ + "flask==1.1.2", + "bcrypt==3.2.0" +] ``` ## Full API spec @@ -132,13 +116,6 @@ description: A JSON object ... """ ``` -## Difference Description -```python -""" -The ... -""" -``` - ## Logic Analysis ```python [ @@ -152,29 +129,18 @@ The ... "game.py", ] ``` - -## Shared Knowledge -```python -""" -'game.py' contains ... -""" -``` - -## Anything UNCLEAR -We need ... how to start. --- ''', }, } OUTPUT_MAPPING = { - "Required Python third-party packages": (List[str], ...), - "Required Other language third-party packages": (List[str], ...), + # "Incremental Requirements": (str, ...), + # ## Incremental Requirements: Provided as a str, the foremost incremental requirements for project management here based on the previous. + "Difference Description": (Union[List[str], str], ...), + "Incremental Required Python third-party packages": (Union[List[str], str], ...), "Full API spec": (str, ...), - "Difference Description": (str, ...), "Logic Analysis": (List[List[str]], ...), "Task list": (List[str], ...), - "Shared Knowledge": (str, ...), - "Anything UNCLEAR": (str, ...), } @@ -192,11 +158,13 @@ class RefineTasks(Action): # Write requirements.txt requirements_path = WORKSPACE_ROOT / ws_name / "requirements.txt" - requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Required Python third-party packages"))) + requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Incremental Required Python third-party packages"))) async def run(self, context, legacy, format=CONFIG.prompt_format): prompt_template, format_example = get_template(templates, format) - prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) + prompt = prompt_template.format(context=context, + legacy=legacy, + format_example=format_example) rsp = await self._aask_v1(prompt, "task", OUTPUT_MAPPING, format=format) self._save(context, rsp) return rsp diff --git a/metagpt/actions/write_code_guide.py b/metagpt/actions/write_code_guide.py new file mode 100644 index 000000000..391173bb3 --- /dev/null +++ b/metagpt/actions/write_code_guide.py @@ -0,0 +1,67 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +from metagpt.actions.action import Action +from metagpt.logs import logger +from metagpt.schema import Message +from metagpt.utils.common import CodeParser +from tenacity import retry, stop_after_attempt, wait_fixed + + +PROMPT_TEMPLATE = """ +NOTICE +Role: You are a professional software engineer, and your main task is to conduct incremental development, proposing incremental development plans and code guideance based on context and legacy code. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". + +## Regulations Review: To make the software directly operable without further coding, follow the regulations below during incremental development: +1) Import all referenced classes. +2) Implement all methods. +3) Add necessary explanation to all methods. +4) Ensure there are no potential bugs. +5) Confirm that the entire project conforms to the tasks proposed by the user. +6) Review the code thoroughly, checking for errors and validating the logic to ensure seamless user interaction without compromising any specified requirements. + +## Incremental Development Plan: Proposed the Minimum essential incremental development plan, based on the following context and legacy code by thinking step by step. +... + +## Code guidelines: Propose the foremost guidelines that how to implement code of modification part for incremental development based on the above context, legacy code and incremental development plan. +```python +... +''' +----- +# Context +{context} + +## Legacy Code +You are tasked with conducting incremental development in the existing code based on the provided legacy code and above information. +``` +{code} +``` +----- + +## Format example +----- +## Incremental Development Guide: +... + +## Code Guidance: +# Implementation the ... +```python +... +''' +----- +""" + + +class WriteCodeGuide(Action): + def __init__(self, name="WriteCodeGuide", context: list[Message] = None, llm=None): + super().__init__(name, context, llm) + + async def run(self, context, code): + prompt = PROMPT_TEMPLATE.format(context=context, code=code) + logger.info(f'Write Code Guide ..') + code_guide = await self._aask(prompt) + # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) + # self._save(context, filename, code) + return code_guide + \ No newline at end of file diff --git a/metagpt/actions/write_code_refine.py b/metagpt/actions/write_code_refine.py index 527952aed..466c30679 100644 --- a/metagpt/actions/write_code_refine.py +++ b/metagpt/actions/write_code_refine.py @@ -7,58 +7,44 @@ from metagpt.schema import Message from metagpt.utils.common import CodeParser from tenacity import retry, stop_after_attempt, wait_fixed - PROMPT_TEMPLATE = """ NOTICE -Role: You are a professional software engineer, and your main task is to conduct incremental development, which includes reviewing existing code, providing modification suggestions, rewriting code, and optimizing the codebase. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". +Role: You are a professional engineer; your primary goal is to write PEP8 compliant, elegant, modular, easy-to-read, and maintainable Python 3.9 code (or any other programming language of your choice). +Requirements: You should modify the corresponding code based on the guidance. Then, output the complete code, fixing all errors according to the context. Ensure that you adhere to the specified guidelines for incremental development and modification of legacy code. +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format should be carefully referenced using the "Format example". Only output the current modified code, nothing else. In the modified code, if unchanged, you should output it, the complete code. -## Code Review: Based on the following context and legacy code, and following the checklist, provide key, clear, concise, and specific code modification suggestions, up to 5. -``` -1. Check 0: Is the code implemented as per the requirements? -2. Check 1: Are there any issues with the code logic? -3. Check 2: Does the existing code follow the "Data structures and interface definitions"? -4. Check 3: Is there a function in the code that is omitted or not fully implemented that needs to be implemented? -5. Check 4: Does the code have unnecessary or lack dependencies? -``` - -## Incremental Development: {filename} Based on the findings from the "Code Review," context, and the legacy code, conduct incremental development by rewriting, optimizing, and adding new code using triple quotes. +## Code: Only Write {filename}, Write code using triple quotes, based on the following list and context. +1. Do your best to implement THIS ONLY ONE FILE. ONLY USE EXISTING API. IF NO API, IMPLEMENT IT. +2. Requirement: Implement one of the following code files based on the provided context. Return the code in the specified format. Your code will be part of the entire project, so ensure it is complete, reliable, and reusable. +3. Attention1: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. +4. Attention2: YOU MUST FOLLOW "Data structures and interface definitions". DONT CHANGE ANY DESIGN. +5. Think before writing: What should be implemented and provided in this document? +6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. +7. Do not use public member functions that do not exist in your design. ----- # Context {context} -## Legacy Code -You are tasked with conducting incremental development in the existing code and creating a new code file, {filename}, based on the provided legacy code and above information. -``` -{code} -``` ----- +## Guidelines: The foremost guidelines of modification for incremental development. +{guide} +----- +## Legacy Code: The Legacy Code that needs to be modified. +{legacy} + +----- ## Format example ----- -{format_example} ------ - -""" - -FORMAT_EXAMPLE = """ - -## Code Review -1. The code ... -2. ... -3. ... -4. ... -5. ... - -## Incremental Development: {filename} +## Modified/Added Code: {filename} ```python -## {filename} - Incremental Development +# {filename} ... ``` +----- """ - class WriteCodeRefine(Action): def __init__(self, name="WriteCodeRefine", context: list[Message] = None, llm=None): super().__init__(name, context, llm) @@ -69,9 +55,8 @@ class WriteCodeRefine(Action): code = CodeParser.parse_code(block="", text=code_rsp) return code - async def run(self, context, code, filename): - format_example = FORMAT_EXAMPLE.format(filename=filename) - prompt = PROMPT_TEMPLATE.format(context=context, code=code, filename=filename, format_example=format_example) + async def run(self, context, code, filename, guide): + prompt = PROMPT_TEMPLATE.format(context=context, legacy=code, filename=filename, guide=guide) logger.info(f'Code refine {filename}..') code = await self.write_code(prompt) # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) From 19b4b27f97bb3d9e0f469b465f87a98e4ff980a1 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 4 Dec 2023 17:14:52 +0800 Subject: [PATCH 228/637] update increment development --- metagpt/actions/refine_design_api.py | 31 ++++------- metagpt/actions/refine_prd.py | 46 +++++----------- metagpt/actions/refine_project_management.py | 58 +++++--------------- metagpt/actions/write_code_guide.py | 45 ++++++++------- metagpt/actions/write_code_refine.py | 22 ++++---- 5 files changed, 77 insertions(+), 125 deletions(-) diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py index 909fa6db9..d6a948b43 100644 --- a/metagpt/actions/refine_design_api.py +++ b/metagpt/actions/refine_design_api.py @@ -20,7 +20,7 @@ templates = { # Context {context} -## Legacy Design +## Legacy {legacy} ## Format example @@ -30,9 +30,7 @@ Role: You are an architect; the goal is to perform incremental development and d Requirement: Fill in the following missing information based on the context, each section name is a key in json. Output exactly as shown in the example, including single and double quotes. Max Output: 8192 chars or 2048 tokens. Try to use them up. -## Difference Description: Provide as list, the foremost differences description for system design here based on the previous. - -## Incremental implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. +## Incremental implementation approach: Provide as Python list[str]. Analyze the difficult points of the requirements, select the appropriate open-source framework. Up to 5. ## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores @@ -46,8 +44,7 @@ Output exactly as shown in the example, including single and double quotes, and "FORMAT_EXAMPLE": """ [CONTENT] { - "Difference Description": ["The ..."], - "Incremental implementation approach": "We will ...", + "Incremental implementation approach": ["We will ...",], "Python package name": "new_name", "Data structures and interface definitions": ' classDiagram @@ -72,7 +69,7 @@ Output exactly as shown in the example, including single and double quotes, and # Context {context} -## Legacy Design +## Legacy {legacy} ## Format example @@ -83,9 +80,7 @@ Requirement: Fill in the following missing information based on the context, not Max Output: 8192 chars or 2048 tokens. Try to use them up. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Difference Description: Provide as list, the foremost differences description for system design here based on the previous. - -## Incremental implementation approach: Provide as Plain text. Analyze the difficult points of the requirements, select the appropriate open-source framework. +## Incremental implementation approach: Provide as Python list[str]. Analyze the difficult points of the requirements, select the appropriate open-source framework. Up to 5. ## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores @@ -96,15 +91,13 @@ Attention: Use '##' to split sections, not '#', and '## ' SHOULD W """, "FORMAT_EXAMPLE": """ --- -## Difference Description -```python -[ - "The ...", -] -``` ## Incremental implementation approach -We will ... +```python +[ + "We will ...", +] +``` ## Python package name ```python @@ -135,8 +128,8 @@ sequenceDiagram OUTPUT_MAPPING = { # "Incremental Requirements": (str, ...), - "Difference Description": (Union[List[str], str], ...), - "Incremental implementation approach": (str, ...), + # "Difference Description": (Union[List[str], str], ...), + "Incremental implementation approach": (Union[List[str], str], ...), "Python package name": (str, ...), # "File list": (List[str], ...), "Data structures and interface definitions": (str, ...), diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py index 0f01b9904..1d8bab5f8 100644 --- a/metagpt/actions/refine_prd.py +++ b/metagpt/actions/refine_prd.py @@ -9,10 +9,9 @@ increment_template = { "json": { "PROMPT_TEMPLATE": """ # Context -## User's Incremental Requirements -{new_requirements} +{context} -## Legacy PRD +## Legacy {legacy} ## Search Information @@ -22,13 +21,9 @@ increment_template = { {format_example} ----- Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. -Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive designOnly output one json, nothing else. +Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design. Only output one json, nothing else. -## Incremental Requirements: Provide as str, the foremost incremental requirements for PRD here based on the previous. - -## Difference Description: Provide as str, the foremost differences description for PRD here based on the previous. - -## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple +## Incremental Development Analysis: Provide as Python list[str], up to 5. incremental development analysis and plans based on the context and the legacy. If the requirement itself is simple, the Incremental Development Analysis should also be simple. output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, and only output the json inside this tag, nothing else @@ -36,9 +31,7 @@ and only output the json inside this tag, nothing else "FORMAT_EXAMPLE": """ [CONTENT] { - "Incremental Requirements": "", - "Difference Description": [], - "Incremental Development Plan": [], + "Incremental Development Analysis": [], } [/CONTENT] """, @@ -46,11 +39,9 @@ and only output the json inside this tag, nothing else "markdown": { "PROMPT_TEMPLATE": """ # Context -You need to refine the requirements based on the Incremental Requirements and the existing requirements' output. -## User's Incremental Requirements -{new_requirements} +{context} -## Legacy PRD +## Legacy {legacy} ## Search Information @@ -63,32 +54,21 @@ Role: You are a professional Product Manager tasked with overseeing incremental Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format.Only output one json, nothing else. -## Incremental Requirements: Provide as str, the foremost incremental requirements for PRD here based on the previous. - -## Difference Description: Provide as str, the foremost differences description for PRD here based on the previous. - -## Incremental Development Plan: Provide as Python list[str], up to 5 clear, incremental development plans. If the requirement itself is simple, the incremental development plan should also be simple +## Incremental Development Analysis: Provide as Python list[str], up to 5. Incremental development analysis and plans based on the context and the legacy. If the requirement itself is simple, the Incremental Development Analysis should also be simple. """, "FORMAT_EXAMPLE": """ --- -## Incremental Requirements -The boss ... -## Difference Description -... - -## Incremental Development Plan +## Incremental Development Analysis [ - "...", + "We will ...", ] """, }, } INCREMENT_OUTPUT_MAPPING = { - "Incremental Requirements": (str, ...), - "Difference Description": (Union[List[str], str], ...), - "Incremental Development Plan": (List[str], ...), + "Incremental Development Analysis": (List[str], ...), } @@ -97,7 +77,7 @@ class RefinePRD(Refine): def __init__(self, name="RefinePRD", context=None, llm=None): super().__init__(name, context, llm) - async def run(self, new_requirements, legacy, format=CONFIG.prompt_format, *args, **kwargs): + async def run(self, context, legacy, format=CONFIG.prompt_format, *args, **kwargs): sas = SearchAndSummarize() rsp = "" info = f"### Search Results\n{sas.result}\n\n### Search Summary\n{rsp}" @@ -107,7 +87,7 @@ class RefinePRD(Refine): prompt_template, format_example = get_template(increment_template, format) prompt = prompt_template.format( - new_requirements=new_requirements, legacy=legacy, search_information=info, + context=context, legacy=legacy, search_information=info, format_example=format_example ) logger.debug(prompt) diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py index dfeeb2db0..2775b1cb6 100644 --- a/metagpt/actions/refine_project_management.py +++ b/metagpt/actions/refine_project_management.py @@ -21,30 +21,23 @@ templates = { ## Format example {format_example} ----- -Role: You are a project manager; the goal is to perform incremental development based on the context and difference descriptions and the legacy. Break down tasks according to PRD/technical design, provide a Task list, and analyze task dependencies to start with the prerequisite modules. +Role: You are a project manager; the goal is to perform incremental development based on the context and the legacy. Break down tasks according to PRD/technical design, provide a Task list, and analyze task dependencies to start with the prerequisite modules. Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. +Output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT]. The following is the attribute description of the JSON object. -## Difference Description: Provide as a python list, the foremost differences description for project management here based on the previous. - -## Incremental Required Python third-party packages: Provided as a python list, the requirements.txt format +## Required Python third-party packages: Provided as a python list, the requirements.txt format ## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. -## Logic Analysis: Only files need to modified, Provided as a Python list[list[str]. If the file has no changes, the file will not be output. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first based on the previous. +## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first. -## Task list: Only files need to modified, provided as Python list[str]. If the file has no changes, the file will not be output. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first - -output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, +Output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, and only output the json inside this tag, nothing else """, "FORMAT_EXAMPLE": ''' { - "Incremental Requirements": "...", - "Difference Description": [ - "...", - ] - "Incremental Required Python third-party packages": [ + "Required Python third-party packages": [ "flask==1.1.2", "bcrypt==3.2.0" ], @@ -53,9 +46,6 @@ and only output the json inside this tag, nothing else ... description: A JSON object ... """, - "Logic Analysis": [ - ["game.py","Contains..."] - ], "Task list": [ "game.py" ] @@ -77,29 +67,16 @@ Role: You are a project manager; the goal is to perform incremental development Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -## Difference Description: Provided as a python list, the foremost differences description for project management here based on the previous. - -## Incremental Required Python third-party packages: Provided as a python list, the requirements.txt format +## Required Python third-party packages: Provided as a python list, the requirements.txt format ## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. -## Logic Analysis: Only files need to modified, Provided as a Python list[list[str]. If the file has no changes, the file will not be output. the first is filename, the second is class/method/function should be implemented in this file. Analyze the dependencies between the files, which work should be done first based on the previous. - -## Task list: Only files need to modified, provided as Python list[str]. If the file has no changes, the file will not be output. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first +## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first. """, "FORMAT_EXAMPLE": ''' --- -## Incremental Requirements -... -## Difference Description -```python -[ - "The ...", -] -``` - -## Incremental Required Python third-party packages +## Required Python third-party packages ```python [ "flask==1.1.2", @@ -116,13 +93,6 @@ description: A JSON object ... """ ``` -## Logic Analysis -```python -[ - ["game.py", "Contains ..."], -] -``` - ## Task list ```python [ @@ -136,16 +106,16 @@ description: A JSON object ... OUTPUT_MAPPING = { # "Incremental Requirements": (str, ...), # ## Incremental Requirements: Provided as a str, the foremost incremental requirements for project management here based on the previous. - "Difference Description": (Union[List[str], str], ...), - "Incremental Required Python third-party packages": (Union[List[str], str], ...), + # "Difference Analysis": (Union[List[str], str], ...), + "Required Python third-party packages": (Union[List[str], str], ...), "Full API spec": (str, ...), - "Logic Analysis": (List[List[str]], ...), + # "Logic Analysis": (List[List[str]], ...), "Task list": (List[str], ...), } class RefineTasks(Action): - def __init__(self, name="CreateTasks", context=None, llm=None): + def __init__(self, name="RefineTasks", context=None, llm=None): super().__init__(name, context, llm) def _save(self, context, rsp): @@ -158,7 +128,7 @@ class RefineTasks(Action): # Write requirements.txt requirements_path = WORKSPACE_ROOT / ws_name / "requirements.txt" - requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Incremental Required Python third-party packages"))) + requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Required Python third-party packages"))) async def run(self, context, legacy, format=CONFIG.prompt_format): prompt_template, format_example = get_template(templates, format) diff --git a/metagpt/actions/write_code_guide.py b/metagpt/actions/write_code_guide.py index 391173bb3..3e51f0d2d 100644 --- a/metagpt/actions/write_code_guide.py +++ b/metagpt/actions/write_code_guide.py @@ -10,24 +10,21 @@ from tenacity import retry, stop_after_attempt, wait_fixed PROMPT_TEMPLATE = """ NOTICE -Role: You are a professional software engineer, and your main task is to conduct incremental development, proposing incremental development plans and code guideance based on context and legacy code. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". +Role: You are a professional software engineer, and your main task is to conduct incremental development, proposing incremental development plans and code guideance based on context and legacy code. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). Output format carefully referenced "Format example". ## Regulations Review: To make the software directly operable without further coding, follow the regulations below during incremental development: +0) Determine the scope of responsibilities of each file and what classes and methods need to be implemented. 1) Import all referenced classes. -2) Implement all methods. -3) Add necessary explanation to all methods. +2) Implement all methods. +3) Add necessary explanation to all methods. 4) Ensure there are no potential bugs. 5) Confirm that the entire project conforms to the tasks proposed by the user. 6) Review the code thoroughly, checking for errors and validating the logic to ensure seamless user interaction without compromising any specified requirements. -## Incremental Development Plan: Proposed the Minimum essential incremental development plan, based on the following context and legacy code by thinking step by step. -... +## Incremental Development Plan: Provided as a Python list containing `filename.py`. Proposed the detail and essential incremental development plan, based on the following context and legacy code by thinking and analyzing step by step. All incremental modules/functions need to be added to the corresponding code files. + +## Code Guidance: Propose the foremost guidelines that how to implement code of modification part for incremental development based on the above context, legacy code and incremental development plan. -## Code guidelines: Propose the foremost guidelines that how to implement code of modification part for incremental development based on the above context, legacy code and incremental development plan. -```python -... -''' ----- # Context {context} @@ -35,20 +32,31 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc ## Legacy Code You are tasked with conducting incremental development in the existing code based on the provided legacy code and above information. ``` -{code} +{legacy} ``` ----- ## Format example ----- -## Incremental Development Guide: -... +## Incremental Development Guide +[ + "`game.py` Contains `Game` and ...", +] -## Code Guidance: -# Implementation the ... + +## Code Guidance +### Implementation `xx` in `xxx.py` ..., else retain the original xxx.py code. ```python +## xxx.py ... -''' +``` +--- +### Implementation of the `Game` in `game.py` ..., else retain the original game.py code. +```python +## game.py +class Game: + ... +``` ----- """ @@ -57,11 +65,10 @@ class WriteCodeGuide(Action): def __init__(self, name="WriteCodeGuide", context: list[Message] = None, llm=None): super().__init__(name, context, llm) - async def run(self, context, code): - prompt = PROMPT_TEMPLATE.format(context=context, code=code) + async def run(self, context, legacy): + prompt = PROMPT_TEMPLATE.format(context=context, legacy=legacy) logger.info(f'Write Code Guide ..') code_guide = await self._aask(prompt) # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) - # self._save(context, filename, code) return code_guide \ No newline at end of file diff --git a/metagpt/actions/write_code_refine.py b/metagpt/actions/write_code_refine.py index 466c30679..5a70b2c68 100644 --- a/metagpt/actions/write_code_refine.py +++ b/metagpt/actions/write_code_refine.py @@ -10,17 +10,19 @@ from tenacity import retry, stop_after_attempt, wait_fixed PROMPT_TEMPLATE = """ NOTICE Role: You are a professional engineer; your primary goal is to write PEP8 compliant, elegant, modular, easy-to-read, and maintainable Python 3.9 code (or any other programming language of your choice). -Requirements: You should modify the corresponding code based on the guidance. Then, output the complete code, fixing all errors according to the context. Ensure that you adhere to the specified guidelines for incremental development and modification of legacy code. +Requirements: Rewrite the complete code based on the Legacy Code so that it can be executed and avoid any potential bugs. You should modify the corresponding code based on the guidance. Output the complete code, fixing all errors according to the context. Ensure that you adhere to the specified guidelines for incremental development and modification of legacy code. ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format should be carefully referenced using the "Format example". Only output the current modified code, nothing else. In the modified code, if unchanged, you should output it, the complete code. -## Code: Only Write {filename}, Write code using triple quotes, based on the following list and context. -1. Do your best to implement THIS ONLY ONE FILE. ONLY USE EXISTING API. IF NO API, IMPLEMENT IT. -2. Requirement: Implement one of the following code files based on the provided context. Return the code in the specified format. Your code will be part of the entire project, so ensure it is complete, reliable, and reusable. -3. Attention1: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. -4. Attention2: YOU MUST FOLLOW "Data structures and interface definitions". DONT CHANGE ANY DESIGN. +## Rewrite Complete Code: Only Write one file {filename}, Write code using triple quotes, based on the following list, context, guidelines and legacy code. +1. Important: Do your best to implement ONLY ONE FILE. ONLY USE EXISTING API. IF NO API, IMPLEMENT IT. +2. Implement one of the following code files based on the provided context. Return the code in the specified format. Your code will be part of the entire project, so ensure it is complete, reliable, and reusable. +3. Attention1: Implement the functions required by the current file scope of responsibility. For example, main only needs to focus on the basic functions of main.py in the legacy code and the incremental functions to be implemented. Reuse existing code as much as possible. You can import functions from other codes instead of reimplementing the function. If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. +4. Attention2: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions, taking into account the guidelines, context, and existing legacy code. Retain the basic function methods from the legacy code, and make sure to preserve the existing code and logic that needs to be retained throughout the incremental development process. Avoid omitting any essential components. 5. Think before writing: What should be implemented and provided in this document? 6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 7. Do not use public member functions that do not exist in your design. +8. The Modified Code is implemented according to the requirements, and there are no issues with the code logic. All functions in the Modified Code are fully implemented; none are omitted or incomplete. + ----- # Context {context} @@ -30,13 +32,13 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format should be carefull {guide} ----- -## Legacy Code: The Legacy Code that needs to be modified. +## Legacy Code: The Legacy Code that needs to be modified. '===' is the separator of each code file in the legacy code. Basic function methods need to be retained in {filename}. {legacy} ----- ## Format example ----- -## Modified/Added Code: {filename} +## Rewrite Complete Code: {filename} ```python # {filename} ... @@ -55,8 +57,8 @@ class WriteCodeRefine(Action): code = CodeParser.parse_code(block="", text=code_rsp) return code - async def run(self, context, code, filename, guide): - prompt = PROMPT_TEMPLATE.format(context=context, legacy=code, filename=filename, guide=guide) + async def run(self, context, legacy, filename, guide): + prompt = PROMPT_TEMPLATE.format(context=context, legacy=legacy, filename=filename, guide=guide) logger.info(f'Code refine {filename}..') code = await self.write_code(prompt) # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) From be47f6171daa61b3a4ef7249379f68aacfd73917 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 26 Dec 2023 14:08:10 +0800 Subject: [PATCH 229/637] resolve CR in MR17. --- metagpt/roles/ml_engineer.py | 23 +++++++++-------------- 1 file changed, 9 insertions(+), 14 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index c2df4bb79..cafd9b968 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -168,22 +168,16 @@ class MLEngineer(Role): \n{json.dumps(list(UDFS_YAML.keys()), indent=2, ensure_ascii=False)}") # set task_type to `udf` self.plan.current_task.task_type = 'udf' - tool_context, code = await WriteCodeWithTools(schema_path=UDFS_YAML).run( - context=context, - plan=self.plan, - column_info=self.data_desc.get("column_info", ""), - ) - debug_context = tool_context - cause_by = WriteCodeWithTools + schema_path = UDFS_YAML else: schema_path = PROJECT_ROOT / "metagpt/tools/functions/schemas" - tool_context, code = await WriteCodeWithTools(schema_path=schema_path).run( - context=context, - plan=self.plan, - column_info=self.data_desc.get("column_info", ""), - ) - debug_context = tool_context - cause_by = WriteCodeWithTools + tool_context, code = await WriteCodeWithTools(schema_path=schema_path).run( + context=context, + plan=self.plan, + column_info=self.data_desc.get("column_info", ""), + ) + debug_context = tool_context + cause_by = WriteCodeWithTools self.working_memory.add( Message(content=code, role="assistant", cause_by=cause_by) ) @@ -301,6 +295,7 @@ class MLEngineer(Role): # tool_code = await make_tools.run(code_prompt) tool_code = await make_tools.run(code) _, success = await self.execute_code.run(tool_code) + make_tool_retries += 1 if make_tool_current_retry > make_tool_retries: logger.error(f"We have tried the maximum number of attempts {make_tool_retries}\ and still have not created tools for task_id {self.plan.current_task_id} successfully,\ From b43cdb23f7921daf9ba4866746928e8d38bc55e6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 26 Dec 2023 14:11:13 +0800 Subject: [PATCH 230/637] update make_tools. --- metagpt/roles/ml_engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index cafd9b968..b991d9329 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -295,7 +295,7 @@ class MLEngineer(Role): # tool_code = await make_tools.run(code_prompt) tool_code = await make_tools.run(code) _, success = await self.execute_code.run(tool_code) - make_tool_retries += 1 + make_tool_current_retry += 1 if make_tool_current_retry > make_tool_retries: logger.error(f"We have tried the maximum number of attempts {make_tool_retries}\ and still have not created tools for task_id {self.plan.current_task_id} successfully,\ From b49db2d62f55db6823335ecad54bf841f348245e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 26 Dec 2023 14:14:18 +0800 Subject: [PATCH 231/637] resolve cr in MR17. --- metagpt/actions/write_analysis_code.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index c9acb32b9..3e912ace5 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -33,7 +33,7 @@ class BaseWriteAnalysisCode(Action): # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): - default_system_msg = system_msg or "" + default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG # 全部转成list if not isinstance(prompt, list): prompt = [prompt] @@ -96,7 +96,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): **kwargs, ) -> str: # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) - prompt = self.process_msg(context, system_msg or self.DEFAULT_SYSTEM_MSG) + prompt = self.process_msg(context, system_msg) code_content = await self.llm.aask_code(prompt, **kwargs) return code_content["code"] From a2743d2b1fe47761db9be24ca6a49e526b9289eb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 26 Dec 2023 15:48:04 +0800 Subject: [PATCH 232/637] resolve cr in MR17. --- metagpt/actions/write_analysis_code.py | 34 ++++++----- metagpt/roles/ml_engineer.py | 80 ++++++++++---------------- tests/metagpt/roles/test_daml.py | 4 ++ 3 files changed, 55 insertions(+), 63 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 3e912ace5..9691f888f 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -249,6 +249,7 @@ class MakeTools(WriteCodeByGenerate): super().__init__(name, context, llm) self.workspace = workspace or str(Path(__file__).parents[1].joinpath("./tools/functions/libs/udf")) self.file_suffix: str = '.py' + self.context = [] def parse_function_name(self, function_code: str) -> str: # 定义正则表达式模式 @@ -270,11 +271,14 @@ class MakeTools(WriteCodeByGenerate): saved_path.write_text(tool_code, encoding='utf-8') @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) - async def run(self, code: str, code_desc: str = None, **kwargs) -> str: + async def run(self, code: str | List[dict], code_desc: str = None, **kwargs) -> str: # 拼接code prompt code_prompt = f"The following code is about {code_desc}, convert it to be a General Function, {code}" - msgs = self.process_msg(code_prompt, self.DEFAULT_SYSTEM_MSG) - logger.info(f"\n\nAsk to Make tools:\n{'-'*60}\n {msgs[-1]}") + if not self.context: + self.context = self.process_msg(code_prompt) + else: + self.context.append(self.process_msg(code_prompt)[-1]) + logger.info(f"\n\nAsk to Make tools:\n{'-'*60}\n {self.context[-1]}") # 更新kwargs if 'code' in kwargs: @@ -282,17 +286,21 @@ class MakeTools(WriteCodeByGenerate): if 'code_desc' in kwargs: kwargs.pop('code_desc') - tool_code = await self.llm.aask_code(msgs, **kwargs) - max_tries, current_try = 3, 1 - func_name = self.parse_function_name(tool_code['code']) - while current_try < max_tries and func_name is None: - logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") - logger.warning(f"No function name found in code, we will retry make tools. \n\n{tool_code['code']}\n") - msgs.append({'role': 'assistant', 'content': 'We need a general function in above code,but not found function.'}) - tool_code = await self.llm.aask_code(msgs, **kwargs) - current_try += 1 + max_tries, current_try = 3, 0 + while True: + tool_code = await self.llm.aask_code(self.context, **kwargs) func_name = self.parse_function_name(tool_code['code']) - if func_name is not None: + current_try += 1 + # make tools failed, add error message to context. + if not func_name: + logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") + logger.error(f"No function name found in code, we will retry make tools.\n{tool_code['code']}\n") + self.context.append({'role': 'user', 'content': 'We need a general function in above code,but not found function.'}) + # end make tools + if func_name is not None or current_try >= max_tries: + if current_try >= max_tries: + logger.error(f"We have tried the maximum number of attempts {max_tries}\ + and still have not created tools successfully, we will skip it.") break logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") self.save(tool_code['code']) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index b991d9329..cec572991 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -286,65 +286,45 @@ class MLEngineer(Role): logger.warning(f"Making tools for task_id {self.plan.current_task_id}: \ `{self.plan.current_task.instruction}` \n code: \n {code}") make_tools = MakeTools() - tool_code = await make_tools.run(code, self.plan.current_task.instruction) - # check tool_code by execute_code - logger.info(f"Checking task_id {self.plan.current_task_id} tool code by executor...") - _, success = await self.execute_code.run(tool_code) - make_tool_retries, make_tool_current_retry = 3, 1 - while not success: - # tool_code = await make_tools.run(code_prompt) - tool_code = await make_tools.run(code) - _, success = await self.execute_code.run(tool_code) + make_tool_retries, make_tool_current_retry = 3, 0 + while True: + # start make tools + tool_code = await make_tools.run(code, self.plan.current_task.instruction) make_tool_current_retry += 1 - if make_tool_current_retry > make_tool_retries: - logger.error(f"We have tried the maximum number of attempts {make_tool_retries}\ - and still have not created tools for task_id {self.plan.current_task_id} successfully,\ - we will skip it.") + + # check tool_code by execute_code + logger.info(f"Checking task_id {self.plan.current_task_id} tool code by executor...") + execute_result, execute_success = await self.execute_code.run(tool_code) + if not execute_success: + logger.error(f"Tool code faild to execute, \n{execute_result}\n.We will try to fix it ...") + # end make tools + if execute_success or make_tool_current_retry >= make_tool_retries: + if make_tool_current_retry >= make_tool_retries: + logger.error(f"We have tried the maximum number of attempts {make_tool_retries}\ + and still have not created tools for task_id {self.plan.current_task_id} successfully,\ + we will skip it.") break # save successful tool code in udf - if success: + if execute_success: make_tools.save(tool_code) if __name__ == "__main__": - requirement = "Run data analysis on sklearn Iris dataset, include a plot" - # requirement = "Run data analysis on sklearn Diabetes dataset, include a plot" - # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - # requirement = "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy" - # requirement = "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: workspace/titanic/train.csv" - - async def run_udfs(requirement: str = requirement, auto_run: bool = True): - role = MLEngineer(goal=requirement, auto_run=auto_run) - # make udfs - role.use_tools = False - role.use_code_steps = False - role.make_udfs = True - role.use_udfs = False - await role.run(requirement) - # use udfs - role.reset() - role.make_udfs = False - role.use_udfs = True - role.use_code_steps = False - role.use_tools = False - await role.run(requirement) - + requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." - # requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." + data_path = f"{DATA_PATH}/titanic" + requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + requirement = f"Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" + data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" + requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." - # data_path = f"{DATA_PATH}/titanic" - # requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - # requirement = f"Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" - # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." + data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" + requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." - # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" - # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report F1 Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." - - # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" - # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - # save_dir = "" - # # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" + data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" + requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + save_dir = "" + # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" async def main(requirement: str = requirement, auto_run: bool = True, use_tools: bool = False, use_code_steps: bool = False, save_dir: str = ""): """ @@ -377,4 +357,4 @@ if __name__ == "__main__": logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") - fire.Fire(run_udfs) + fire.Fire(main) diff --git a/tests/metagpt/roles/test_daml.py b/tests/metagpt/roles/test_daml.py index 672a3daed..55b425316 100644 --- a/tests/metagpt/roles/test_daml.py +++ b/tests/metagpt/roles/test_daml.py @@ -9,6 +9,8 @@ async def make_use_tools(requirement: str, auto_run: bool = True): """make and use tools for requirement.""" role = MLEngineer(goal=requirement, auto_run=auto_run) # make udfs + role.use_tools = False + role.use_code_steps = False role.make_udfs = True role.use_udfs = False await role.run(requirement) @@ -16,6 +18,8 @@ async def make_use_tools(requirement: str, auto_run: bool = True): role.reset() role.make_udfs = False role.use_udfs = True + role.use_code_steps = False + role.use_tools = False await role.run(requirement) From f7205645b20451030bcfea4564dd2ac030854f3f Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 26 Dec 2023 20:25:41 +0800 Subject: [PATCH 233/637] Update increment development for 0.5.x version: Delete files with 'refine' and write_code_guide.py. Add write_code_guide_an.py. Update write_code.py for guiding write code. --- metagpt/actions/__init__.py | 1 - metagpt/actions/prepare_documents.py | 2 +- metagpt/actions/refine.py | 10 - metagpt/actions/refine_design_api.py | 222 ------------------ metagpt/actions/refine_prd.py | 95 -------- metagpt/actions/refine_project_management.py | 146 ------------ metagpt/actions/write_code.py | 76 +++++-- metagpt/actions/write_code_guide.py | 74 ------ metagpt/actions/write_code_guide_an.py | 223 +++++++++++++++++++ metagpt/actions/write_code_refine.py | 67 ------ metagpt/provider/openai_api.py | 2 +- metagpt/roles/engineer.py | 15 +- 12 files changed, 295 insertions(+), 638 deletions(-) delete mode 100644 metagpt/actions/refine.py delete mode 100644 metagpt/actions/refine_design_api.py delete mode 100644 metagpt/actions/refine_prd.py delete mode 100644 metagpt/actions/refine_project_management.py delete mode 100644 metagpt/actions/write_code_guide.py create mode 100644 metagpt/actions/write_code_guide_an.py delete mode 100644 metagpt/actions/write_code_refine.py diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index 99a4175f6..c34c72ed2 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -14,7 +14,6 @@ from metagpt.actions.debug_error import DebugError from metagpt.actions.design_api import WriteDesign from metagpt.actions.design_api_review import DesignReview from metagpt.actions.project_management import AssignTasks, WriteTasks -from metagpt.actions.refine import Refine from metagpt.actions.research import CollectLinks, WebBrowseAndSummarize, ConductResearch from metagpt.actions.run_code import RunCode from metagpt.actions.search_and_summarize import SearchAndSummarize diff --git a/metagpt/actions/prepare_documents.py b/metagpt/actions/prepare_documents.py index 696dc9a89..259553644 100644 --- a/metagpt/actions/prepare_documents.py +++ b/metagpt/actions/prepare_documents.py @@ -37,7 +37,7 @@ class PrepareDocuments(Action): name = CONFIG.project_name or FileRepository.new_filename() path = Path(CONFIG.workspace_path) / name - if path.exists() and not CONFIG.inc: + if Path(path).exists() and not CONFIG.inc: shutil.rmtree(path) CONFIG.git_repo = GitRepository(local_path=path, auto_init=True) diff --git a/metagpt/actions/refine.py b/metagpt/actions/refine.py deleted file mode 100644 index beea40fc8..000000000 --- a/metagpt/actions/refine.py +++ /dev/null @@ -1,10 +0,0 @@ -from metagpt.actions import Action - - -# 增量开发动作的基类 -class Refine(Action): - def __init__(self, name="Refine", context=None, llm=None): - super().__init__(name, context, llm) - - def run(self, *args, **kwargs): - raise NotImplementedError diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py deleted file mode 100644 index d6a948b43..000000000 --- a/metagpt/actions/refine_design_api.py +++ /dev/null @@ -1,222 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -import re -import shutil -from pathlib import Path -from typing import List, Union - -from metagpt.actions import Action, ActionOutput -from metagpt.config import CONFIG -from metagpt.const import WORKSPACE_ROOT -from metagpt.logs import logger -from metagpt.utils.common import CodeParser -from metagpt.utils.get_template import get_template -from metagpt.utils.json_to_markdown import json_to_markdown -from metagpt.utils.mermaid import mermaid_to_file - -templates = { - "json": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Format example -{format_example} ------ -Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and legacy design. Make the best use of good open source tools. -Requirement: Fill in the following missing information based on the context, each section name is a key in json. Output exactly as shown in the example, including single and double quotes. -Max Output: 8192 chars or 2048 tokens. Try to use them up. - -## Incremental implementation approach: Provide as Python list[str]. Analyze the difficult points of the requirements, select the appropriate open-source framework. Up to 5. - -## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores - -## Data structures and interface definitions: Use single quotes to wrap content. Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. - -## Program call flow: Use single quotes to wrap content. Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. - -output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example. -Output exactly as shown in the example, including single and double quotes, and only output the json inside this tag, nothing else -""", - "FORMAT_EXAMPLE": """ -[CONTENT] -{ - "Incremental implementation approach": ["We will ...",], - "Python package name": "new_name", - "Data structures and interface definitions": ' - classDiagram - class Game{ - +int score - } - ... - Game "1" -- "1" Food: has - ', - "Program call flow": ' - sequenceDiagram - participant M as Main - ... - G->>M: end game - ' -} -[/CONTENT] -""", - }, - "markdown": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Format example -{format_example} ------ -Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and legacy design. Make the best use of good open source tools. -Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately. Output exactly as shown in the example, including single and double quotes. -Max Output: 8192 chars or 2048 tokens. Try to use them up. -Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. - -## Incremental implementation approach: Provide as Python list[str]. Analyze the difficult points of the requirements, select the appropriate open-source framework. Up to 5. - -## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores - -## Data structures and interface definitions: Use single quotes to wrap content. Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. - -## Program call flow: Use single quotes to wrap content. Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. - -""", - "FORMAT_EXAMPLE": """ ---- - -## Incremental implementation approach -```python -[ - "We will ...", -] -``` - -## Python package name -```python -"new_name" -``` - -## Data structures and interface definitions -```mermaid -classDiagram - class Game{ - +int score - } - ... - Game "1" -- "1" Food: has -``` - -## Program call flow -```mermaid -sequenceDiagram - participant M as Main - ... - G->>M: end game -``` ---- -""", - }, -} - -OUTPUT_MAPPING = { - # "Incremental Requirements": (str, ...), - # "Difference Description": (Union[List[str], str], ...), - "Incremental implementation approach": (Union[List[str], str], ...), - "Python package name": (str, ...), - # "File list": (List[str], ...), - "Data structures and interface definitions": (str, ...), - "Program call flow": (str, ...) -} - - -class RefineDesign(Action): - def __init__(self, name, context=None, llm=None): - super().__init__(name, context, llm) - self.desc = ( - "Based on the PRD, think about the system design, and design the corresponding APIs, " - "data structures, library tables, processes, and paths. Please provide your design, feedback " - "clearly and in detail." - ) - - def recreate_workspace(self, workspace: Path): - try: - shutil.rmtree(workspace) - except FileNotFoundError: - pass # Folder does not exist, but we don't care - workspace.mkdir(parents=True, exist_ok=True) - - def create_or_increment_workspace(self, workspace: Path): - # 如果工作空间已存在,添加数字以区分 - original_workspace = workspace - index = 1 - while workspace.exists(): - ws_name_match = re.match(r'^(.*)_([\d]+)$', original_workspace.name) - if ws_name_match: - base_name, existing_index = ws_name_match.groups() - index = int(existing_index) - index += 1 - workspace = original_workspace.parent / f"{base_name}_{index}" - else: - workspace = original_workspace.parent / f"{original_workspace.name}_{index}" - index += 1 - - # 创建工作空间,包括所有必要的父文件夹 - workspace.mkdir(parents=True, exist_ok=True) - return workspace - - async def _save_prd(self, docs_path, resources_path, context): - prd_file = docs_path / "prd.md" - - if context[-1].instruct_content: - logger.info(f"Saving PRD to {prd_file}") - prd_file.write_text(json_to_markdown(context[-1].instruct_content.dict())) - - async def _save_system_design(self, docs_path, resources_path, system_design): - data_api_design = system_design.instruct_content.dict()[ - "Data structures and interface definitions" - ] # CodeParser.parse_code(block="Data structures and interface definitions", text=content) - seq_flow = system_design.instruct_content.dict()[ - "Program call flow" - ] # CodeParser.parse_code(block="Program call flow", text=content) - await mermaid_to_file(data_api_design, resources_path / "data_api_design") - await mermaid_to_file(seq_flow, resources_path / "seq_flow") - system_design_file = docs_path / "system_design.md" - logger.info(f"Saving System Designs to {system_design_file}") - system_design_file.write_text((json_to_markdown(system_design.instruct_content.dict()))) - - async def _save(self, context, system_design): - if isinstance(system_design, ActionOutput): - ws_name = system_design.instruct_content.dict()["Python package name"] - else: - ws_name = CodeParser.parse_str(block="Python package name", text=system_design) - workspace = WORKSPACE_ROOT / ws_name - # workspace = self.create_or_increment_workspace(workspace) - self.recreate_workspace(workspace) - docs_path = workspace / "docs" - resources_path = workspace / "resources" - docs_path.mkdir(parents=True, exist_ok=True) - resources_path.mkdir(parents=True, exist_ok=True) - await self._save_prd(docs_path, resources_path, context) - await self._save_system_design(docs_path, resources_path, system_design) - - async def run(self, context, legacy, format=CONFIG.prompt_format): - prompt_template, format_example = get_template(templates, format) - prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) - # system_design = await self._aask(prompt) - system_design = await self._aask_v1(prompt, "system_design", OUTPUT_MAPPING, format=format) - # fix Python package name, we can't system_design.instruct_content.python_package_name = "xxx" since "Python package name" contain space, have to use setattr - setattr( - system_design.instruct_content, - "Python package name", - system_design.instruct_content.dict()["Python package name"].strip().strip("'").strip('"'), - ) - await self._save(context, system_design) - return system_design diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py deleted file mode 100644 index 1d8bab5f8..000000000 --- a/metagpt/actions/refine_prd.py +++ /dev/null @@ -1,95 +0,0 @@ -from typing import List, Union - -from metagpt.actions import Refine, ActionOutput, SearchAndSummarize -from metagpt.config import CONFIG -from metagpt.logs import logger -from metagpt.utils.get_template import get_template - -increment_template = { - "json": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Search Information -{search_information} - -## Format example -{format_example} ------ -Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. -Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design. Only output one json, nothing else. - -## Incremental Development Analysis: Provide as Python list[str], up to 5. incremental development analysis and plans based on the context and the legacy. If the requirement itself is simple, the Incremental Development Analysis should also be simple. - -output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, -and only output the json inside this tag, nothing else -""", - "FORMAT_EXAMPLE": """ -[CONTENT] -{ - "Incremental Development Analysis": [], -} -[/CONTENT] -""", - }, - "markdown": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Search Information -{search_information} - -## Format example -{format_example} ------ -Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. -Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format.Only output one json, nothing else. - -## Incremental Development Analysis: Provide as Python list[str], up to 5. Incremental development analysis and plans based on the context and the legacy. If the requirement itself is simple, the Incremental Development Analysis should also be simple. -""", - "FORMAT_EXAMPLE": """ ---- - -## Incremental Development Analysis -[ - "We will ...", -] -""", - }, -} - -INCREMENT_OUTPUT_MAPPING = { - "Incremental Development Analysis": (List[str], ...), -} - - -class RefinePRD(Refine): - - def __init__(self, name="RefinePRD", context=None, llm=None): - super().__init__(name, context, llm) - - async def run(self, context, legacy, format=CONFIG.prompt_format, *args, **kwargs): - sas = SearchAndSummarize() - rsp = "" - info = f"### Search Results\n{sas.result}\n\n### Search Summary\n{rsp}" - if sas.result: - logger.info(sas.result) - logger.info(rsp) - - prompt_template, format_example = get_template(increment_template, format) - prompt = prompt_template.format( - context=context, legacy=legacy, search_information=info, - format_example=format_example - ) - logger.debug(prompt) - prd = await self._aask_v1(prompt, "prd", INCREMENT_OUTPUT_MAPPING, format=format) - return prd diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py deleted file mode 100644 index 2775b1cb6..000000000 --- a/metagpt/actions/refine_project_management.py +++ /dev/null @@ -1,146 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -from typing import List, Union - -from metagpt.actions.action import Action -from metagpt.config import CONFIG -from metagpt.const import WORKSPACE_ROOT -from metagpt.utils.common import CodeParser -from metagpt.utils.get_template import get_template -from metagpt.utils.json_to_markdown import json_to_markdown - -templates = { - "json": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Format example -{format_example} ------ -Role: You are a project manager; the goal is to perform incremental development based on the context and the legacy. Break down tasks according to PRD/technical design, provide a Task list, and analyze task dependencies to start with the prerequisite modules. -Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. -Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -Output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT]. The following is the attribute description of the JSON object. - -## Required Python third-party packages: Provided as a python list, the requirements.txt format - -## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. - -## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first. - -Output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, -and only output the json inside this tag, nothing else -""", - "FORMAT_EXAMPLE": ''' -{ - "Required Python third-party packages": [ - "flask==1.1.2", - "bcrypt==3.2.0" - ], - "Full API spec": """ - openapi: 3.0.0 - ... - description: A JSON object ... - """, - "Task list": [ - "game.py" - ] -} -''', - }, - "markdown": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Format example -{format_example} ------ -Role: You are a project manager; the goal is to perform incremental development based on the context and the legacy. Break down tasks according to PRD/technical design, provide a Task list need to modified files, and analyze task dependencies to start with the prerequisite modules. -Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. -Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. - -## Required Python third-party packages: Provided as a python list, the requirements.txt format - -## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. - -## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first. -""", - "FORMAT_EXAMPLE": ''' ---- - -## Required Python third-party packages -```python -[ - "flask==1.1.2", - "bcrypt==3.2.0" -] -``` - -## Full API spec -```python -""" -openapi: 3.0.0 -... -description: A JSON object ... -""" -``` - -## Task list -```python -[ - "game.py", -] -``` ---- -''', - }, -} -OUTPUT_MAPPING = { - # "Incremental Requirements": (str, ...), - # ## Incremental Requirements: Provided as a str, the foremost incremental requirements for project management here based on the previous. - # "Difference Analysis": (Union[List[str], str], ...), - "Required Python third-party packages": (Union[List[str], str], ...), - "Full API spec": (str, ...), - # "Logic Analysis": (List[List[str]], ...), - "Task list": (List[str], ...), -} - - -class RefineTasks(Action): - def __init__(self, name="RefineTasks", context=None, llm=None): - super().__init__(name, context, llm) - - def _save(self, context, rsp): - if context[-1].instruct_content: - ws_name = context[-1].instruct_content.dict()["Python package name"] - else: - ws_name = CodeParser.parse_str(block="Python package name", text=context[-1].content) - file_path = WORKSPACE_ROOT / ws_name / "docs/api_spec_and_tasks.md" - file_path.write_text(json_to_markdown(rsp.instruct_content.dict())) - - # Write requirements.txt - requirements_path = WORKSPACE_ROOT / ws_name / "requirements.txt" - requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Required Python third-party packages"))) - - async def run(self, context, legacy, format=CONFIG.prompt_format): - prompt_template, format_example = get_template(templates, format) - prompt = prompt_template.format(context=context, - legacy=legacy, - format_example=format_example) - rsp = await self._aask_v1(prompt, "task", OUTPUT_MAPPING, format=format) - self._save(context, rsp) - return rsp - - -class AssignTasks(Action): - async def run(self, *args, **kwargs): - # Here you should implement the actual action - pass diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 4d0690e0f..fd6ad3eb1 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -21,6 +21,7 @@ from pydantic import Field from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions.action import Action +from metagpt.actions.write_code_guide_an import WRITE_CODE_INCREMENT_TEMPLATE from metagpt.config import CONFIG from metagpt.const import ( BUGFIX_FILENAME, @@ -114,20 +115,35 @@ class WriteCode(Action): test_detail = RunCodeResult.loads(test_doc.content) logs = test_detail.stderr + guideline = kwargs.get("guideline", "") if bug_feedback: code_context = coding_context.code_doc.content + elif guideline: + code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename, mode="guide") else: code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename) - prompt = PROMPT_TEMPLATE.format( - design=coding_context.design_doc.content if coding_context.design_doc else "", - tasks=coding_context.task_doc.content if coding_context.task_doc else "", - code=code_context, - logs=logs, - feedback=bug_feedback.content if bug_feedback else "", - filename=self.context.filename, - summary_log=summary_doc.content if summary_doc else "", - ) + if guideline: # guide write code 也有两种方式,进行尝试 + prompt = WRITE_CODE_INCREMENT_TEMPLATE.format( + guideline=guideline, + design=coding_context.design_doc.content if coding_context.design_doc else "", + tasks=coding_context.task_doc.content if coding_context.task_doc else "", + code=code_context, + logs=logs, + feedback=bug_feedback.content if bug_feedback else "", + filename=self.context.filename, + summary_log=summary_doc.content if summary_doc else "", + ) + else: + prompt = PROMPT_TEMPLATE.format( + design=coding_context.design_doc.content if coding_context.design_doc else "", + tasks=coding_context.task_doc.content if coding_context.task_doc else "", + code=code_context, + logs=logs, + feedback=bug_feedback.content if bug_feedback else "", + filename=self.context.filename, + summary_log=summary_doc.content if summary_doc else "", + ) logger.info(f"Writing {coding_context.filename}..") code = await self.write_code(prompt) if not coding_context.code_doc: @@ -138,7 +154,7 @@ class WriteCode(Action): return coding_context @staticmethod - async def get_codes(task_doc, exclude) -> str: + async def get_codes(task_doc, exclude, mode="normal") -> str: if not task_doc: return "" if not task_doc.content: @@ -147,11 +163,37 @@ class WriteCode(Action): code_filenames = m.get("Task list", []) codes = [] src_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.src_workspace) - for filename in code_filenames: - if filename == exclude: - continue - doc = await src_file_repo.get(filename=filename) - if not doc: - continue - codes.append(f"----- {filename}\n" + doc.content) + old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) + + src_files = src_file_repo.all_files + old_files = old_file_repo.all_files + union_files_list = list(set(src_files) | set(old_files)) + + if mode == "guide": + # 从两个repo中取code,并结合在一起 + for filename in union_files_list: + if filename == exclude: + if filename in old_files: + doc = await old_file_repo.get(filename=filename) # 使用原始代码 + else: + continue + + else: + doc = await src_file_repo.get(filename=filename) # 使用先前生成的代码 + if not doc: + if filename in old_files: + doc = await old_file_repo.get(filename=filename) # 使用原始代码 + else: + continue + codes.append(f"----- {filename}\n```{doc.content}```") + + else: + for filename in code_filenames: + if filename == exclude: + continue + doc = await src_file_repo.get(filename=filename) + if not doc: + continue + codes.append(f"----- {filename}\n```{doc.content}```") + return "\n".join(codes) diff --git a/metagpt/actions/write_code_guide.py b/metagpt/actions/write_code_guide.py deleted file mode 100644 index 3e51f0d2d..000000000 --- a/metagpt/actions/write_code_guide.py +++ /dev/null @@ -1,74 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- - -from metagpt.actions.action import Action -from metagpt.logs import logger -from metagpt.schema import Message -from metagpt.utils.common import CodeParser -from tenacity import retry, stop_after_attempt, wait_fixed - - -PROMPT_TEMPLATE = """ -NOTICE -Role: You are a professional software engineer, and your main task is to conduct incremental development, proposing incremental development plans and code guideance based on context and legacy code. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). Output format carefully referenced "Format example". - -## Regulations Review: To make the software directly operable without further coding, follow the regulations below during incremental development: -0) Determine the scope of responsibilities of each file and what classes and methods need to be implemented. -1) Import all referenced classes. -2) Implement all methods. -3) Add necessary explanation to all methods. -4) Ensure there are no potential bugs. -5) Confirm that the entire project conforms to the tasks proposed by the user. -6) Review the code thoroughly, checking for errors and validating the logic to ensure seamless user interaction without compromising any specified requirements. - -## Incremental Development Plan: Provided as a Python list containing `filename.py`. Proposed the detail and essential incremental development plan, based on the following context and legacy code by thinking and analyzing step by step. All incremental modules/functions need to be added to the corresponding code files. - -## Code Guidance: Propose the foremost guidelines that how to implement code of modification part for incremental development based on the above context, legacy code and incremental development plan. - ------ -# Context -{context} - -## Legacy Code -You are tasked with conducting incremental development in the existing code based on the provided legacy code and above information. -``` -{legacy} -``` ------ - -## Format example ------ -## Incremental Development Guide -[ - "`game.py` Contains `Game` and ...", -] - - -## Code Guidance -### Implementation `xx` in `xxx.py` ..., else retain the original xxx.py code. -```python -## xxx.py -... -``` ---- -### Implementation of the `Game` in `game.py` ..., else retain the original game.py code. -```python -## game.py -class Game: - ... -``` ------ -""" - - -class WriteCodeGuide(Action): - def __init__(self, name="WriteCodeGuide", context: list[Message] = None, llm=None): - super().__init__(name, context, llm) - - async def run(self, context, legacy): - prompt = PROMPT_TEMPLATE.format(context=context, legacy=legacy) - logger.info(f'Write Code Guide ..') - code_guide = await self._aask(prompt) - # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) - return code_guide - \ No newline at end of file diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guide_an.py new file mode 100644 index 000000000..df273aa2b --- /dev/null +++ b/metagpt/actions/write_code_guide_an.py @@ -0,0 +1,223 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2023/12/26 +@Author : mannaandpoem +@File : write_code_guide_an.py +""" +import asyncio + +from metagpt.actions.action_node import ActionNode + +from pydantic import Field + +from metagpt.actions.action import Action +from metagpt.llm import LLM +from metagpt.provider.base_gpt_api import BaseGPTAPI +from metagpt.schema import Document + + +GUIDELINE = ActionNode( + key="Code Guideline", + expected_type=list[str], + instruction="You are a professional software engineer, and your main task is to " + "proposing incremental development plans and code guidance", + example=[ + "`calculator.py` should be extended to include methods for subtraction, multiplication, and division. Error handling should be implemented for division to prevent division by zero.", + "New endpoints for subtraction, multiplication, and division should be added to `main.py`.", + ], +) + +INCREMENTAL_CHANGE = ActionNode( + key="Incremental Change", + expected_type=str, + instruction="Write Incremental Change by making a code draft that how to implement incremental development based on the context and Code Guideline.", + example="""1. Extend `Calculator` class in `calculator.py` with new methods for subtraction, multiplication, and division. +```python +## calculator.py +class Calculator: + ... + def subtract_numbers(self, num1: int, num2: int) -> int: + return num1 - num2 + def multiply_numbers(self, num1: int, num2: int) -> int: + return num1 * num2 + def divide_numbers(self, num1: int, num2: int) -> float: + if num2 == 0: + raise ValueError('Cannot divide by zero') + return num1 / num2 +``` +2. Implement new endpoints in `main.py` for the subtraction, multiplication, and division methods. +```python +## main.py +from flask import Flask, request, jsonify +from calculator import Calculator +app = Flask(__name__) +calculator = Calculator() +... +@app.route('/subtract_numbers', methods=['POST']) +def subtract_numbers(): + data = request.get_json() + num1 = data.get('num1', 0) + num2 = data.get('num2', 0) + result = calculator.subtract_numbers(num1, num2) + return jsonify({'result': result}), 200 +@app.route('/multiply_numbers', methods=['POST']) +def multiply_numbers(): + data = request.get_json() + num1 = data.get('num1', 0) + num2 = data.get('num2', 0) + result = calculator.multiply_numbers(num1, num2) + return jsonify({'result': result}), 200 +@app.route('/divide_numbers', methods=['POST']) +def divide_numbers(): + data = request.get_json() + num1 = data.get('num1', 1) + num2 = data.get('num2', 1) + try: + result = calculator.divide_numbers(num1, num2) + except ValueError as e: + return jsonify({'error': str(e)}), 400 + return jsonify({'result': result}), 200 +if __name__ == '__main__': + app.run() +```""" +) + + +CODE_GUIDE_CONTEXT = """ +NOTICE +Role: You are a professional software engineer, and your main task is to write code Guideline and code craft with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". +1. Determine the scope of responsibilities of each file and what classes and methods need to be implemented. +2. Import all referenced classes. +3. Implement all methods. +4. Add necessary explanation to all methods. +5. Ensure there are no potential bugs. +6. Confirm that the entire project conforms to the tasks proposed by the user. +7. Examine the code closely to find and fix errors, and confirm that the logic is sound to ensure smooth user interaction while meeting all specified requirements. +8. Attention: Legacy Code may be more or less files than Tasks List in Tasks. However, only code guidance and Incremental Change are written for the files in Tasks List. + +### Requirement +{requirement} + +### Design +{design} + +### Tasks +{tasks} + +### Legacy Code +{code} +""" + +WRITE_CODE_INCREMENT_TEMPLATE = """ +NOTICE +Role: You are a professional engineer; The main goal is to complete incremental development by combining Legacy Code and Guideline to rewrite the complete code. +Language: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese. +ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". + +# Context +## Guideline +{guideline} + +## Design +{design} + +## Tasks +{tasks} + +## Legacy Code +```Code +{code} +``` + +## Debug logs +```text +{logs} + +{summary_log} +``` + +## Bug Feedback logs +```text +{feedback} +``` + +# Format example +## Code: {filename} +```python +## {filename} +... +``` + +# Instruction: Based on the context, follow "Format example", write code. + +## Rewrite Complete Code: Only Write one file {filename}, Write code using triple quotes, based on the following attentions and context. +1. Only One file: do your best to implement THIS ONLY ONE FILE. +2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets. +3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import. +4. Follow design: YOU MUST FOLLOW "Data structures and interfaces". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design. +5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. +6. Before using a external variable/module, make sure you import it first. +7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. +8. Attention1: Implement the functions required by the current file scope of responsibility. +9. Attention2: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions. +""" + +CODE_GUIDE_CONTEXT_EXAMPLE = """ +### Legacy Code +## main.py + +from flask import Flask, request, jsonify +from calculator import Calculator + +app = Flask(__name__) +calculator = Calculator() + +@app.route('/add_numbers', methods=['POST']) +def add_numbers(): + data = request.get_json() + num1 = data.get('num1', 0) + num2 = data.get('num2', 0) + result = calculator.add_numbers(num1, num2) + return jsonify({'result': result}), 200 + +if __name__ == '__main__': + app.run() + +## calculator.py + +class Calculator: + def __init__(self, num1: int = 0, num2: int = 0): + self.num1 = num1 + self.num2 = num2 + + def add_numbers(self, num1: int, num2: int) -> int: + return num1 + num2 +""" + + +GUIDE_NODES = [ + GUIDELINE, + INCREMENTAL_CHANGE +] + +WRITE_CODE_GUIDE_NODE = ActionNode.from_children("WriteCodeGuide", GUIDE_NODES) + + +class WriteCodeGuide(Action): + name: str = "WriteCodeGuide" + context: Document = Field(default_factory=Document) + llm: BaseGPTAPI = Field(default_factory=LLM) + + async def run(self): + rsp = await WRITE_CODE_GUIDE_NODE.fill(context=CODE_GUIDE_CONTEXT, llm=self.llm, schema="json") + return rsp + + +def main(): + action = WriteCodeGuide() + return asyncio.run(action.run()) + + +if __name__ == "__main__": + main() diff --git a/metagpt/actions/write_code_refine.py b/metagpt/actions/write_code_refine.py deleted file mode 100644 index 5a70b2c68..000000000 --- a/metagpt/actions/write_code_refine.py +++ /dev/null @@ -1,67 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- - -from metagpt.actions.action import Action -from metagpt.logs import logger -from metagpt.schema import Message -from metagpt.utils.common import CodeParser -from tenacity import retry, stop_after_attempt, wait_fixed - -PROMPT_TEMPLATE = """ -NOTICE -Role: You are a professional engineer; your primary goal is to write PEP8 compliant, elegant, modular, easy-to-read, and maintainable Python 3.9 code (or any other programming language of your choice). -Requirements: Rewrite the complete code based on the Legacy Code so that it can be executed and avoid any potential bugs. You should modify the corresponding code based on the guidance. Output the complete code, fixing all errors according to the context. Ensure that you adhere to the specified guidelines for incremental development and modification of legacy code. -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format should be carefully referenced using the "Format example". Only output the current modified code, nothing else. In the modified code, if unchanged, you should output it, the complete code. - -## Rewrite Complete Code: Only Write one file {filename}, Write code using triple quotes, based on the following list, context, guidelines and legacy code. -1. Important: Do your best to implement ONLY ONE FILE. ONLY USE EXISTING API. IF NO API, IMPLEMENT IT. -2. Implement one of the following code files based on the provided context. Return the code in the specified format. Your code will be part of the entire project, so ensure it is complete, reliable, and reusable. -3. Attention1: Implement the functions required by the current file scope of responsibility. For example, main only needs to focus on the basic functions of main.py in the legacy code and the incremental functions to be implemented. Reuse existing code as much as possible. You can import functions from other codes instead of reimplementing the function. If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. -4. Attention2: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions, taking into account the guidelines, context, and existing legacy code. Retain the basic function methods from the legacy code, and make sure to preserve the existing code and logic that needs to be retained throughout the incremental development process. Avoid omitting any essential components. -5. Think before writing: What should be implemented and provided in this document? -6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. -7. Do not use public member functions that do not exist in your design. -8. The Modified Code is implemented according to the requirements, and there are no issues with the code logic. All functions in the Modified Code are fully implemented; none are omitted or incomplete. - ------ -# Context -{context} - ------ -## Guidelines: The foremost guidelines of modification for incremental development. -{guide} - ------ -## Legacy Code: The Legacy Code that needs to be modified. '===' is the separator of each code file in the legacy code. Basic function methods need to be retained in {filename}. -{legacy} - ------ -## Format example ------ -## Rewrite Complete Code: {filename} -```python -# {filename} -... -``` ------ -""" - - -class WriteCodeRefine(Action): - def __init__(self, name="WriteCodeRefine", context: list[Message] = None, llm=None): - super().__init__(name, context, llm) - - @retry(stop=stop_after_attempt(2), wait=wait_fixed(1)) - async def write_code(self, prompt): - code_rsp = await self._aask(prompt) - code = CodeParser.parse_code(block="", text=code_rsp) - return code - - async def run(self, context, legacy, filename, guide): - prompt = PROMPT_TEMPLATE.format(context=context, legacy=legacy, filename=filename, guide=guide) - logger.info(f'Code refine {filename}..') - code = await self.write_code(prompt) - # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) - # self._save(context, filename, code) - return code - \ No newline at end of file diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 8fd8959b2..4b3cf479a 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -237,7 +237,7 @@ class OpenAIGPTAPI(BaseGPTAPI, RateLimiter): "n": 1, "stop": None, "temperature": 0.3, - "timeout": 3, + "timeout": 30, "model": self.model, } if configs: diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index e0234f378..23e4d56ae 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -20,6 +20,7 @@ from __future__ import annotations import json +import os from collections import defaultdict from pathlib import Path from typing import Set @@ -27,6 +28,7 @@ from typing import Set from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks from metagpt.actions.fix_bug import FixBug from metagpt.actions.summarize_code import SummarizeCode +from metagpt.actions.write_code_guide_an import CODE_GUIDE_CONTEXT, WRITE_CODE_GUIDE_NODE, WriteCodeGuide from metagpt.config import CONFIG from metagpt.const import ( CODE_SUMMARIES_FILE_REPO, @@ -77,6 +79,7 @@ class Engineer(Role): ) n_borg: int = 1 use_code_review: bool = False + use_code_guide: bool = True code_todos: list = [] summarize_todos = [] @@ -84,14 +87,14 @@ class Engineer(Role): super().__init__(**kwargs) self._init_actions([WriteCode]) - self._watch([WriteTasks, SummarizeCode, WriteCode, WriteCodeReview, FixBug]) + self._watch([WriteTasks, SummarizeCode, WriteCode, WriteCodeReview, FixBug, WriteCodeGuide]) @staticmethod def _parse_tasks(task_msg: Document) -> list[str]: m = json.loads(task_msg.content) return m.get("Task list") - async def _act_sp_with_cr(self, review=False) -> Set[str]: + async def _act_sp_with_cr(self, review=False, guideline="") -> Set[str]: changed_files = set() src_file_repo = CONFIG.git_repo.new_file_repository(CONFIG.src_workspace) for todo in self.code_todos: @@ -102,7 +105,7 @@ class Engineer(Role): 3. Do we need other codes (currently needed)? TODO: The goal is not to need it. After clear task decomposition, based on the design idea, you should be able to write a single file without needing other codes. If you can't, it means you need a clearer definition. This is the key to writing longer code. """ - coding_context = await todo.run() + coding_context = await todo.run(guideline=guideline) # Code review if review: action = WriteCodeReview(context=coding_context, llm=self._llm) @@ -134,7 +137,11 @@ class Engineer(Role): return None async def _act_write_code(self): - changed_files = await self._act_sp_with_cr(review=self.use_code_review) + if self.use_code_guide: + code_guideline = await self._write_code_guideline() + changed_files = await self._act_sp_with_cr(review=self.use_code_review, guideline=code_guideline) + else: + changed_files = await self._act_sp_with_cr(review=self.use_code_review) return Message( content="\n".join(changed_files), role=self.profile, From a15e88f963bafb3de6ada2874adf6bb19d1e46c2 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 26 Dec 2023 21:36:37 +0800 Subject: [PATCH 234/637] Update increment development for 0.5.x version: Delete files with 'refine' and write_code_guide.py. Add write_code_guide_an.py. Update write_code.py for guiding write code. --- metagpt/actions/write_code_guide_an.py | 7 +++---- metagpt/roles/engineer.py | 27 ++++++++++++++++++++++++-- 2 files changed, 28 insertions(+), 6 deletions(-) diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guide_an.py index df273aa2b..d2739ba2e 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guide_an.py @@ -209,14 +209,13 @@ class WriteCodeGuide(Action): context: Document = Field(default_factory=Document) llm: BaseGPTAPI = Field(default_factory=LLM) - async def run(self): - rsp = await WRITE_CODE_GUIDE_NODE.fill(context=CODE_GUIDE_CONTEXT, llm=self.llm, schema="json") - return rsp + async def run(self, context): + return await WRITE_CODE_GUIDE_NODE.fill(context=context, llm=self.llm, schema="json") def main(): action = WriteCodeGuide() - return asyncio.run(action.run()) + return asyncio.run(action.run(CODE_GUIDE_CONTEXT)) if __name__ == "__main__": diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 23e4d56ae..3ed5f2e94 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -87,7 +87,7 @@ class Engineer(Role): super().__init__(**kwargs) self._init_actions([WriteCode]) - self._watch([WriteTasks, SummarizeCode, WriteCode, WriteCodeReview, FixBug, WriteCodeGuide]) + self._watch([WriteTasks, SummarizeCode, WriteCode, WriteCodeReview, FixBug]) @staticmethod def _parse_tasks(task_msg: Document) -> list[str]: @@ -220,7 +220,7 @@ class Engineer(Role): @staticmethod async def _new_coding_context( - filename, src_file_repo, task_file_repo, design_file_repo, dependency + filename, src_file_repo, task_file_repo, design_file_repo, dependency ) -> CodingContext: old_code_doc = await src_file_repo.get(filename) if not old_code_doc: @@ -308,3 +308,26 @@ class Engineer(Role): self.summarize_todos.append(SummarizeCode(context=ctx, llm=self._llm)) if self.summarize_todos: self._rc.todo = self.summarize_todos[0] + + async def _write_code_guideline(self): + logger.info("Writing code guideline..") + + requirement = str(self._rc.memory.get_by_role("Human")[0]) + task_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) + design_file_repo = CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO) + tasks = await task_file_repo.get_all()[0] + design = await design_file_repo.get_all()[0] + old_codes = await self.get_old_codes() + + context = CODE_GUIDE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) + node = await WriteCodeGuide().run(context=context) + guideline = node.instruct_content.json(ensure_ascii=False) + return guideline + + @staticmethod + async def get_old_codes() -> str: + CONFIG.old_workspace = CONFIG.git_repo.workdir / os.path.basename(CONFIG.project_path) + old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) + old_codes = await old_file_repo.get_all() + codes = [f"----- \n```{code.content}```" for code in old_codes] + return "\n".join(codes) From 263595b980e9bc34c225b762a84f2b968f1a91d1 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Wed, 27 Dec 2023 11:03:39 +0800 Subject: [PATCH 235/637] support load tools from file or file list --- metagpt/actions/write_analysis_code.py | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 9691f888f..2d9110e91 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -118,7 +118,13 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): schema_module = schema_module or 'udf' self.available_tools.update({schema_module: schema_path}) else: - yml_files = schema_path.glob("*.yml") + if isinstance(schema_path, list): + yml_files = schema_path + elif isinstance(schema_path, Path) and schema_path.is_file(): + yml_files = [schema_path] + else: + yml_files = schema_path.glob("*.yml") + for yml_file in yml_files: module = yml_file.stem with open(yml_file, "r", encoding="utf-8") as f: From d47cc0448f699f1edb684d110b0feb79613b8656 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 27 Dec 2023 12:20:15 +0800 Subject: [PATCH 236/637] Update engineer.py and prompt in write_code_guide_an.py. --- metagpt/actions/write_code_guide_an.py | 15 ++++++++++----- metagpt/roles/engineer.py | 18 ++++++++++++------ 2 files changed, 22 insertions(+), 11 deletions(-) diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guide_an.py index d2739ba2e..adf21220b 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guide_an.py @@ -85,7 +85,7 @@ if __name__ == '__main__': CODE_GUIDE_CONTEXT = """ -NOTICE +### NOTICE Role: You are a professional software engineer, and your main task is to write code Guideline and code craft with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". 1. Determine the scope of responsibilities of each file and what classes and methods need to be implemented. 2. Import all referenced classes. @@ -94,11 +94,14 @@ Role: You are a professional software engineer, and your main task is to write c 5. Ensure there are no potential bugs. 6. Confirm that the entire project conforms to the tasks proposed by the user. 7. Examine the code closely to find and fix errors, and confirm that the logic is sound to ensure smooth user interaction while meeting all specified requirements. -8. Attention: Legacy Code may be more or less files than Tasks List in Tasks. However, only code guidance and Incremental Change are written for the files in Tasks List. +8. Attention: Code files in the task list may have a different number of files compared to legacy code files. This requires integrating legacy code files that do not appear in the task list into the code files of the task list. Therefore, when writing code guidance and incremental changes for the code files in the task list, also include how to seamlessly merge and adjust legacy code files. ### Requirement {requirement} +### Prd +{prd} + ### Design {design} @@ -159,8 +162,8 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc 5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 6. Before using a external variable/module, make sure you import it first. 7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. -8. Attention1: Implement the functions required by the current file scope of responsibility. -9. Attention2: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions. +8. When Task List code files do not include Legacy Code files, you need to seamlessly merge and adjust Legacy Code files that do not appear in the Task List into the code files of the task list being written based on the guideline. +9. Attention: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions. """ CODE_GUIDE_CONTEXT_EXAMPLE = """ @@ -202,7 +205,9 @@ GUIDE_NODES = [ ] WRITE_CODE_GUIDE_NODE = ActionNode.from_children("WriteCodeGuide", GUIDE_NODES) - +# 1. 对最后的全部代码进行review一次,或者测试 +# 2. 将user requirement也作为write code的输入 +# 3. 对前置的action进行重新设计 class WriteCodeGuide(Action): name: str = "WriteCodeGuide" diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 3ed5f2e94..e2f4edba1 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -34,7 +34,7 @@ from metagpt.const import ( CODE_SUMMARIES_FILE_REPO, CODE_SUMMARIES_PDF_FILE_REPO, SYSTEM_DESIGN_FILE_REPO, - TASK_FILE_REPO, + TASK_FILE_REPO, PRDS_FILE_REPO, ) from metagpt.logs import logger from metagpt.roles import Role @@ -313,13 +313,19 @@ class Engineer(Role): logger.info("Writing code guideline..") requirement = str(self._rc.memory.get_by_role("Human")[0]) - task_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) + prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) design_file_repo = CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO) - tasks = await task_file_repo.get_all()[0] - design = await design_file_repo.get_all()[0] + task_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) + prd = await prd_file_repo.get_all() + prd = "\n".join([doc.content for doc in prd]) + design = await design_file_repo.get_all() + design = "\n".join([doc.content for doc in design]) + tasks = await task_file_repo.get_all() + tasks = "\n".join([doc.content for doc in tasks]) old_codes = await self.get_old_codes() - context = CODE_GUIDE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) + context = CODE_GUIDE_CONTEXT.format(requirement=requirement, prd=prd, tasks=tasks, design=design, + code=old_codes) node = await WriteCodeGuide().run(context=context) guideline = node.instruct_content.json(ensure_ascii=False) return guideline @@ -329,5 +335,5 @@ class Engineer(Role): CONFIG.old_workspace = CONFIG.git_repo.workdir / os.path.basename(CONFIG.project_path) old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) old_codes = await old_file_repo.get_all() - codes = [f"----- \n```{code.content}```" for code in old_codes] + codes = [f"----- {code.filename}\n```{code.content}```" for code in old_codes] return "\n".join(codes) From 99b1666bffb3e9c9a27b458efecc0762366527bc Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 27 Dec 2023 14:29:01 +0800 Subject: [PATCH 237/637] Update prompt in write_code_guide_an.py. --- metagpt/actions/write_code_guide_an.py | 17 +++++------------ 1 file changed, 5 insertions(+), 12 deletions(-) diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guide_an.py index adf21220b..e99e6b839 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guide_an.py @@ -7,21 +7,19 @@ """ import asyncio -from metagpt.actions.action_node import ActionNode - from pydantic import Field from metagpt.actions.action import Action +from metagpt.actions.action_node import ActionNode from metagpt.llm import LLM from metagpt.provider.base_gpt_api import BaseGPTAPI from metagpt.schema import Document - GUIDELINE = ActionNode( key="Code Guideline", expected_type=list[str], instruction="You are a professional software engineer, and your main task is to " - "proposing incremental development plans and code guidance", + "proposing incremental development plans and code guidance", example=[ "`calculator.py` should be extended to include methods for subtraction, multiplication, and division. Error handling should be implemented for division to prevent division by zero.", "New endpoints for subtraction, multiplication, and division should be added to `main.py`.", @@ -80,7 +78,7 @@ def divide_numbers(): return jsonify({'result': result}), 200 if __name__ == '__main__': app.run() -```""" +```""", ) @@ -199,15 +197,10 @@ class Calculator: """ -GUIDE_NODES = [ - GUIDELINE, - INCREMENTAL_CHANGE -] +GUIDE_NODES = [GUIDELINE, INCREMENTAL_CHANGE] WRITE_CODE_GUIDE_NODE = ActionNode.from_children("WriteCodeGuide", GUIDE_NODES) -# 1. 对最后的全部代码进行review一次,或者测试 -# 2. 将user requirement也作为write code的输入 -# 3. 对前置的action进行重新设计 + class WriteCodeGuide(Action): name: str = "WriteCodeGuide" From e65f16741c2dd843b1d2a78d7346f5e1da92ed38 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 27 Dec 2023 14:29:01 +0800 Subject: [PATCH 238/637] Update prompt in write_code_guide_an.py. --- metagpt/actions/write_code_guide_an.py | 19 +++++-------------- 1 file changed, 5 insertions(+), 14 deletions(-) diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guide_an.py index adf21220b..9f1e16405 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guide_an.py @@ -7,21 +7,19 @@ """ import asyncio -from metagpt.actions.action_node import ActionNode - from pydantic import Field from metagpt.actions.action import Action +from metagpt.actions.action_node import ActionNode from metagpt.llm import LLM from metagpt.provider.base_gpt_api import BaseGPTAPI from metagpt.schema import Document - GUIDELINE = ActionNode( key="Code Guideline", expected_type=list[str], instruction="You are a professional software engineer, and your main task is to " - "proposing incremental development plans and code guidance", + "proposing incremental development plans and code guidance", example=[ "`calculator.py` should be extended to include methods for subtraction, multiplication, and division. Error handling should be implemented for division to prevent division by zero.", "New endpoints for subtraction, multiplication, and division should be added to `main.py`.", @@ -80,10 +78,9 @@ def divide_numbers(): return jsonify({'result': result}), 200 if __name__ == '__main__': app.run() -```""" +```""", ) - CODE_GUIDE_CONTEXT = """ ### NOTICE Role: You are a professional software engineer, and your main task is to write code Guideline and code craft with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". @@ -198,16 +195,10 @@ class Calculator: return num1 + num2 """ - -GUIDE_NODES = [ - GUIDELINE, - INCREMENTAL_CHANGE -] +GUIDE_NODES = [GUIDELINE, INCREMENTAL_CHANGE] WRITE_CODE_GUIDE_NODE = ActionNode.from_children("WriteCodeGuide", GUIDE_NODES) -# 1. 对最后的全部代码进行review一次,或者测试 -# 2. 将user requirement也作为write code的输入 -# 3. 对前置的action进行重新设计 + class WriteCodeGuide(Action): name: str = "WriteCodeGuide" From 31c797279a73ebf020f15a9a893831856ffabb55 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 27 Dec 2023 14:41:13 +0800 Subject: [PATCH 239/637] Update prompt in write_code_guide_an.py. --- metagpt/actions/write_prd_an.py | 1 - 1 file changed, 1 deletion(-) diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index d58d72f64..dc1ae1dd1 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -136,7 +136,6 @@ REASON = ActionNode( key="reason", expected_type=str, instruction="Explain the reasoning process from question to answer", example="..." ) - NODES = [ LANGUAGE, PROGRAMMING_LANGUAGE, From 1ee35c930e4875ad212a7541c62a96994dd00ce9 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 27 Dec 2023 15:08:01 +0800 Subject: [PATCH 240/637] Delete and modify some files --- metagpt/actions/__init__.py | 1 - metagpt/actions/refine_design_api.py | 7 +- metagpt/actions/refine_prd.py | 9 ++- metagpt/actions/refine_project_management.py | 4 +- metagpt/actions/write_code.py | 2 +- metagpt/actions/write_code_guide.py | 74 -------------------- metagpt/actions/write_code_refine.py | 67 ------------------ metagpt/roles/engineer.py | 12 ++-- tests/metagpt/provider/test_zhipuai_api.py | 5 +- 9 files changed, 20 insertions(+), 161 deletions(-) delete mode 100644 metagpt/actions/write_code_guide.py delete mode 100644 metagpt/actions/write_code_refine.py diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index 99a4175f6..c34c72ed2 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -14,7 +14,6 @@ from metagpt.actions.debug_error import DebugError from metagpt.actions.design_api import WriteDesign from metagpt.actions.design_api_review import DesignReview from metagpt.actions.project_management import AssignTasks, WriteTasks -from metagpt.actions.refine import Refine from metagpt.actions.research import CollectLinks, WebBrowseAndSummarize, ConductResearch from metagpt.actions.run_code import RunCode from metagpt.actions.search_and_summarize import SearchAndSummarize diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py index d6a948b43..dba783323 100644 --- a/metagpt/actions/refine_design_api.py +++ b/metagpt/actions/refine_design_api.py @@ -7,7 +7,6 @@ from typing import List, Union from metagpt.actions import Action, ActionOutput from metagpt.config import CONFIG -from metagpt.const import WORKSPACE_ROOT from metagpt.logs import logger from metagpt.utils.common import CodeParser from metagpt.utils.get_template import get_template @@ -133,7 +132,7 @@ OUTPUT_MAPPING = { "Python package name": (str, ...), # "File list": (List[str], ...), "Data structures and interface definitions": (str, ...), - "Program call flow": (str, ...) + "Program call flow": (str, ...), } @@ -158,7 +157,7 @@ class RefineDesign(Action): original_workspace = workspace index = 1 while workspace.exists(): - ws_name_match = re.match(r'^(.*)_([\d]+)$', original_workspace.name) + ws_name_match = re.match(r"^(.*)_([\d]+)$", original_workspace.name) if ws_name_match: base_name, existing_index = ws_name_match.groups() index = int(existing_index) @@ -192,7 +191,7 @@ class RefineDesign(Action): logger.info(f"Saving System Designs to {system_design_file}") system_design_file.write_text((json_to_markdown(system_design.instruct_content.dict()))) - async def _save(self, context, system_design): + async def _save(self, context, system_design, WORKSPACE_ROOT=None): if isinstance(system_design, ActionOutput): ws_name = system_design.instruct_content.dict()["Python package name"] else: diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py index 1d8bab5f8..e7fb080f8 100644 --- a/metagpt/actions/refine_prd.py +++ b/metagpt/actions/refine_prd.py @@ -1,6 +1,7 @@ -from typing import List, Union +from typing import List -from metagpt.actions import Refine, ActionOutput, SearchAndSummarize +from metagpt.actions import SearchAndSummarize +from metagpt.actions.refine import Refine from metagpt.config import CONFIG from metagpt.logs import logger from metagpt.utils.get_template import get_template @@ -73,7 +74,6 @@ INCREMENT_OUTPUT_MAPPING = { class RefinePRD(Refine): - def __init__(self, name="RefinePRD", context=None, llm=None): super().__init__(name, context, llm) @@ -87,8 +87,7 @@ class RefinePRD(Refine): prompt_template, format_example = get_template(increment_template, format) prompt = prompt_template.format( - context=context, legacy=legacy, search_information=info, - format_example=format_example + context=context, legacy=legacy, search_information=info, format_example=format_example ) logger.debug(prompt) prd = await self._aask_v1(prompt, "prd", INCREMENT_OUTPUT_MAPPING, format=format) diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py index 2775b1cb6..1cd1d1e0f 100644 --- a/metagpt/actions/refine_project_management.py +++ b/metagpt/actions/refine_project_management.py @@ -132,9 +132,7 @@ class RefineTasks(Action): async def run(self, context, legacy, format=CONFIG.prompt_format): prompt_template, format_example = get_template(templates, format) - prompt = prompt_template.format(context=context, - legacy=legacy, - format_example=format_example) + prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) rsp = await self._aask_v1(prompt, "task", OUTPUT_MAPPING, format=format) self._save(context, rsp) return rsp diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index fd6ad3eb1..9f4cba300 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -123,7 +123,7 @@ class WriteCode(Action): else: code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename) - if guideline: # guide write code 也有两种方式,进行尝试 + if guideline: # guide write code 也有两种方式,进行尝试 prompt = WRITE_CODE_INCREMENT_TEMPLATE.format( guideline=guideline, design=coding_context.design_doc.content if coding_context.design_doc else "", diff --git a/metagpt/actions/write_code_guide.py b/metagpt/actions/write_code_guide.py deleted file mode 100644 index 3e51f0d2d..000000000 --- a/metagpt/actions/write_code_guide.py +++ /dev/null @@ -1,74 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- - -from metagpt.actions.action import Action -from metagpt.logs import logger -from metagpt.schema import Message -from metagpt.utils.common import CodeParser -from tenacity import retry, stop_after_attempt, wait_fixed - - -PROMPT_TEMPLATE = """ -NOTICE -Role: You are a professional software engineer, and your main task is to conduct incremental development, proposing incremental development plans and code guideance based on context and legacy code. Existing code and logic that need to be retained must also appear in the code after incremental development, do not omit it. Ensure that the code conforms to the PEP8 standards, is elegantly designed and modularized, easy to read and maintain, and is written in Python 3.9 (or in another programming language). Output format carefully referenced "Format example". - -## Regulations Review: To make the software directly operable without further coding, follow the regulations below during incremental development: -0) Determine the scope of responsibilities of each file and what classes and methods need to be implemented. -1) Import all referenced classes. -2) Implement all methods. -3) Add necessary explanation to all methods. -4) Ensure there are no potential bugs. -5) Confirm that the entire project conforms to the tasks proposed by the user. -6) Review the code thoroughly, checking for errors and validating the logic to ensure seamless user interaction without compromising any specified requirements. - -## Incremental Development Plan: Provided as a Python list containing `filename.py`. Proposed the detail and essential incremental development plan, based on the following context and legacy code by thinking and analyzing step by step. All incremental modules/functions need to be added to the corresponding code files. - -## Code Guidance: Propose the foremost guidelines that how to implement code of modification part for incremental development based on the above context, legacy code and incremental development plan. - ------ -# Context -{context} - -## Legacy Code -You are tasked with conducting incremental development in the existing code based on the provided legacy code and above information. -``` -{legacy} -``` ------ - -## Format example ------ -## Incremental Development Guide -[ - "`game.py` Contains `Game` and ...", -] - - -## Code Guidance -### Implementation `xx` in `xxx.py` ..., else retain the original xxx.py code. -```python -## xxx.py -... -``` ---- -### Implementation of the `Game` in `game.py` ..., else retain the original game.py code. -```python -## game.py -class Game: - ... -``` ------ -""" - - -class WriteCodeGuide(Action): - def __init__(self, name="WriteCodeGuide", context: list[Message] = None, llm=None): - super().__init__(name, context, llm) - - async def run(self, context, legacy): - prompt = PROMPT_TEMPLATE.format(context=context, legacy=legacy) - logger.info(f'Write Code Guide ..') - code_guide = await self._aask(prompt) - # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) - return code_guide - \ No newline at end of file diff --git a/metagpt/actions/write_code_refine.py b/metagpt/actions/write_code_refine.py deleted file mode 100644 index 5a70b2c68..000000000 --- a/metagpt/actions/write_code_refine.py +++ /dev/null @@ -1,67 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- - -from metagpt.actions.action import Action -from metagpt.logs import logger -from metagpt.schema import Message -from metagpt.utils.common import CodeParser -from tenacity import retry, stop_after_attempt, wait_fixed - -PROMPT_TEMPLATE = """ -NOTICE -Role: You are a professional engineer; your primary goal is to write PEP8 compliant, elegant, modular, easy-to-read, and maintainable Python 3.9 code (or any other programming language of your choice). -Requirements: Rewrite the complete code based on the Legacy Code so that it can be executed and avoid any potential bugs. You should modify the corresponding code based on the guidance. Output the complete code, fixing all errors according to the context. Ensure that you adhere to the specified guidelines for incremental development and modification of legacy code. -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format should be carefully referenced using the "Format example". Only output the current modified code, nothing else. In the modified code, if unchanged, you should output it, the complete code. - -## Rewrite Complete Code: Only Write one file {filename}, Write code using triple quotes, based on the following list, context, guidelines and legacy code. -1. Important: Do your best to implement ONLY ONE FILE. ONLY USE EXISTING API. IF NO API, IMPLEMENT IT. -2. Implement one of the following code files based on the provided context. Return the code in the specified format. Your code will be part of the entire project, so ensure it is complete, reliable, and reusable. -3. Attention1: Implement the functions required by the current file scope of responsibility. For example, main only needs to focus on the basic functions of main.py in the legacy code and the incremental functions to be implemented. Reuse existing code as much as possible. You can import functions from other codes instead of reimplementing the function. If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. -4. Attention2: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions, taking into account the guidelines, context, and existing legacy code. Retain the basic function methods from the legacy code, and make sure to preserve the existing code and logic that needs to be retained throughout the incremental development process. Avoid omitting any essential components. -5. Think before writing: What should be implemented and provided in this document? -6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. -7. Do not use public member functions that do not exist in your design. -8. The Modified Code is implemented according to the requirements, and there are no issues with the code logic. All functions in the Modified Code are fully implemented; none are omitted or incomplete. - ------ -# Context -{context} - ------ -## Guidelines: The foremost guidelines of modification for incremental development. -{guide} - ------ -## Legacy Code: The Legacy Code that needs to be modified. '===' is the separator of each code file in the legacy code. Basic function methods need to be retained in {filename}. -{legacy} - ------ -## Format example ------ -## Rewrite Complete Code: {filename} -```python -# {filename} -... -``` ------ -""" - - -class WriteCodeRefine(Action): - def __init__(self, name="WriteCodeRefine", context: list[Message] = None, llm=None): - super().__init__(name, context, llm) - - @retry(stop=stop_after_attempt(2), wait=wait_fixed(1)) - async def write_code(self, prompt): - code_rsp = await self._aask(prompt) - code = CodeParser.parse_code(block="", text=code_rsp) - return code - - async def run(self, context, legacy, filename, guide): - prompt = PROMPT_TEMPLATE.format(context=context, legacy=legacy, filename=filename, guide=guide) - logger.info(f'Code refine {filename}..') - code = await self.write_code(prompt) - # code_rsp = await self._aask_v1(prompt, "code_rsp", OUTPUT_MAPPING) - # self._save(context, filename, code) - return code - \ No newline at end of file diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index e2f4edba1..165ffadc2 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -28,13 +28,14 @@ from typing import Set from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks from metagpt.actions.fix_bug import FixBug from metagpt.actions.summarize_code import SummarizeCode -from metagpt.actions.write_code_guide_an import CODE_GUIDE_CONTEXT, WRITE_CODE_GUIDE_NODE, WriteCodeGuide +from metagpt.actions.write_code_guide_an import CODE_GUIDE_CONTEXT, WriteCodeGuide from metagpt.config import CONFIG from metagpt.const import ( CODE_SUMMARIES_FILE_REPO, CODE_SUMMARIES_PDF_FILE_REPO, + PRDS_FILE_REPO, SYSTEM_DESIGN_FILE_REPO, - TASK_FILE_REPO, PRDS_FILE_REPO, + TASK_FILE_REPO, ) from metagpt.logs import logger from metagpt.roles import Role @@ -220,7 +221,7 @@ class Engineer(Role): @staticmethod async def _new_coding_context( - filename, src_file_repo, task_file_repo, design_file_repo, dependency + filename, src_file_repo, task_file_repo, design_file_repo, dependency ) -> CodingContext: old_code_doc = await src_file_repo.get(filename) if not old_code_doc: @@ -324,8 +325,9 @@ class Engineer(Role): tasks = "\n".join([doc.content for doc in tasks]) old_codes = await self.get_old_codes() - context = CODE_GUIDE_CONTEXT.format(requirement=requirement, prd=prd, tasks=tasks, design=design, - code=old_codes) + context = CODE_GUIDE_CONTEXT.format( + requirement=requirement, prd=prd, tasks=tasks, design=design, code=old_codes + ) node = await WriteCodeGuide().run(context=context) guideline = node.instruct_content.json(ensure_ascii=False) return guideline diff --git a/tests/metagpt/provider/test_zhipuai_api.py b/tests/metagpt/provider/test_zhipuai_api.py index dc8b63cc3..8ce0f8f63 100644 --- a/tests/metagpt/provider/test_zhipuai_api.py +++ b/tests/metagpt/provider/test_zhipuai_api.py @@ -36,9 +36,12 @@ async def test_zhipuai_acompletion(mocker): assert resp["code"] == 200 assert "chatglm-turbo" in resp["data"]["choices"][0]["content"] + def test_zhipuai_proxy(mocker): import openai + from metagpt.config import CONFIG - CONFIG.openai_proxy = 'http://127.0.0.1:8080' + + CONFIG.openai_proxy = "http://127.0.0.1:8080" _ = ZhiPuAIGPTAPI() assert openai.proxy == CONFIG.openai_proxy From 8952deaa8b65c2f9be3e47bc552cfcd8478c8d34 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 27 Dec 2023 15:17:21 +0800 Subject: [PATCH 241/637] Delete files with 'refine' --- metagpt/actions/refine.py | 10 - metagpt/actions/refine_design_api.py | 221 ------------------- metagpt/actions/refine_prd.py | 94 -------- metagpt/actions/refine_project_management.py | 144 ------------ 4 files changed, 469 deletions(-) delete mode 100644 metagpt/actions/refine.py delete mode 100644 metagpt/actions/refine_design_api.py delete mode 100644 metagpt/actions/refine_prd.py delete mode 100644 metagpt/actions/refine_project_management.py diff --git a/metagpt/actions/refine.py b/metagpt/actions/refine.py deleted file mode 100644 index beea40fc8..000000000 --- a/metagpt/actions/refine.py +++ /dev/null @@ -1,10 +0,0 @@ -from metagpt.actions import Action - - -# 增量开发动作的基类 -class Refine(Action): - def __init__(self, name="Refine", context=None, llm=None): - super().__init__(name, context, llm) - - def run(self, *args, **kwargs): - raise NotImplementedError diff --git a/metagpt/actions/refine_design_api.py b/metagpt/actions/refine_design_api.py deleted file mode 100644 index dba783323..000000000 --- a/metagpt/actions/refine_design_api.py +++ /dev/null @@ -1,221 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -import re -import shutil -from pathlib import Path -from typing import List, Union - -from metagpt.actions import Action, ActionOutput -from metagpt.config import CONFIG -from metagpt.logs import logger -from metagpt.utils.common import CodeParser -from metagpt.utils.get_template import get_template -from metagpt.utils.json_to_markdown import json_to_markdown -from metagpt.utils.mermaid import mermaid_to_file - -templates = { - "json": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Format example -{format_example} ------ -Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and legacy design. Make the best use of good open source tools. -Requirement: Fill in the following missing information based on the context, each section name is a key in json. Output exactly as shown in the example, including single and double quotes. -Max Output: 8192 chars or 2048 tokens. Try to use them up. - -## Incremental implementation approach: Provide as Python list[str]. Analyze the difficult points of the requirements, select the appropriate open-source framework. Up to 5. - -## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores - -## Data structures and interface definitions: Use single quotes to wrap content. Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. - -## Program call flow: Use single quotes to wrap content. Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. - -output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example. -Output exactly as shown in the example, including single and double quotes, and only output the json inside this tag, nothing else -""", - "FORMAT_EXAMPLE": """ -[CONTENT] -{ - "Incremental implementation approach": ["We will ...",], - "Python package name": "new_name", - "Data structures and interface definitions": ' - classDiagram - class Game{ - +int score - } - ... - Game "1" -- "1" Food: has - ', - "Program call flow": ' - sequenceDiagram - participant M as Main - ... - G->>M: end game - ' -} -[/CONTENT] -""", - }, - "markdown": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Format example -{format_example} ------ -Role: You are an architect; the goal is to perform incremental development and design a state-of-the-art (SOTA) PEP8-compliant Python system based on the context and legacy design. Make the best use of good open source tools. -Requirement: Fill in the following missing information based on the context, note that all sections are response with code form separately. Output exactly as shown in the example, including single and double quotes. -Max Output: 8192 chars or 2048 tokens. Try to use them up. -Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. - -## Incremental implementation approach: Provide as Python list[str]. Analyze the difficult points of the requirements, select the appropriate open-source framework. Up to 5. - -## Python package name: Provide as Python str with python triple quoto, concise and clear, characters only use a combination of all lowercase and underscores - -## Data structures and interface definitions: Use single quotes to wrap content. Use mermaid classDiagram code syntax, including classes (INCLUDING __init__ method) and functions (with type annotations), CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design. - -## Program call flow: Use single quotes to wrap content. Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT. - -""", - "FORMAT_EXAMPLE": """ ---- - -## Incremental implementation approach -```python -[ - "We will ...", -] -``` - -## Python package name -```python -"new_name" -``` - -## Data structures and interface definitions -```mermaid -classDiagram - class Game{ - +int score - } - ... - Game "1" -- "1" Food: has -``` - -## Program call flow -```mermaid -sequenceDiagram - participant M as Main - ... - G->>M: end game -``` ---- -""", - }, -} - -OUTPUT_MAPPING = { - # "Incremental Requirements": (str, ...), - # "Difference Description": (Union[List[str], str], ...), - "Incremental implementation approach": (Union[List[str], str], ...), - "Python package name": (str, ...), - # "File list": (List[str], ...), - "Data structures and interface definitions": (str, ...), - "Program call flow": (str, ...), -} - - -class RefineDesign(Action): - def __init__(self, name, context=None, llm=None): - super().__init__(name, context, llm) - self.desc = ( - "Based on the PRD, think about the system design, and design the corresponding APIs, " - "data structures, library tables, processes, and paths. Please provide your design, feedback " - "clearly and in detail." - ) - - def recreate_workspace(self, workspace: Path): - try: - shutil.rmtree(workspace) - except FileNotFoundError: - pass # Folder does not exist, but we don't care - workspace.mkdir(parents=True, exist_ok=True) - - def create_or_increment_workspace(self, workspace: Path): - # 如果工作空间已存在,添加数字以区分 - original_workspace = workspace - index = 1 - while workspace.exists(): - ws_name_match = re.match(r"^(.*)_([\d]+)$", original_workspace.name) - if ws_name_match: - base_name, existing_index = ws_name_match.groups() - index = int(existing_index) - index += 1 - workspace = original_workspace.parent / f"{base_name}_{index}" - else: - workspace = original_workspace.parent / f"{original_workspace.name}_{index}" - index += 1 - - # 创建工作空间,包括所有必要的父文件夹 - workspace.mkdir(parents=True, exist_ok=True) - return workspace - - async def _save_prd(self, docs_path, resources_path, context): - prd_file = docs_path / "prd.md" - - if context[-1].instruct_content: - logger.info(f"Saving PRD to {prd_file}") - prd_file.write_text(json_to_markdown(context[-1].instruct_content.dict())) - - async def _save_system_design(self, docs_path, resources_path, system_design): - data_api_design = system_design.instruct_content.dict()[ - "Data structures and interface definitions" - ] # CodeParser.parse_code(block="Data structures and interface definitions", text=content) - seq_flow = system_design.instruct_content.dict()[ - "Program call flow" - ] # CodeParser.parse_code(block="Program call flow", text=content) - await mermaid_to_file(data_api_design, resources_path / "data_api_design") - await mermaid_to_file(seq_flow, resources_path / "seq_flow") - system_design_file = docs_path / "system_design.md" - logger.info(f"Saving System Designs to {system_design_file}") - system_design_file.write_text((json_to_markdown(system_design.instruct_content.dict()))) - - async def _save(self, context, system_design, WORKSPACE_ROOT=None): - if isinstance(system_design, ActionOutput): - ws_name = system_design.instruct_content.dict()["Python package name"] - else: - ws_name = CodeParser.parse_str(block="Python package name", text=system_design) - workspace = WORKSPACE_ROOT / ws_name - # workspace = self.create_or_increment_workspace(workspace) - self.recreate_workspace(workspace) - docs_path = workspace / "docs" - resources_path = workspace / "resources" - docs_path.mkdir(parents=True, exist_ok=True) - resources_path.mkdir(parents=True, exist_ok=True) - await self._save_prd(docs_path, resources_path, context) - await self._save_system_design(docs_path, resources_path, system_design) - - async def run(self, context, legacy, format=CONFIG.prompt_format): - prompt_template, format_example = get_template(templates, format) - prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) - # system_design = await self._aask(prompt) - system_design = await self._aask_v1(prompt, "system_design", OUTPUT_MAPPING, format=format) - # fix Python package name, we can't system_design.instruct_content.python_package_name = "xxx" since "Python package name" contain space, have to use setattr - setattr( - system_design.instruct_content, - "Python package name", - system_design.instruct_content.dict()["Python package name"].strip().strip("'").strip('"'), - ) - await self._save(context, system_design) - return system_design diff --git a/metagpt/actions/refine_prd.py b/metagpt/actions/refine_prd.py deleted file mode 100644 index e7fb080f8..000000000 --- a/metagpt/actions/refine_prd.py +++ /dev/null @@ -1,94 +0,0 @@ -from typing import List - -from metagpt.actions import SearchAndSummarize -from metagpt.actions.refine import Refine -from metagpt.config import CONFIG -from metagpt.logs import logger -from metagpt.utils.get_template import get_template - -increment_template = { - "json": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Search Information -{search_information} - -## Format example -{format_example} ------ -Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. -Requirements: According to the context, fill in the following missing information, each section name is a key in json ,If the requirements are unclear, ensure minimum viability and avoid excessive design. Only output one json, nothing else. - -## Incremental Development Analysis: Provide as Python list[str], up to 5. incremental development analysis and plans based on the context and the legacy. If the requirement itself is simple, the Incremental Development Analysis should also be simple. - -output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, -and only output the json inside this tag, nothing else -""", - "FORMAT_EXAMPLE": """ -[CONTENT] -{ - "Incremental Development Analysis": [], -} -[/CONTENT] -""", - }, - "markdown": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Search Information -{search_information} - -## Format example -{format_example} ------ -Role: You are a professional Product Manager tasked with overseeing incremental development and crafting Product Requirements Documents (PRDs) for a concise, usable, and efficient product. -Requirements: According to the context, fill in the following missing information, note that each sections are returned in Python code triple quote form seperatedly. If the requirements are unclear, ensure minimum viability and avoid excessive design -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. AND '## ' SHOULD WRITE BEFORE the code and triple quote. Output carefully referenced "Format example" in format.Only output one json, nothing else. - -## Incremental Development Analysis: Provide as Python list[str], up to 5. Incremental development analysis and plans based on the context and the legacy. If the requirement itself is simple, the Incremental Development Analysis should also be simple. -""", - "FORMAT_EXAMPLE": """ ---- - -## Incremental Development Analysis -[ - "We will ...", -] -""", - }, -} - -INCREMENT_OUTPUT_MAPPING = { - "Incremental Development Analysis": (List[str], ...), -} - - -class RefinePRD(Refine): - def __init__(self, name="RefinePRD", context=None, llm=None): - super().__init__(name, context, llm) - - async def run(self, context, legacy, format=CONFIG.prompt_format, *args, **kwargs): - sas = SearchAndSummarize() - rsp = "" - info = f"### Search Results\n{sas.result}\n\n### Search Summary\n{rsp}" - if sas.result: - logger.info(sas.result) - logger.info(rsp) - - prompt_template, format_example = get_template(increment_template, format) - prompt = prompt_template.format( - context=context, legacy=legacy, search_information=info, format_example=format_example - ) - logger.debug(prompt) - prd = await self._aask_v1(prompt, "prd", INCREMENT_OUTPUT_MAPPING, format=format) - return prd diff --git a/metagpt/actions/refine_project_management.py b/metagpt/actions/refine_project_management.py deleted file mode 100644 index 1cd1d1e0f..000000000 --- a/metagpt/actions/refine_project_management.py +++ /dev/null @@ -1,144 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -from typing import List, Union - -from metagpt.actions.action import Action -from metagpt.config import CONFIG -from metagpt.const import WORKSPACE_ROOT -from metagpt.utils.common import CodeParser -from metagpt.utils.get_template import get_template -from metagpt.utils.json_to_markdown import json_to_markdown - -templates = { - "json": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Format example -{format_example} ------ -Role: You are a project manager; the goal is to perform incremental development based on the context and the legacy. Break down tasks according to PRD/technical design, provide a Task list, and analyze task dependencies to start with the prerequisite modules. -Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. -Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. -Output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT]. The following is the attribute description of the JSON object. - -## Required Python third-party packages: Provided as a python list, the requirements.txt format - -## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. - -## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first. - -Output a properly formatted JSON, wrapped inside [CONTENT][/CONTENT] like format example, -and only output the json inside this tag, nothing else -""", - "FORMAT_EXAMPLE": ''' -{ - "Required Python third-party packages": [ - "flask==1.1.2", - "bcrypt==3.2.0" - ], - "Full API spec": """ - openapi: 3.0.0 - ... - description: A JSON object ... - """, - "Task list": [ - "game.py" - ] -} -''', - }, - "markdown": { - "PROMPT_TEMPLATE": """ -# Context -{context} - -## Legacy -{legacy} - -## Format example -{format_example} ------ -Role: You are a project manager; the goal is to perform incremental development based on the context and the legacy. Break down tasks according to PRD/technical design, provide a Task list need to modified files, and analyze task dependencies to start with the prerequisite modules. -Requirements: Based on the context and the Legacy Project Management and Legacy Code, fill in the following missing information. Note that Please try your best to reuse legacy code, and all sections are returned in Python code triple quote form seperatedly. Here the granularity of the task is a file that need to modified. -Attention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the code and triple quote. - -## Required Python third-party packages: Provided as a python list, the requirements.txt format - -## Full API spec: Use OpenAPI 3.0. Describe all APIs that may be used by both frontend and backend based on the previous. - -## Task list: Provided as Python list[str]. Each str is a filename, the more at the beginning, the more it is a prerequisite dependency, should be done first. -""", - "FORMAT_EXAMPLE": ''' ---- - -## Required Python third-party packages -```python -[ - "flask==1.1.2", - "bcrypt==3.2.0" -] -``` - -## Full API spec -```python -""" -openapi: 3.0.0 -... -description: A JSON object ... -""" -``` - -## Task list -```python -[ - "game.py", -] -``` ---- -''', - }, -} -OUTPUT_MAPPING = { - # "Incremental Requirements": (str, ...), - # ## Incremental Requirements: Provided as a str, the foremost incremental requirements for project management here based on the previous. - # "Difference Analysis": (Union[List[str], str], ...), - "Required Python third-party packages": (Union[List[str], str], ...), - "Full API spec": (str, ...), - # "Logic Analysis": (List[List[str]], ...), - "Task list": (List[str], ...), -} - - -class RefineTasks(Action): - def __init__(self, name="RefineTasks", context=None, llm=None): - super().__init__(name, context, llm) - - def _save(self, context, rsp): - if context[-1].instruct_content: - ws_name = context[-1].instruct_content.dict()["Python package name"] - else: - ws_name = CodeParser.parse_str(block="Python package name", text=context[-1].content) - file_path = WORKSPACE_ROOT / ws_name / "docs/api_spec_and_tasks.md" - file_path.write_text(json_to_markdown(rsp.instruct_content.dict())) - - # Write requirements.txt - requirements_path = WORKSPACE_ROOT / ws_name / "requirements.txt" - requirements_path.write_text("\n".join(rsp.instruct_content.dict().get("Required Python third-party packages"))) - - async def run(self, context, legacy, format=CONFIG.prompt_format): - prompt_template, format_example = get_template(templates, format) - prompt = prompt_template.format(context=context, legacy=legacy, format_example=format_example) - rsp = await self._aask_v1(prompt, "task", OUTPUT_MAPPING, format=format) - self._save(context, rsp) - return rsp - - -class AssignTasks(Action): - async def run(self, *args, **kwargs): - # Here you should implement the actual action - pass From 81934e2202d436c8379767282522fda810d14020 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 27 Dec 2023 15:33:33 +0800 Subject: [PATCH 242/637] update --- metagpt/actions/write_prd_an.py | 1 + metagpt/provider/openai_api.py | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index dc1ae1dd1..d58d72f64 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -136,6 +136,7 @@ REASON = ActionNode( key="reason", expected_type=str, instruction="Explain the reasoning process from question to answer", example="..." ) + NODES = [ LANGUAGE, PROGRAMMING_LANGUAGE, diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 3c8f094d3..0b6fdd869 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -237,7 +237,7 @@ class OpenAIGPTAPI(BaseGPTAPI, RateLimiter): "n": 1, "stop": None, "temperature": 0.3, - "timeout": 30, + "timeout": 3, "model": self.model, } if configs: From db4e3343f111986f5e1954da2ee775d986ba58dc Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 28 Dec 2023 20:17:33 +0800 Subject: [PATCH 243/637] general planner, code interpreter --- metagpt/actions/ask_review.py | 62 ++++++ metagpt/actions/ml_da_action.py | 56 ----- metagpt/actions/write_analysis_code.py | 2 +- metagpt/plan/__init__.py | 1 + metagpt/plan/planner.py | 109 ++++++++++ metagpt/prompts/ml_engineer.py | 11 - metagpt/roles/code_interpreter.py | 80 +++++++ metagpt/roles/ml_engineer.py | 287 ++++++------------------- metagpt/roles/ml_engineer_simple.py | 2 +- metagpt/schema.py | 24 ++- metagpt/utils/recovery_util.py | 2 +- tests/metagpt/roles/test_daml.py | 10 +- tests/metagpt/test_schema.py | 3 +- 13 files changed, 338 insertions(+), 311 deletions(-) create mode 100644 metagpt/actions/ask_review.py create mode 100644 metagpt/plan/__init__.py create mode 100644 metagpt/plan/planner.py create mode 100644 metagpt/roles/code_interpreter.py diff --git a/metagpt/actions/ask_review.py b/metagpt/actions/ask_review.py new file mode 100644 index 000000000..eec5e49aa --- /dev/null +++ b/metagpt/actions/ask_review.py @@ -0,0 +1,62 @@ +from typing import List + +from metagpt.actions import Action +from metagpt.schema import Message, Plan +from metagpt.logs import logger + + +class ReviewConst: + TASK_REVIEW_TRIGGER = "task" + CODE_REVIEW_TRIGGER = "code" + CONTINUE_WORD = ["confirm", "continue", "c", "yes", "y"] + CHANGE_WORD = ["change"] + EXIT_WORD = ["exit"] + TASK_REVIEW_INSTRUCTION = ( + f"If you want to change, add, delete a task or merge tasks in the plan, say '{CHANGE_WORD[0]} task task_id or current task, ... (things to change)' " + f"If you confirm the output from the current task and wish to continue, type: {CONTINUE_WORD[0]}" + ) + CODE_REVIEW_INSTRUCTION = ( + f"If you want the codes to be rewritten, say '{CHANGE_WORD[0]} ... (your change advice)' " + f"If you want to leave it as is, type: {CONTINUE_WORD[0]} or {CONTINUE_WORD[1]}" + ) + EXIT_INSTRUCTION = f"If you want to terminate the process, type: {EXIT_WORD[0]}" + + +class AskReview(Action): + async def run( + self, context: List[Message], plan: Plan = None, trigger: str = "task" + ): + logger.info("Current overall plan:") + logger.info( + "\n".join( + [ + f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" + for task in plan.tasks + ] + ) + ) + + logger.info("most recent context:") + latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" + review_instruction = ( + ReviewConst.TASK_REVIEW_INSTRUCTION + if trigger == ReviewConst.TASK_REVIEW_TRIGGER + else ReviewConst.CODE_REVIEW_INSTRUCTION + ) + prompt = ( + f"This is a <{trigger}> review. Please review output from {latest_action}\n" + f"{review_instruction}\n" + f"{ReviewConst.EXIT_INSTRUCTION}\n" + "Please type your review below:\n" + ) + + rsp = input(prompt) + + if rsp.lower() in ReviewConst.EXIT_WORD: + exit() + + # Confirmation can be one of "confirm", "continue", "c", "yes", "y" exactly, or sentences containing "confirm". + # One could say "confirm this task, but change the next task to ..." + confirmed = rsp.lower() in ReviewConst.CONTINUE_WORD or ReviewConst.CONTINUE_WORD[0] in rsp.lower() + + return rsp, confirmed diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py index b6270f12f..50d1d2420 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_da_action.py @@ -10,62 +10,6 @@ from metagpt.prompts.ml_engineer import ( PRINT_DATA_COLUMNS ) -class ReviewConst: - TASK_REVIEW_TRIGGER = "task" - CODE_REVIEW_TRIGGER = "code" - CONTINUE_WORD = ["confirm", "continue", "c", "yes", "y"] - CHANGE_WORD = ["change"] - EXIT_WORD = ["exit"] - TASK_REVIEW_INSTRUCTION = ( - f"If you want to change, add, delete a task or merge tasks in the plan, say '{CHANGE_WORD[0]} task task_id or current task, ... (things to change)' " - f"If you confirm the output from the current task and wish to continue, type: {CONTINUE_WORD[0]}" - ) - CODE_REVIEW_INSTRUCTION = ( - f"If you want the codes to be rewritten, say '{CHANGE_WORD[0]} ... (your change advice)' " - f"If you want to leave it as is, type: {CONTINUE_WORD[0]} or {CONTINUE_WORD[1]}" - ) - EXIT_INSTRUCTION = f"If you want to terminate the process, type: {EXIT_WORD[0]}" - - -class AskReview(Action): - async def run( - self, context: List[Message], plan: Plan = None, trigger: str = "task" - ): - logger.info("Current overall plan:") - logger.info( - "\n".join( - [ - f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" - for task in plan.tasks - ] - ) - ) - - logger.info("most recent context:") - latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" - review_instruction = ( - ReviewConst.TASK_REVIEW_INSTRUCTION - if trigger == ReviewConst.TASK_REVIEW_TRIGGER - else ReviewConst.CODE_REVIEW_INSTRUCTION - ) - prompt = ( - f"This is a <{trigger}> review. Please review output from {latest_action}\n" - f"{review_instruction}\n" - f"{ReviewConst.EXIT_INSTRUCTION}\n" - "Please type your review below:\n" - ) - - rsp = input(prompt) - - if rsp.lower() in ReviewConst.EXIT_WORD: - exit() - - # Confirmation can be one of "confirm", "continue", "c", "yes", "y" exactly, or sentences containing "confirm". - # One could say "confirm this task, but change the next task to ..." - confirmed = rsp.lower() in ReviewConst.CONTINUE_WORD or ReviewConst.CONTINUE_WORD[0] in rsp.lower() - - return rsp, confirmed - class SummarizeAnalysis(Action): PROMPT_TEMPLATE = """ diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 2d9110e91..21add3159 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -277,7 +277,7 @@ class MakeTools(WriteCodeByGenerate): saved_path.write_text(tool_code, encoding='utf-8') @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) - async def run(self, code: str | List[dict], code_desc: str = None, **kwargs) -> str: + async def run(self, code: Union[str, List[dict]], code_desc: str = None, **kwargs) -> str: # 拼接code prompt code_prompt = f"The following code is about {code_desc}, convert it to be a General Function, {code}" if not self.context: diff --git a/metagpt/plan/__init__.py b/metagpt/plan/__init__.py new file mode 100644 index 000000000..5ad35e100 --- /dev/null +++ b/metagpt/plan/__init__.py @@ -0,0 +1 @@ +from metagpt.plan.planner import Planner \ No newline at end of file diff --git a/metagpt/plan/planner.py b/metagpt/plan/planner.py new file mode 100644 index 000000000..c2b430817 --- /dev/null +++ b/metagpt/plan/planner.py @@ -0,0 +1,109 @@ +import json + +from metagpt.logs import logger +from metagpt.memory import Memory +from metagpt.schema import Message, Plan, Task +from metagpt.actions.ask_review import AskReview, ReviewConst +from metagpt.actions.write_plan import WritePlan, update_plan_from_rsp, precheck_update_plan_from_rsp + + +STRUCTURAL_CONTEXT = """ +## User Requirement +{user_requirement} +## Context +{context} +## Current Plan +{tasks} +## Current Task +{current_task} +""" + + +class Planner: + def __init__(self, goal: str, working_memory: Memory, auto_run: bool = False): + self.plan = Plan(goal=goal) + self.auto_run = auto_run + + # memory for working on each task, discarded each time a task is done + self.working_memory = working_memory + + @property + def current_task(self): + return self.plan.current_task + + @property + def current_task_id(self): + return self.plan.current_task_id + + async def ask_review(self, task_to_review: Task = None, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): + """ + Ask to review the task result, reviewer needs to provide confirmation or request change. + If human confirms the task result, then we deem the task completed, regardless of whether the code run succeeds; + if auto mode, then the code run has to succeed for the task to be considered completed. + """ + auto_run = auto_run or self.auto_run + if not auto_run: + context = self.get_useful_memories() + review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan, trigger=trigger) + if not confirmed: + self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) + return review, confirmed + confirmed = task_to_review.is_success if task_to_review else True + return "", confirmed + + async def confirm_task(self, task, updated_task, review): + assert updated_task.task_id == task.task_id + self.plan.replace_task(updated_task) + self.plan.finish_current_task() + self.working_memory.clear() + + confirmed_and_more = (ReviewConst.CONTINUE_WORD[0] in review.lower() + and review.lower() not in ReviewConst.CONTINUE_WORD[0]) # "confirm, ... (more content, such as changing downstream tasks)" + if confirmed_and_more: + self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) + await self.update_plan(review) + + async def update_plan(self, review: str = "", max_tasks: int = 3, max_retries: int = 3, **kwargs): + plan_confirmed = False + while not plan_confirmed: + context = self.get_useful_memories() + rsp = await WritePlan().run( + context, max_tasks=max_tasks, **kwargs + ) + self.working_memory.add( + Message(content=rsp, role="assistant", cause_by=WritePlan) + ) + + # precheck plan before asking reviews + is_plan_valid, error = precheck_update_plan_from_rsp(rsp, self.plan) + if not is_plan_valid and max_retries > 0: + error_msg = f"The generated plan is not valid with error: {error}, try regenerating, remember to generate either the whole plan or the single changed task only" + logger.warning(error_msg) + self.working_memory.add(Message(content=error_msg, role="assistant", cause_by=WritePlan)) + max_retries -= 1 + continue + + _, plan_confirmed = await self.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) + + update_plan_from_rsp(rsp, self.plan) + + self.working_memory.clear() + + def get_useful_memories(self, task_exclude_field=None) -> list[Message]: + """find useful memories only to reduce context length and improve performance""" + # TODO dataset description , code steps + if task_exclude_field is None: + # Shorten the context as we don't need code steps after we get the codes. + # This doesn't affect current_task below, which should hold the code steps + task_exclude_field = {'code_steps'} + user_requirement = self.plan.goal + context = self.plan.context + tasks = [task.dict(exclude=task_exclude_field) for task in self.plan.tasks] + tasks = json.dumps(tasks, indent=4, ensure_ascii=False) + current_task = self.plan.current_task.json() if self.plan.current_task else {} + context = STRUCTURAL_CONTEXT.format( + user_requirement=user_requirement, context=context, tasks=tasks, current_task=current_task + ) + context_msg = [Message(content=context, role="user")] + + return context_msg + self.working_memory.get() diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 6af40bf97..c4b0ad8ae 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -309,14 +309,3 @@ ML_MODULE_MAP = { "feature_engineering": "metagpt.tools.functions.libs.feature_engineering", "udf": "metagpt.tools.functions.libs.udf", } - -STRUCTURAL_CONTEXT = """ -## User Requirement -{user_requirement} -## Data Description -{data_desc} -## Current Plan -{tasks} -## Current Task -{current_task} -""" diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py new file mode 100644 index 000000000..32f530548 --- /dev/null +++ b/metagpt/roles/code_interpreter.py @@ -0,0 +1,80 @@ +import json +from datetime import datetime + +from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.ask_review import ReviewConst +from metagpt.actions.write_analysis_code import WriteCodeByGenerate +from metagpt.logs import logger +from metagpt.roles import Role +from metagpt.schema import Message, Task +from metagpt.utils.save_code import save_code_file + + +class CodeInterpreter(Role): + def __init__( + self, name="Charlie", profile="CodeInterpreter", goal="", auto_run=False, + ): + super().__init__(name=name, profile=profile, goal=goal) + self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run) + self.execute_code = ExecutePyCode() + + @property + def working_memory(self): + return self._rc.working_memory + + async def _plan_and_act(self): + + rsp = await super()._plan_and_act() + + # save code using datetime.now or keywords related to the goal of your project (plan.goal). + project_record = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") + save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") + + return rsp + + async def _act_on_task(self, current_task) -> Task: + code, result, success = await self._write_and_exec_code() + task_copy_with_result = current_task.copy( + update={"code": code, "result": result, "is_success": success}, + deep=True + ) + return task_copy_with_result + + async def _write_and_exec_code(self, max_retry: int = 3): + + counter = 0 + success = False + + while not success and counter < max_retry: + context = self.planner.get_useful_memories() + + logger.info("Write code with pure generation") + + code = await WriteCodeByGenerate().run( + context=context, plan=self.planner.plan, temperature=0.0 + ) + cause_by = WriteCodeByGenerate + + self.working_memory.add( + Message(content=code, role="assistant", cause_by=cause_by) + ) + + result, success = await self.execute_code.run(code) + print(result) + + self.working_memory.add( + Message(content=result, role="user", cause_by=ExecutePyCode) + ) + + if "!pip" in code: + success = False + + counter += 1 + + if not success and counter >= max_retry: + logger.info("coding failed!") + review, _ = await self.planner.ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) + if ReviewConst.CHANGE_WORD[0] in review: + counter = 0 # redo the task again with help of human suggestions + + return code, result, success diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index aaace9693..e29d8fce5 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,135 +1,62 @@ -from typing import List import json -from datetime import datetime - -import fire from metagpt.actions.debug_code import DebugCode from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis, Reflect, ReviewConst, UpdateDataColumns +from metagpt.actions.ask_review import ReviewConst from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools, MakeTools from metagpt.actions.write_code_steps import WriteCodeSteps -from metagpt.actions.write_plan import WritePlan -from metagpt.actions.write_plan import update_plan_from_rsp, precheck_update_plan_from_rsp -from metagpt.const import DATA_PATH, PROJECT_ROOT +from metagpt.const import PROJECT_ROOT from metagpt.logs import logger -from metagpt.memory import Memory -from metagpt.prompts.ml_engineer import STRUCTURAL_CONTEXT -from metagpt.roles import Role -from metagpt.roles.kaggle_manager import DownloadData, SubmitResult -from metagpt.schema import Message, Plan -from metagpt.utils.save_code import save_code_file -from metagpt.utils.recovery_util import save_history, load_history +from metagpt.schema import Message from metagpt.utils.common import remove_comments +from metagpt.actions.ml_da_action import SummarizeAnalysis, Reflect, UpdateDataColumns +from metagpt.roles.code_interpreter import CodeInterpreter +from metagpt.roles.kaggle_manager import DownloadData, SubmitResult +from metagpt.tools.functions.libs.udf import UDFS_YAML -class MLEngineer(Role): +class MLEngineer(CodeInterpreter): def __init__( - self, name="ABC", profile="MLEngineer", goal="", auto_run: bool = False, use_tools=False, use_code_steps=False, + self, name="Mark", profile="MLEngineer", goal="", auto_run=False, use_tools=False, use_code_steps=False, + make_udfs=False, use_udfs=False ): - super().__init__(name=name, profile=profile, goal=goal) - self._set_react_mode(react_mode="plan_and_act") + super().__init__(name=name, profile=profile, goal=goal, auto_run=auto_run) self._watch([DownloadData, SubmitResult]) - - self.plan = Plan(goal=goal) - self.make_udfs = False # user-defined functions - self.use_udfs = False - self.execute_code = ExecutePyCode() - self.auto_run = auto_run + self.use_tools = use_tools self.use_code_steps = use_code_steps + self.make_udfs = make_udfs # user-defined functions + self.use_udfs = use_udfs self.data_desc = {} - - # memory for working on each task, discarded each time a task is done - self.working_memory = Memory() async def _plan_and_act(self): - ### Actions in a multi-agent multi-turn setting ### + ### Actions in a multi-agent multi-turn setting, a new attempt on the data ### memories = self.get_memories() if memories: latest_event = memories[-1].cause_by if latest_event == DownloadData: - self.plan.context = memories[-1].content + self.planner.plan.context = memories[-1].content elif latest_event == SubmitResult: # self reflect on previous plan outcomes and think about how to improve the plan, add to working memory await self._reflect() # get feedback for improvement from human, add to working memory - await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) + await self.planner.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - ### Common Procedure in both single- and multi-agent setting ### - # create initial plan and update until confirmation - await self._update_plan() + ### general plan process ### + await super()._plan_and_act() - while self.plan.current_task: - task = self.plan.current_task - logger.info(f"ready to take on task {task}") - - # take on current task - code, result, success = await self._write_and_exec_code() - - # ask for acceptance, users can other refuse and change tasks in the plan - review, task_result_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - - if self.auto_run: - # if human confirms the task result, then we deem the task completed, regardless of whether the code run succeeds; - # if auto mode, then the code run has to succeed for the task to be considered completed - task_result_confirmed = success - - if task_result_confirmed: - # tick off this task and record progress - task.code = code - task.result = result - self.plan.finish_current_task() - self.working_memory.clear() - - if (self.use_tools and task.task_type not in ['model_train', 'model_evaluate']) or self.use_udfs: - success, new_code = await self._update_data_columns() - if success: - task.code = task.code + "\n\n" + new_code - - confirmed_and_more = (ReviewConst.CONTINUE_WORD[0] in review.lower() - and review.lower() not in ReviewConst.CONTINUE_WORD[0]) # "confirm, ... (more content, such as changing downstream tasks)" - if confirmed_and_more: - self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) - await self._update_plan(review) - - elif "redo" in review: - # Ask the Role to redo this task with help of review feedback, - # useful when the code run is successful but the procedure or result is not what we want - continue - - else: - # update plan according to user's feedback and to take on changed tasks - await self._update_plan(review) - - completed_plan_memory = self.get_useful_memories() # completed plan as a outcome - self._rc.memory.add(completed_plan_memory[0]) # add to persistent memory - - summary = await SummarizeAnalysis().run(self.plan) + ### summarize analysis ### + summary = await SummarizeAnalysis().run(self.planner.plan) rsp = Message(content=summary, cause_by=SummarizeAnalysis) self._rc.memory.add(rsp) - # save code using datetime.now or keywords related to the goal of your project (plan.goal). - project_record = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") - save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") return rsp - - async def _update_data_columns(self): - rsp = await UpdateDataColumns().run(self.plan) - is_update, code = rsp["is_update"], rsp["code"] - success = False - if is_update: - result, success = await self.execute_code.run(code) - if success: - print(result) - self.data_desc["column_info"] = result - return success, code - + async def _write_and_exec_code(self, max_retry: int = 3): - self.plan.current_task.code_steps = ( - await WriteCodeSteps().run(self.plan) + self.planner.current_task.code_steps = ( + await WriteCodeSteps().run(self.planner.plan) if self.use_code_steps else "" ) @@ -139,46 +66,49 @@ class MLEngineer(Role): debug_context = [] while not success and counter < max_retry: - context = self.get_useful_memories() + + context = self.planner.get_useful_memories() + if counter > 0 and (self.use_tools or self.use_udfs): logger.warning('We got a bug code, now start to debug...') code = await DebugCode().run( - plan=self.plan.current_task.instruction, + plan=self.planner.current_task.instruction, code=code, runtime_result=self.working_memory.get(), context=debug_context ) logger.info(f"new code \n{code}") cause_by = DebugCode + elif (not self.use_tools and not self.use_udfs) or ( - self.plan.current_task.task_type == 'other' and not self.use_udfs): + self.planner.current_task.task_type == 'other' and not self.use_udfs): logger.info("Write code with pure generation") - # TODO: 添加基于current_task.instruction-code_path的k-v缓存 code = await WriteCodeByGenerate().run( - context=context, plan=self.plan, temperature=0.0 + context=context, plan=self.planner.plan, temperature=0.0 ) - debug_context = [self.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] + debug_context = [self.planner.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] cause_by = WriteCodeByGenerate + else: logger.info("Write code with tools") if self.use_udfs: # use user-defined function tools. - from metagpt.tools.functions.libs.udf import UDFS_YAML logger.warning("Writing code with user-defined function tools by WriteCodeWithTools.") logger.info(f"Local user defined function as following:\ \n{json.dumps(list(UDFS_YAML.keys()), indent=2, ensure_ascii=False)}") # set task_type to `udf` - self.plan.current_task.task_type = 'udf' + self.planner.current_task.task_type = 'udf' schema_path = UDFS_YAML else: schema_path = PROJECT_ROOT / "metagpt/tools/functions/schemas" tool_context, code = await WriteCodeWithTools(schema_path=schema_path).run( context=context, - plan=self.plan, + plan=self.planner.plan, column_info=self.data_desc.get("column_info", ""), ) debug_context = tool_context cause_by = WriteCodeWithTools + self.working_memory.add( Message(content=code, role="assistant", cause_by=cause_by) ) @@ -200,47 +130,29 @@ class MLEngineer(Role): if not success and counter >= max_retry: logger.info("coding failed!") - review, _ = await self._ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) + review, _ = await self.planner.ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions - + + if success: + if (self.use_tools and self.planner.current_task.task_type not in ['model_train', 'model_evaluate']) or self.use_udfs: + update_success, new_code = await self._update_data_columns() + if update_success: + code = code + "\n\n" + new_code + return code, result, success - async def _ask_review(self, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): - auto_run = auto_run or self.auto_run - if not auto_run: - context = self.get_useful_memories() - review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan, trigger=trigger) - if not confirmed: - self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) - return review, confirmed - return "", True - - async def _update_plan(self, review: str = "", max_tasks: int = 3, max_retries: int = 3): - plan_confirmed = False - while not plan_confirmed: - context = self.get_useful_memories() - rsp = await WritePlan().run( - context, max_tasks=max_tasks, use_tools=self.use_tools - ) - self.working_memory.add( - Message(content=rsp, role="assistant", cause_by=WritePlan) - ) - - # precheck plan before asking reviews - is_plan_valid, error = precheck_update_plan_from_rsp(rsp, self.plan) - if not is_plan_valid and max_retries > 0: - error_msg = f"The generated plan is not valid with error: {error}, try regenerating, remember to generate either the whole plan or the single changed task only" - logger.warning(error_msg) - self.working_memory.add(Message(content=error_msg, role="assistant", cause_by=WritePlan)) - max_retries -= 1 - continue - - _, plan_confirmed = await self._ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - - update_plan_from_rsp(rsp, self.plan) - - self.working_memory.clear() + async def _update_data_columns(self): + logger.info("Check columns in updated data") + rsp = await UpdateDataColumns().run(self.planner.plan) + is_update, code = rsp["is_update"], rsp["code"] + success = False + if is_update: + result, success = await self.execute_code.run(code) + if success: + print(result) + self.data_desc["column_info"] = result + return success, code async def _reflect(self): context = self.get_memories() @@ -249,34 +161,6 @@ class MLEngineer(Role): reflection = await Reflect().run(context=context) self.working_memory.add(Message(content=reflection, role="assistant")) self.working_memory.add(Message(content=Reflect.REWRITE_PLAN_INSTRUCTION, role="user")) - - def get_useful_memories(self, task_exclude_field=None) -> List[Message]: - """find useful memories only to reduce context length and improve performance""" - # TODO dataset description , code steps - if task_exclude_field is None: - # Shorten the context as we don't need code steps after we get the codes. - # This doesn't affect current_task below, which should hold the code steps - task_exclude_field = {'code_steps'} - user_requirement = self.plan.goal - data_desc = self.plan.context - tasks = [task.dict(exclude=task_exclude_field) for task in self.plan.tasks] - tasks = json.dumps(tasks, indent=4, ensure_ascii=False) - current_task = self.plan.current_task.json() if self.plan.current_task else {} - context = STRUCTURAL_CONTEXT.format( - user_requirement=user_requirement, data_desc=data_desc, tasks=tasks, current_task=current_task - ) - context_msg = [Message(content=context, role="user")] - - return context_msg + self.get_working_memories() - - def get_working_memories(self) -> List[Message]: - return self.working_memory.get() - - def reset(self): - """Restart role with the same goal.""" - self.plan = Plan(goal=self.plan.goal) - self.execute_code = ExecutePyCode() - self.working_memory = Memory() async def make_tools(self, code: str): """Make user-defined functions(udfs, aka tools) for pure generation code. @@ -284,17 +168,17 @@ class MLEngineer(Role): Args: code (str): pure generation code by class WriteCodeByGenerate. """ - logger.warning(f"Making tools for task_id {self.plan.current_task_id}: \ - `{self.plan.current_task.instruction}` \n code: \n {code}") + logger.warning(f"Making tools for task_id {self.planner.current_task_id}: \ + `{self.planner.current_task.instruction}` \n code: \n {code}") make_tools = MakeTools() make_tool_retries, make_tool_current_retry = 3, 0 while True: # start make tools - tool_code = await make_tools.run(code, self.plan.current_task.instruction) + tool_code = await make_tools.run(code, self.planner.current_task.instruction) make_tool_current_retry += 1 # check tool_code by execute_code - logger.info(f"Checking task_id {self.plan.current_task_id} tool code by executor...") + logger.info(f"Checking task_id {self.planner.current_task_id} tool code by executor...") execute_result, execute_success = await self.execute_code.run(tool_code) if not execute_success: logger.error(f"Tool code faild to execute, \n{execute_result}\n.We will try to fix it ...") @@ -302,60 +186,9 @@ class MLEngineer(Role): if execute_success or make_tool_current_retry >= make_tool_retries: if make_tool_current_retry >= make_tool_retries: logger.error(f"We have tried the maximum number of attempts {make_tool_retries}\ - and still have not created tools for task_id {self.plan.current_task_id} successfully,\ + and still have not created tools for task_id {self.planner.current_task_id} successfully,\ we will skip it.") break # save successful tool code in udf if execute_success: make_tools.save(tool_code) - - -if __name__ == "__main__": - requirement = "Perform data analysis on the provided data. Train a model to predict the target variable Survived. Include data preprocessing, feature engineering, and modeling in your pipeline. The metric is accuracy." - - data_path = f"{DATA_PATH}/titanic" - requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - requirement = f"Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" - requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." - - # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" - # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report AUC Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." - - data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" - requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - save_dir = "" - # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" - - async def main(requirement: str = requirement, auto_run: bool = True, use_tools: bool = False, use_code_steps: bool = False, save_dir: str = ""): - """ - The main function to run the MLEngineer with optional history loading. - - Args: - requirement (str): The requirement for the MLEngineer. - auto_run (bool): Whether to auto-run the MLEngineer. - save_dir (str): The directory from which to load the history or to save the new history. - - Raises: - Exception: If an error occurs during execution, log the error and save the history. - """ - if save_dir: - logger.info("Resuming from history trajectory") - plan, nb = load_history(save_dir) - role = MLEngineer(goal=requirement, auto_run=auto_run, use_tools=use_tools, use_code_steps=use_code_steps) - role.plan = Plan(**plan) - role.execute_code = ExecutePyCode(nb) - - else: - logger.info("Run from scratch") - role = MLEngineer(goal=requirement, auto_run=auto_run, use_tools=use_tools, use_code_steps=use_code_steps) - - try: - await role.run(requirement) - except Exception as e: - - save_path = save_history(role, save_dir) - - logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") - - fire.Fire(main) diff --git a/metagpt/roles/ml_engineer_simple.py b/metagpt/roles/ml_engineer_simple.py index cc7d8fc97..7214e37c2 100644 --- a/metagpt/roles/ml_engineer_simple.py +++ b/metagpt/roles/ml_engineer_simple.py @@ -10,7 +10,7 @@ from metagpt.schema import Message from metagpt.memory import Memory from metagpt.logs import logger from metagpt.actions.write_analysis_code import WriteCodeByGenerate -from metagpt.actions.ml_da_action import AskReview, ReviewConst +from metagpt.actions.ask_review import AskReview, ReviewConst from metagpt.actions.execute_code import ExecutePyCode from metagpt.roles.kaggle_manager import DownloadData from metagpt.utils.save_code import save_code_file diff --git a/metagpt/schema.py b/metagpt/schema.py index 8eb7e31ca..f46da0fde 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -81,6 +81,7 @@ class Task(BaseModel): code_steps: str = "" code: str = "" result: str = "" + is_success: bool = False is_finished: bool = False @@ -169,6 +170,7 @@ class Plan(BaseModel): task = self.task_map[task_id] task.code = "" task.result = "" + task.is_success = False task.is_finished = False def replace_task(self, new_task: Task): @@ -181,18 +183,18 @@ class Plan(BaseModel): Returns: None """ - if new_task.task_id in self.task_map: - # Replace the task in the task map and the task list - self.task_map[new_task.task_id] = new_task - for i, task in enumerate(self.tasks): - if task.task_id == new_task.task_id: - self.tasks[i] = new_task - break + assert new_task.task_id in self.task_map + # Replace the task in the task map and the task list + self.task_map[new_task.task_id] = new_task + for i, task in enumerate(self.tasks): + if task.task_id == new_task.task_id: + self.tasks[i] = new_task + break - # Reset dependent tasks - for task in self.tasks: - if new_task.task_id in task.dependent_task_ids: - self.reset_task(task.task_id) + # Reset dependent tasks + for task in self.tasks: + if new_task.task_id in task.dependent_task_ids: + self.reset_task(task.task_id) def append_task(self, new_task: Task): """ diff --git a/metagpt/utils/recovery_util.py b/metagpt/utils/recovery_util.py index afe7fc021..cef302d6b 100644 --- a/metagpt/utils/recovery_util.py +++ b/metagpt/utils/recovery_util.py @@ -46,7 +46,7 @@ def save_history(role: Role, save_dir: str = ""): # overwrite exist trajectory save_path.mkdir(parents=True, exist_ok=True) - plan = role.plan.dict() + plan = role.planner.plan.dict() with open(save_path / "plan.json", "w", encoding="utf-8") as plan_file: json.dump(plan, plan_file, indent=4, ensure_ascii=False) diff --git a/tests/metagpt/roles/test_daml.py b/tests/metagpt/roles/test_daml.py index 55b425316..dbb4fb38f 100644 --- a/tests/metagpt/roles/test_daml.py +++ b/tests/metagpt/roles/test_daml.py @@ -2,8 +2,14 @@ import pytest from tqdm import tqdm from metagpt.logs import logger -from metagpt.roles.ml_engineer import MLEngineer +from metagpt.schema import Plan +from metagpt.roles.ml_engineer import MLEngineer, ExecutePyCode +def reset(role): + """Restart role with the same goal.""" + role.working_memory.clear() + role.planner.plan = Plan(goal=role.planner.plan.goal) + role.execute_code = ExecutePyCode() async def make_use_tools(requirement: str, auto_run: bool = True): """make and use tools for requirement.""" @@ -15,7 +21,7 @@ async def make_use_tools(requirement: str, auto_run: bool = True): role.use_udfs = False await role.run(requirement) # use udfs - role.reset() + reset(role) role.make_udfs = False role.use_udfs = True role.use_code_steps = False diff --git a/tests/metagpt/test_schema.py b/tests/metagpt/test_schema.py index b5d49b7a1..65fa7574d 100644 --- a/tests/metagpt/test_schema.py +++ b/tests/metagpt/test_schema.py @@ -141,7 +141,8 @@ class TestPlan: task = Task(task_id="1", instruction="First Task") plan.add_tasks([task]) new_task = Task(task_id="2", instruction="New Task") - plan.replace_task(new_task) # Task with ID 2 does not exist in plan + with pytest.raises(AssertionError): + plan.replace_task(new_task) # Task with ID 2 does not exist in plan assert "1" in plan.task_map assert "2" not in plan.task_map From 0b6b3a0df625c61a5b55668c18d6c82b5ee2d488 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 28 Dec 2023 20:24:43 +0800 Subject: [PATCH 244/637] support plan and act in role --- metagpt/roles/kaggle_manager.py | 3 +- metagpt/roles/role.py | 60 ++++++++++++++- tests/metagpt/roles/run_code_interpreter.py | 85 +++++++++++++++++++++ 3 files changed, 143 insertions(+), 5 deletions(-) create mode 100644 tests/metagpt/roles/run_code_interpreter.py diff --git a/metagpt/roles/kaggle_manager.py b/metagpt/roles/kaggle_manager.py index 18ac6733a..cad12a16a 100644 --- a/metagpt/roles/kaggle_manager.py +++ b/metagpt/roles/kaggle_manager.py @@ -10,7 +10,8 @@ from metagpt.config import CONFIG from metagpt.const import WORKSPACE_ROOT from metagpt.roles import Role from metagpt.actions import Action, BossRequirement -from metagpt.actions.ml_da_action import AskReview, SummarizeAnalysis +from metagpt.actions.ask_review import AskReview +from metagpt.actions.ml_da_action import SummarizeAnalysis from metagpt.schema import Message, Task, Plan from metagpt.logs import logger from metagpt.utils.common import CodeParser diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index b96c361c0..8c68a7ab4 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -18,7 +18,8 @@ from metagpt.actions import Action, ActionOutput from metagpt.llm import LLM, HumanProvider from metagpt.logs import logger from metagpt.memory import Memory, LongTermMemory -from metagpt.schema import Message +from metagpt.schema import Message, Task +from metagpt.plan.planner import Planner PREFIX_TEMPLATE = """You are a {profile}, named {name}, your goal is {goal}, and the constraint is {constraints}. """ @@ -79,6 +80,7 @@ class RoleContext(BaseModel): env: 'Environment' = Field(default=None) memory: Memory = Field(default_factory=Memory) long_term_memory: LongTermMemory = Field(default_factory=LongTermMemory) + working_memory: Memory = Field(default_factory=Memory) state: int = Field(default=-1) # -1 indicates initial or termination state where todo is None todo: Action = Field(default=None) watch: set[Type[Action]] = Field(default_factory=set) @@ -115,6 +117,7 @@ class Role: self._actions = [] self._role_id = str(self._setting) self._rc = RoleContext() + self.planner = None def _reset(self): self._states = [] @@ -134,7 +137,7 @@ class Role: self._actions.append(i) self._states.append(f"{idx}. {action}") - def _set_react_mode(self, react_mode: str, max_react_loop: int = 1): + def _set_react_mode(self, react_mode: str, max_react_loop: int = 1, auto_run: bool = True): """Set strategy of the Role reacting to observed Message. Variation lies in how this Role elects action to perform during the _think stage, especially if it is capable of multiple Actions. @@ -154,6 +157,8 @@ class Role: self._rc.react_mode = react_mode if react_mode == RoleReactMode.REACT: self._rc.max_react_loop = max_react_loop + elif react_mode == RoleReactMode.PLAN_AND_ACT: + self.planner = Planner(goal=self._setting.goal, working_memory=self._rc.working_memory, auto_run=auto_run) def _watch(self, actions: Iterable[Type[Action]]): """Listen to the corresponding behaviors""" @@ -274,8 +279,55 @@ class Role: async def _plan_and_act(self) -> Message: """first plan, then execute an action sequence, i.e. _think (of a plan) -> _act -> _act -> ... Use llm to come up with the plan dynamically.""" - # TODO: to be implemented - return Message("") + + ### Common Procedure in both single- and multi-agent setting ### + # create initial plan and update until confirmation + await self.planner.update_plan() + + while self.planner.current_task: + task = self.planner.current_task + logger.info(f"ready to take on task {task}") + + # take on current task + task_copy_with_result = await self._act_on_task(task) + + # ask for acceptance, users can other refuse and change tasks in the plan + review, task_result_confirmed = await self.planner.ask_review(task_copy_with_result) + + if task_result_confirmed: + # tick off this task and record progress + await self.planner.confirm_task(task, task_copy_with_result, review) + + elif "redo" in review: + # Ask the Role to redo this task with help of review feedback, + # useful when the code run is successful but the procedure or result is not what we want + continue + + else: + # update plan according to user's feedback and to take on changed tasks + await self.planner.update_plan(review) + + completed_plan_memory = self.planner.get_useful_memories() # completed plan as a outcome + + rsp = completed_plan_memory[0] + + self._rc.memory.add(rsp) # add to persistent memory + + return rsp + + async def _act_on_task(self, current_task: Task) -> Task: + """Taking specific action to handle one task in plan + + Args: + current_task (Task): current task to take on + + Raises: + NotImplementedError: Specific Role must implement this method if expected to use planner + + Returns: + Task: A copy of the current task with result from actions + """ + raise NotImplementedError async def react(self) -> Message: """Entry to one of three strategies by which Role reacts to the observed Message""" diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py new file mode 100644 index 000000000..daa6bbe05 --- /dev/null +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -0,0 +1,85 @@ +import fire + +from metagpt.actions.execute_code import ExecutePyCode +from metagpt.const import DATA_PATH +from metagpt.logs import logger +from metagpt.roles.code_interpreter import CodeInterpreter +from metagpt.roles.ml_engineer import MLEngineer +from metagpt.schema import Plan +from metagpt.utils.recovery_util import save_history, load_history + + +async def run_code_interpreter(role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir): + """ + The main function to run the MLEngineer with optional history loading. + + Args: + requirement (str): The requirement for the MLEngineer. + auto_run (bool): Whether to auto-run the MLEngineer. + save_dir (str): The directory from which to load the history or to save the new history. + + Raises: + Exception: If an error occurs during execution, log the error and save the history. + """ + + if role_class == "ci": + role = CodeInterpreter(goal=requirement, auto_run=auto_run) + else: + role = MLEngineer( + goal=requirement, auto_run=auto_run, use_tools=use_tools, use_code_steps=use_code_steps, + make_udfs=make_udfs, use_udfs=use_udfs + ) + + if save_dir: + logger.info("Resuming from history trajectory") + plan, nb = load_history(save_dir) + role.planner.plan = Plan(**plan) + role.execute_code = ExecutePyCode(nb) + + else: + logger.info("Run from scratch") + + + try: + await role.run(requirement) + except Exception as e: + + save_path = save_history(role, save_dir) + + logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") + + +if __name__ == "__main__": + requirement = "Run data analysis on sklearn Iris dataset, include a plot" + # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" + # data_path = f"{DATA_PATH}/titanic" + # requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" + # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." + # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" + # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report AUC Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." + # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" + # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + + save_dir = "" + # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" + + role_class = "ci" + # role_class = "mle" + auto_run = True + # auto_run = False + # use_tools = True + use_tools = False + # make_udfs = True + make_udfs = False + # use_udfs = True + use_udfs = False + + async def main( + role_class: str = role_class, requirement: str = requirement, auto_run: bool = auto_run, + use_tools: bool = use_tools, use_code_steps: bool = False, make_udfs: bool = make_udfs, use_udfs: bool = use_udfs, + save_dir: str = save_dir + ): + await run_code_interpreter(role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir) + + fire.Fire(main) From 850c3ec0943603ddb2dda05ed73842cd089bf33a Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 28 Dec 2023 21:55:52 +0800 Subject: [PATCH 245/637] Modify prompt in write_code_guide_an.py and get_codes function in write_code.py --- metagpt/actions/write_code.py | 18 +++++++++--------- metagpt/actions/write_code_guide_an.py | 8 +++----- 2 files changed, 12 insertions(+), 14 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 9f4cba300..c1d56f523 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -163,14 +163,13 @@ class WriteCode(Action): code_filenames = m.get("Task list", []) codes = [] src_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.src_workspace) - old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) - - src_files = src_file_repo.all_files - old_files = old_file_repo.all_files - union_files_list = list(set(src_files) | set(old_files)) if mode == "guide": # 从两个repo中取code,并结合在一起 + src_files = src_file_repo.all_files + old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) + old_files = old_file_repo.all_files + union_files_list = list(set(src_files) | set(old_files)) for filename in union_files_list: if filename == exclude: if filename in old_files: @@ -181,10 +180,11 @@ class WriteCode(Action): else: doc = await src_file_repo.get(filename=filename) # 使用先前生成的代码 if not doc: - if filename in old_files: - doc = await old_file_repo.get(filename=filename) # 使用原始代码 - else: - continue + # if filename in old_files: + # doc = await old_file_repo.get(filename=filename) # 使用原始代码 + # else: + # continue + continue # 跳过 codes.append(f"----- {filename}\n```{doc.content}```") else: diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guide_an.py index 9f1e16405..579008732 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guide_an.py @@ -18,8 +18,7 @@ from metagpt.schema import Document GUIDELINE = ActionNode( key="Code Guideline", expected_type=list[str], - instruction="You are a professional software engineer, and your main task is to " - "proposing incremental development plans and code guidance", + instruction="crafting comprehensive incremental development plans and providing detailed code guidance", example=[ "`calculator.py` should be extended to include methods for subtraction, multiplication, and division. Error handling should be implemented for division to prevent division by zero.", "New endpoints for subtraction, multiplication, and division should be added to `main.py`.", @@ -83,7 +82,7 @@ if __name__ == '__main__': CODE_GUIDE_CONTEXT = """ ### NOTICE -Role: You are a professional software engineer, and your main task is to write code Guideline and code craft with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". +Role: You are a professional software engineer, and your main task is to craft comprehensive incremental development plans and provide detailed code guidance with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". 1. Determine the scope of responsibilities of each file and what classes and methods need to be implemented. 2. Import all referenced classes. 3. Implement all methods. @@ -159,8 +158,7 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc 5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 6. Before using a external variable/module, make sure you import it first. 7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. -8. When Task List code files do not include Legacy Code files, you need to seamlessly merge and adjust Legacy Code files that do not appear in the Task List into the code files of the task list being written based on the guideline. -9. Attention: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions. +8. Attention: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions. """ CODE_GUIDE_CONTEXT_EXAMPLE = """ From db22ed214fc84c9173d16a1626fff993d0c51ba6 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 29 Dec 2023 13:55:30 +0800 Subject: [PATCH 246/637] Modify prompt in write_code_guide_an.py and write_code.py --- metagpt/actions/write_code.py | 8 ++++---- metagpt/actions/write_code_guide_an.py | 15 ++++++++------- metagpt/roles/engineer.py | 3 +-- 3 files changed, 13 insertions(+), 13 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index c1d56f523..77976a696 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -123,7 +123,7 @@ class WriteCode(Action): else: code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename) - if guideline: # guide write code 也有两种方式,进行尝试 + if guideline: prompt = WRITE_CODE_INCREMENT_TEMPLATE.format( guideline=guideline, design=coding_context.design_doc.content if coding_context.design_doc else "", @@ -165,17 +165,17 @@ class WriteCode(Action): src_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.src_workspace) if mode == "guide": - # 从两个repo中取code,并结合在一起 src_files = src_file_repo.all_files old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) old_files = old_file_repo.all_files union_files_list = list(set(src_files) | set(old_files)) for filename in union_files_list: if filename == exclude: - if filename in old_files: + if filename in old_files and filename != "main.py": doc = await old_file_repo.get(filename=filename) # 使用原始代码 else: continue + codes.append(f"----- Legacy {filename}\n```{doc.content}```") else: doc = await src_file_repo.get(filename=filename) # 使用先前生成的代码 @@ -185,7 +185,7 @@ class WriteCode(Action): # else: # continue continue # 跳过 - codes.append(f"----- {filename}\n```{doc.content}```") + codes.append(f"----- {filename}\n```{doc.content}```") else: for filename in code_filenames: diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guide_an.py index 579008732..8bf7da1a6 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guide_an.py @@ -18,10 +18,10 @@ from metagpt.schema import Document GUIDELINE = ActionNode( key="Code Guideline", expected_type=list[str], - instruction="crafting comprehensive incremental development plans and providing detailed code guidance", + instruction="Developing comprehensive and incremental software development plans while providing detailed code guidance.", example=[ - "`calculator.py` should be extended to include methods for subtraction, multiplication, and division. Error handling should be implemented for division to prevent division by zero.", - "New endpoints for subtraction, multiplication, and division should be added to `main.py`.", + "Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Implement robust error handling for the division operation to mitigate potential issues related to division by zero.", + "Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Ensure seamless integration with the overall application architecture and maintain consistency with coding standards.", ], ) @@ -155,10 +155,11 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc 2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets. 3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import. 4. Follow design: YOU MUST FOLLOW "Data structures and interfaces". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design. -5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. -6. Before using a external variable/module, make sure you import it first. -7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. -8. Attention: Make modifications and additions to the legacy code in accordance with the provided guidelines and API. Ensure that the complete code is implemented without any omissions. +5. Follow Guideline: If Legacy Code files contain {filename}, you are required to follow the Guideline to merge the Incremental Change into the Legacy {filename} file when rewriting {filename} file. +6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. +7. Before using a external variable/module, make sure you import it first. +8. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. +9. Attention: Implement the functionality required within the current file's scope, reusing existing code whenever possible. For instance, main.py achieves its purpose by instantiating an already implemented class, rather than manually implementing a class in main.py. """ CODE_GUIDE_CONTEXT_EXAMPLE = """ diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 165ffadc2..2eee4c477 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -80,7 +80,6 @@ class Engineer(Role): ) n_borg: int = 1 use_code_review: bool = False - use_code_guide: bool = True code_todos: list = [] summarize_todos = [] @@ -138,7 +137,7 @@ class Engineer(Role): return None async def _act_write_code(self): - if self.use_code_guide: + if CONFIG.inc: code_guideline = await self._write_code_guideline() changed_files = await self._act_sp_with_cr(review=self.use_code_review, guideline=code_guideline) else: From 6743a4f3b1231ac4b1a8e7ebe09805a8becca791 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Sun, 31 Dec 2023 12:56:15 +0800 Subject: [PATCH 247/637] update prompt --- metagpt/actions/design_api.py | 4 +- metagpt/actions/design_api_an.py | 89 +++++++++++++++- metagpt/actions/project_management.py | 4 +- metagpt/actions/project_management_an.py | 100 +++++++++++++++++- metagpt/actions/write_code.py | 6 +- metagpt/actions/write_code_guide_an.py | 30 +++--- metagpt/actions/write_prd.py | 12 ++- metagpt/actions/write_prd_an.py | 123 ++++++++++++++++++++++- metagpt/roles/engineer.py | 11 +- metagpt/utils/mermaid.py | 38 +++++++ 10 files changed, 381 insertions(+), 36 deletions(-) diff --git a/metagpt/actions/design_api.py b/metagpt/actions/design_api.py index 055365421..0cc3395ed 100644 --- a/metagpt/actions/design_api.py +++ b/metagpt/actions/design_api.py @@ -16,7 +16,7 @@ from typing import Optional from pydantic import Field from metagpt.actions import Action, ActionOutput -from metagpt.actions.design_api_an import DESIGN_API_NODE +from metagpt.actions.design_api_an import DESIGN_API_NODE, REFINE_DESIGN_NODES from metagpt.config import CONFIG from metagpt.const import ( DATA_API_DESIGN_FILE_REPO, @@ -87,7 +87,7 @@ class WriteDesign(Action): async def _merge(self, prd_doc, system_design_doc, schema=CONFIG.prompt_schema): context = NEW_REQ_TEMPLATE.format(old_design=system_design_doc.content, context=prd_doc.content) - node = await DESIGN_API_NODE.fill(context=context, llm=self.llm, schema=schema) + node = await REFINE_DESIGN_NODES.fill(context=context, llm=self.llm, schema=schema) system_design_doc.content = node.instruct_content.json(ensure_ascii=False) return system_design_doc diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index 7d6802381..740348481 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -9,7 +9,7 @@ from typing import List from metagpt.actions.action_node import ActionNode from metagpt.logs import logger -from metagpt.utils.mermaid import MMC1, MMC2 +from metagpt.utils.mermaid import MMC1, MMC1_INC_AND_REFINE, MMC2 IMPLEMENTATION_APPROACH = ActionNode( key="Implementation approach", @@ -18,6 +18,24 @@ IMPLEMENTATION_APPROACH = ActionNode( example="We will ...", ) +INC_IMPLEMENTATION_APPROACH = ActionNode( + key="Incremental Implementation approach", + expected_type=str, + instruction="Analyze the challenging aspects of the requirements and select a suitable open-source framework. " + "Outline the incremental steps involved in the implementation process with a list of detailed strategies.", + example="we will ...", +) + +REFINE_IMPLEMENTATION_APPROACH = ActionNode( + key="Implementation Approach", + expected_type=str, + instruction="Update and extend the original implementation approach to reflect the evolving challenges and requirements " + "due to incremental development. Provide detailed strategies for incremental steps in the implementation process." + "etain any content unrelated to incremental development for coherence and clarity.", + example="We will refine ...", +) + + PROJECT_NAME = ActionNode( key="Project name", expected_type=str, instruction="The project name with underline", example="game_2048" ) @@ -29,6 +47,14 @@ FILE_LIST = ActionNode( example=["main.py", "game.py"], ) +REFINE_FILE_LIST = ActionNode( + key="File List", + expected_type=List[str], + instruction="Update and expand the original file list, including only relative paths. " + "Ensure that the refined file list reflects the evolving structure of the project due to incremental development.", + example=["main.py", "game.py", "utils.py", "new_feature.py"], +) + DATA_STRUCTURES_AND_INTERFACES = ActionNode( key="Data structures and interfaces", expected_type=str, @@ -38,6 +64,27 @@ DATA_STRUCTURES_AND_INTERFACES = ActionNode( example=MMC1, ) +INC_DATA_STRUCTURES_AND_INTERFACES = ActionNode( + key="Incremental Data structures and interfaces", + expected_type=str, + instruction="Extend the existing mermaid classDiagram code syntax to incorporate new classes, " + "methods (including __init__), and functions with precise type annotations. Clearly delineate additional " + "relationships between classes, maintaining adherence to PEP8 standards. Enhance the level of detail in data " + "structures, ensuring a comprehensive API design that seamlessly integrates with the existing structure.", + example=MMC1_INC_AND_REFINE, +) + +REFINE_DATA_STRUCTURES_AND_INTERFACES = ActionNode( + key="Data Structures and Interfaces", + expected_type=str, + instruction="Update and extend the existing mermaid classDiagram code syntax to incorporate new classes, " + "methods (including __init__), and functions with precise type annotations. Delineate additional " + "relationships between classes, ensuring clarity and adherence to PEP8 standards. Further enhance the " + "detail in data structures for a comprehensive API design that seamlessly integrates with the evolving structure." + "Retain any content unrelated to incremental development for coherence and clarity.", + example=MMC1_INC_AND_REFINE, +) + PROGRAM_CALL_FLOW = ActionNode( key="Program call flow", expected_type=str, @@ -53,6 +100,31 @@ ANYTHING_UNCLEAR = ActionNode( example="Clarification needed on third-party API integration, ...", ) +INC_DESIGN_CONTEXT = """ +### Legacy Content +{old_design} + +### New Requirements +{requirements} + +### PRD Increment Content +{prd_increment} +""" + +REFINE_DESIGN_CONTEXT = """ +Role: You are a professional Architect tasked with overseeing incremental development. +Based on new requirements, review and refine the system design. Integrate existing architecture with incremental design changes, ensuring the refined design encompasses all architectural elements, enhancements, and adjustments. Retain content unrelated to incremental development needs for coherence and clarity. + +### New Requirements +{requirements} + +### Legacy Content +{old_design} + +### Design Increment Content +{design_increment} +""" + NODES = [ IMPLEMENTATION_APPROACH, # PROJECT_NAME, @@ -62,11 +134,24 @@ NODES = [ ANYTHING_UNCLEAR, ] +INC_NODES = [INC_IMPLEMENTATION_APPROACH, INC_DATA_STRUCTURES_AND_INTERFACES] + +REFINE_NODES = [ + REFINE_IMPLEMENTATION_APPROACH, + # PROJECT_NAME, + REFINE_FILE_LIST, + REFINE_DATA_STRUCTURES_AND_INTERFACES, + PROGRAM_CALL_FLOW, + ANYTHING_UNCLEAR, +] + DESIGN_API_NODE = ActionNode.from_children("DesignAPI", NODES) +INC_DESIGN_NODES = ActionNode.from_children("Incremental Design API", INC_NODES) +REFINE_DESIGN_NODES = ActionNode.from_children("Refine Design API", REFINE_NODES) def main(): - prompt = DESIGN_API_NODE.compile(context="") + prompt = REFINE_DESIGN_NODES.compile(context="12313") logger.info(prompt) diff --git a/metagpt/actions/project_management.py b/metagpt/actions/project_management.py index 095881e60..5d2791ea8 100644 --- a/metagpt/actions/project_management.py +++ b/metagpt/actions/project_management.py @@ -17,7 +17,7 @@ from pydantic import Field from metagpt.actions import ActionOutput from metagpt.actions.action import Action -from metagpt.actions.project_management_an import PM_NODE +from metagpt.actions.project_management_an import PM_NODE, REFINE_PM_NODES from metagpt.config import CONFIG from metagpt.const import ( PACKAGE_REQUIREMENTS_FILENAME, @@ -101,7 +101,7 @@ class WriteTasks(Action): async def _merge(self, system_design_doc, task_doc, schema=CONFIG.prompt_schema) -> Document: context = NEW_REQ_TEMPLATE.format(context=system_design_doc.content, old_tasks=task_doc.content) - node = await PM_NODE.fill(context, self.llm, schema) + node = await REFINE_PM_NODES.fill(context, self.llm, schema) task_doc.content = node.instruct_content.json(ensure_ascii=False) return task_doc diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index 215a67202..a13ce2dcd 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -35,6 +35,31 @@ LOGIC_ANALYSIS = ActionNode( ], ) +INC_LOGIC_ANALYSIS = ActionNode( + key="Increment Logic Analysis", + expected_type=List[List[str]], + instruction="Provide a list of files with the classes/methods/functions to be implemented or modified incrementally. Include thorough dependency analysis, consider potential impacts on existing code, and document necessary imports.", + example=[ + ["new_feature.py", "Introduces NewFeature class and related functions"], + ["utils.py", "Modifies existing utility functions to support incremental changes"], + ], +) + +REFINE_LOGIC_ANALYSIS = ActionNode( + key="Logic Analysis", + expected_type=List[List[str]], + instruction="Review and refine the logic analysis by merging the Legacy Content and Incremental Content. " + "Provide a comprehensive list of files with classes/methods/functions to be implemented or modified incrementally. " + "Include thorough dependency analysis, consider potential impacts on existing code, and document necessary imports." + "Retain any content unrelated to incremental development for coherence and clarity.", + example=[ + ["game.py", "Contains Game class and ... functions"], + ["main.py", "Contains main function, from game import Game"], + ["new_feature.py", "Introduces NewFeature class and related functions"], + ["utils.py", "Modifies existing utility functions to support incremental changes"], + ], +) + TASK_LIST = ActionNode( key="Task list", expected_type=List[str], @@ -42,6 +67,23 @@ TASK_LIST = ActionNode( example=["game.py", "main.py"], ) +INC_TASK_LIST = ActionNode( + key="Incremental Task list", + expected_type=List[str], + instruction="Break down the incremental development tasks into a prioritized list of filenames. " + "Organize the tasks based on dependency order, ensuring a systematic and efficient implementation.", + example=["new_feature.py", "utils.py", "main.py"], +) + +REFINE_TASK_LIST = ActionNode( + key="Task list", + expected_type=List[str], + instruction="Review and refine the combined task list after the merger of Legacy Content and Incremental Content. " + "Ensure that tasks are organized in a logical and prioritized order, considering dependencies for a streamlined and" + " efficient development process.", + example=["game.py", "utils.py", "new_feature.py", "main.py"], +) + FULL_API_SPEC = ActionNode( key="Full API spec", expected_type=str, @@ -54,9 +96,27 @@ SHARED_KNOWLEDGE = ActionNode( key="Shared Knowledge", expected_type=str, instruction="Detail any shared knowledge, like common utility functions or configuration variables.", - example="'game.py' contains functions shared across the project.", + example="`game.py` contains functions shared across the project.", ) +INC_SHARED_KNOWLEDGE = ActionNode( + key="Increment Shared Knowledge", + expected_type=str, + instruction="Document any new shared knowledge generated during incremental development. This includes common " + "utility functions, configuration variables, or any information vital for team collaboration.", + example="`new_module.py` introduces shared utility functions for improved code reusability.", +) + +REFINE_SHARED_KNOWLEDGE = ActionNode( + key="Shared Knowledge", + expected_type=str, + instruction="Update and expand shared knowledge to reflect any new elements introduced during incremental development. " + "This includes common utility functions, configuration variables, or any information vital for team collaboration." + "Retain any content unrelated to incremental development for coherence and clarity.", + example="`new_module.py` enhances shared utility functions for improved code reusability and collaboration.", +) + + ANYTHING_UNCLEAR_PM = ActionNode( key="Anything UNCLEAR", expected_type=str, @@ -64,6 +124,31 @@ ANYTHING_UNCLEAR_PM = ActionNode( example="Clarification needed on how to start and initialize third-party libraries.", ) +INC_PM_CONTEXT = """ +### Legacy Content +{old_tasks} + +### New Requirements +{requirements} + +### Design Increment Content +{design_increment} +""" + +REFINE_PM_CONTEXT = """ +Role: You are a professional Project Manager tasked with overseeing incremental development. +Based on New Requirements, refine the project context to account for incremental development. Ensure the context offers a comprehensive overview of the project's evolving scope, covering both legacy content and incremental content. Retain any content unrelated to incremental development. + +### New Requirements +{requirements} + +### Legacy Content +{old_tasks} + +### Increment Content +{tasks_increment} +""" + NODES = [ REQUIRED_PYTHON_PACKAGES, REQUIRED_OTHER_LANGUAGE_PACKAGES, @@ -74,8 +159,21 @@ NODES = [ ANYTHING_UNCLEAR_PM, ] +INC_NODES = [INC_LOGIC_ANALYSIS, INC_TASK_LIST, INC_SHARED_KNOWLEDGE] + +REFINE_NODES = [ + REQUIRED_PYTHON_PACKAGES, + REQUIRED_OTHER_LANGUAGE_PACKAGES, + REFINE_LOGIC_ANALYSIS, + REFINE_TASK_LIST, + FULL_API_SPEC, + REFINE_SHARED_KNOWLEDGE, + ANYTHING_UNCLEAR_PM, +] PM_NODE = ActionNode.from_children("PM_NODE", NODES) +INC_PM_NODES = ActionNode.from_children("Incremental_PM_NODES", INC_NODES) +REFINE_PM_NODES = ActionNode.from_children("Refine_PM_NODES", REFINE_NODES) def main(): diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 77976a696..ce63968b1 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -27,6 +27,7 @@ from metagpt.const import ( BUGFIX_FILENAME, CODE_SUMMARIES_FILE_REPO, DOCS_FILE_REPO, + REQUIREMENT_FILENAME, TASK_FILE_REPO, TEST_OUTPUTS_FILE_REPO, ) @@ -115,6 +116,8 @@ class WriteCode(Action): test_detail = RunCodeResult.loads(test_doc.content) logs = test_detail.stderr + docs_file_repo = CONFIG.git_repo.new_file_repository(relative_path=DOCS_FILE_REPO) + requirement_doc = await docs_file_repo.get(filename=REQUIREMENT_FILENAME) guideline = kwargs.get("guideline", "") if bug_feedback: code_context = coding_context.code_doc.content @@ -125,6 +128,7 @@ class WriteCode(Action): if guideline: prompt = WRITE_CODE_INCREMENT_TEMPLATE.format( + requirement=requirement_doc.content if requirement_doc else "", guideline=guideline, design=coding_context.design_doc.content if coding_context.design_doc else "", tasks=coding_context.task_doc.content if coding_context.task_doc else "", @@ -175,7 +179,7 @@ class WriteCode(Action): doc = await old_file_repo.get(filename=filename) # 使用原始代码 else: continue - codes.append(f"----- Legacy {filename}\n```{doc.content}```") + codes.insert(0, f"-----Now, {filename} need to be rewritten\n```{doc.content}```\n=====") else: doc = await src_file_repo.get(filename=filename) # 使用先前生成的代码 diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guide_an.py index 8bf7da1a6..5b0a438c3 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guide_an.py @@ -28,8 +28,8 @@ GUIDELINE = ActionNode( INCREMENTAL_CHANGE = ActionNode( key="Incremental Change", expected_type=str, - instruction="Write Incremental Change by making a code draft that how to implement incremental development based on the context and Code Guideline.", - example="""1. Extend `Calculator` class in `calculator.py` with new methods for subtraction, multiplication, and division. + instruction="Write Incremental Change by making a code draft that how to implement incremental development including detailed steps based on the context.", + example="""- calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Implement robust error handling for the division operation to mitigate potential issues related to division by zero. ```python ## calculator.py class Calculator: @@ -43,7 +43,8 @@ class Calculator: raise ValueError('Cannot divide by zero') return num1 / num2 ``` -2. Implement new endpoints in `main.py` for the subtraction, multiplication, and division methods. + +- main.py: Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Ensure seamless integration with the overall application architecture and maintain consistency with coding standards. ```python ## main.py from flask import Flask, request, jsonify @@ -95,9 +96,6 @@ Role: You are a professional software engineer, and your main task is to craft c ### Requirement {requirement} -### Prd -{prd} - ### Design {design} @@ -110,12 +108,13 @@ Role: You are a professional software engineer, and your main task is to craft c WRITE_CODE_INCREMENT_TEMPLATE = """ NOTICE -Role: You are a professional engineer; The main goal is to complete incremental development by combining Legacy Code and Guideline to rewrite the complete code. -Language: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese. -ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example". +Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and Incremental Change, ensuring the integration of new features. # Context -## Guideline +## New Requirement +{requirement} + +## Incremental Change {guideline} ## Design @@ -148,18 +147,17 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc ... ``` -# Instruction: Based on the context, follow "Format example", write code. - -## Rewrite Complete Code: Only Write one file {filename}, Write code using triple quotes, based on the following attentions and context. +# Instruction: Based on the context, follow "Format example", write or rewrite code. +## Write/Rewrite Code: Only write one file {filename}, write or rewrite complete code using triple quotes based on the following attentions and context. +### Important Attention: If Legacy Code files contain "{filename} to be rewritten", you are required to merge the Incremental Change into the {filename} file and retain any content unrelated to incremental development to maintain clarity and coherence, when rewriting "{filename} to be rewritten". 1. Only One file: do your best to implement THIS ONLY ONE FILE. 2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets. 3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import. 4. Follow design: YOU MUST FOLLOW "Data structures and interfaces". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design. -5. Follow Guideline: If Legacy Code files contain {filename}, you are required to follow the Guideline to merge the Incremental Change into the Legacy {filename} file when rewriting {filename} file. +5. Merge Incremental Change: If there is any Incremental Change, you must merge it into the code file. 6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 7. Before using a external variable/module, make sure you import it first. 8. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. -9. Attention: Implement the functionality required within the current file's scope, reusing existing code whenever possible. For instance, main.py achieves its purpose by instantiating an already implemented class, rather than manually implementing a class in main.py. """ CODE_GUIDE_CONTEXT_EXAMPLE = """ @@ -194,7 +192,7 @@ class Calculator: return num1 + num2 """ -GUIDE_NODES = [GUIDELINE, INCREMENTAL_CHANGE] +GUIDE_NODES = [INCREMENTAL_CHANGE] WRITE_CODE_GUIDE_NODE = ActionNode.from_children("WriteCodeGuide", GUIDE_NODES) diff --git a/metagpt/actions/write_prd.py b/metagpt/actions/write_prd.py index 1223e5486..dc1af0735 100644 --- a/metagpt/actions/write_prd.py +++ b/metagpt/actions/write_prd.py @@ -23,6 +23,8 @@ from metagpt.actions import Action, ActionOutput from metagpt.actions.action_node import ActionNode from metagpt.actions.fix_bug import FixBug from metagpt.actions.write_prd_an import ( + REFINE_PRD_NODE, + REFINE_PRD_SIMPLE_CONTEXT, WP_IS_RELATIVE_NODE, WP_ISSUE_TYPE_NODE, WRITE_PRD_NODE, @@ -135,8 +137,14 @@ class WritePRD(Action): async def _merge(self, new_requirement_doc, prd_doc, schema=CONFIG.prompt_schema) -> Document: if not CONFIG.project_name: CONFIG.project_name = Path(CONFIG.project_path).name - prompt = NEW_REQ_TEMPLATE.format(requirements=new_requirement_doc.content, old_prd=prd_doc.content) - node = await WRITE_PRD_NODE.fill(context=prompt, llm=self.llm, schema=schema) + + project_name = CONFIG.project_name if CONFIG.project_name else "" + prompt = REFINE_PRD_SIMPLE_CONTEXT.format( + requirements=new_requirement_doc.content, + old_prd=prd_doc.content, + project_name=project_name, + ) + node = await REFINE_PRD_NODE.fill(context=prompt, llm=self.llm, schema=schema) prd_doc.content = node.instruct_content.json(ensure_ascii=False) await self._rename_workspace(node) return prd_doc diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index d58d72f64..5ba694b0f 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -31,6 +31,14 @@ ORIGINAL_REQUIREMENTS = ActionNode( example="Create a 2048 game", ) +REFINE_REQUIREMENTS = ActionNode( + key="Original Requirements", + expected_type=str, + instruction="Update and expand the original user's requirements to reflect the evolving needs of the project." + "Retain any content unrelated to incremental development", + example="Create a 2048 game with a new feature that ...", +) + PROJECT_NAME = ActionNode( key="Project Name", expected_type=str, @@ -38,6 +46,13 @@ PROJECT_NAME = ActionNode( example="game_2048", ) +REFINE_PROJECT_NAME = ActionNode( + key="Project Name", + expected_type=str, + instruction="Update the project name based on the context.", + example="game_2048_new", +) + PRODUCT_GOALS = ActionNode( key="Product Goals", expected_type=List[str], @@ -45,6 +60,19 @@ PRODUCT_GOALS = ActionNode( example=["Create an engaging user experience", "Improve accessibility, be responsive", "More beautiful UI"], ) +REFINE_PRODUCT_GOALS = ActionNode( + key="Product Goals", + expected_type=List[str], + instruction="Update and expand the original product goals to reflect the evolving needs due to incremental " + "development.Ensure that the refined goals align with the current project direction and contribute to its success." + "Retain any content unrelated to incremental development", + example=[ + "Enhance user engagement through new features", + "Optimize performance for scalability", + "Integrate innovative UI enhancements", + ], +) + USER_STORIES = ActionNode( key="User Stories", expected_type=List[str], @@ -58,6 +86,21 @@ USER_STORIES = ActionNode( ], ) +REFINE_USER_STORIES = ActionNode( + key="User Stories", + expected_type=List[str], + instruction="Update and expand the original scenario-based user stories to reflect the evolving needs due to " + "incremental development, no less than 7. Ensure that the refined user stories capture incremental features and " + "improvements. Retain any content unrelated to incremental development", + example=[ + "As a player, I want to choose difficulty levels to challenge my skills", + "As a player, I want a visually appealing score display after each game for a better gaming experience", + "As a player, I want a convenient restart button displayed when I lose to quickly start a new game", + "As a player, I want an enhanced and aesthetically pleasing UI to elevate the overall gaming experience", + "As a player, I want the ability to play the game seamlessly on my mobile phone for on-the-go entertainment", + ], +) + COMPETITIVE_ANALYSIS = ActionNode( key="Competitive Analysis", expected_type=List[str], @@ -104,6 +147,14 @@ REQUIREMENT_POOL = ActionNode( example=[["P0", "The main code ..."], ["P0", "The game algorithm ..."]], ) +REFINE_REQUIREMENT_POOL = ActionNode( + key="Requirement Pool", + expected_type=List[List[str]], + instruction="List no less than 7 requirements with their priority (P0, P1, P2). " + "Cover both legacy content and incremental content. Retain any content unrelated to incremental development", + example=[["P0", "The main code ..."], ["P0", "The game algorithm ..."]], +) + UI_DESIGN_DRAFT = ActionNode( key="UI Design draft", expected_type=str, @@ -121,8 +172,10 @@ ANYTHING_UNCLEAR = ActionNode( ISSUE_TYPE = ActionNode( key="issue_type", expected_type=str, - instruction="Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement", - example="BUG", + instruction="Answer BUG/REFINE/OVERHAUL. If it is a bugfix, answer BUG;" + "if it is a minor improvement, answer REFINE;" + "if it is a major overhaul, answer OVERHAUL that most likely not answer in most cases.", + example="REFINE", ) IS_RELATIVE = ActionNode( @@ -137,6 +190,50 @@ REASON = ActionNode( ) +REFINE_PRD_CONTEXT = """ +Role: You are a professional Product Manager tasked with overseeing incremental development. +Based on New Requirements, output a New PRD that seamlessly integrates both the Legacy Content and the Incremental Content. Ensure the resulting document captures the complete scope of features, enhancements, and retain content unrelated to incremental development needs for coherence and clarity. + +### New Project Name +{project_name} + +### New Requirements +{requirements} + +### Legacy Content +{old_prd} + +### PRD Incremental Content +{prd_increment} + +### Search Information +- +""" + +REFINE_PRD_SIMPLE_CONTEXT = """ +You are a professional Product Manager tasked with overseeing incremental development. + +### New Project Name +{project_name} + +### New Requirements +{requirements} + +### Legacy Content +{old_prd} + +### Search Information +- +""" + +INCREMENTAL_DEVELOPMENT_ANALYSIS = ActionNode( + key="Requirement Analysis", + expected_type=List[str], + instruction="Propose the comprehensive incremental development requirement analysis on new features and enhanced features for New Requirements.", + example=["Require add/update/modify ..."], +) + + NODES = [ LANGUAGE, PROGRAMMING_LANGUAGE, @@ -152,13 +249,33 @@ NODES = [ ANYTHING_UNCLEAR, ] +REFINE_NODES = [ + LANGUAGE, + PROGRAMMING_LANGUAGE, + REFINE_REQUIREMENTS, + REFINE_PROJECT_NAME, + REFINE_PRODUCT_GOALS, + REFINE_USER_STORIES, + COMPETITIVE_ANALYSIS, + COMPETITIVE_QUADRANT_CHART, + INCREMENTAL_DEVELOPMENT_ANALYSIS, + REFINE_REQUIREMENT_POOL, + UI_DESIGN_DRAFT, + ANYTHING_UNCLEAR, +] + +INCREMENT_PRD_NODES = [INCREMENTAL_DEVELOPMENT_ANALYSIS, REQUIREMENT_POOL] + WRITE_PRD_NODE = ActionNode.from_children("WritePRD", NODES) +REFINE_PRD_NODE = ActionNode.from_children("RefinePRD", REFINE_NODES) +INCREMENT_NODE = ActionNode.from_children("IncrementPRD", INCREMENT_PRD_NODES) WP_ISSUE_TYPE_NODE = ActionNode.from_children("WP_ISSUE_TYPE", [ISSUE_TYPE, REASON]) WP_IS_RELATIVE_NODE = ActionNode.from_children("WP_IS_RELATIVE", [IS_RELATIVE, REASON]) def main(): - prompt = WRITE_PRD_NODE.compile(context="") + # prompt = WRITE_PRD_NODE.compile(context="") + prompt = INCREMENT_NODE.compile(context=REFINE_PRD_CONTEXT) logger.info(prompt) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 2eee4c477..4aa8f209d 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -33,7 +33,6 @@ from metagpt.config import CONFIG from metagpt.const import ( CODE_SUMMARIES_FILE_REPO, CODE_SUMMARIES_PDF_FILE_REPO, - PRDS_FILE_REPO, SYSTEM_DESIGN_FILE_REPO, TASK_FILE_REPO, ) @@ -313,20 +312,18 @@ class Engineer(Role): logger.info("Writing code guideline..") requirement = str(self._rc.memory.get_by_role("Human")[0]) - prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) + # prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) design_file_repo = CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO) task_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) - prd = await prd_file_repo.get_all() - prd = "\n".join([doc.content for doc in prd]) + # prd = await prd_file_repo.get_all() + # prd = "\n".join([doc.content for doc in prd]) design = await design_file_repo.get_all() design = "\n".join([doc.content for doc in design]) tasks = await task_file_repo.get_all() tasks = "\n".join([doc.content for doc in tasks]) old_codes = await self.get_old_codes() - context = CODE_GUIDE_CONTEXT.format( - requirement=requirement, prd=prd, tasks=tasks, design=design, code=old_codes - ) + context = CODE_GUIDE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) node = await WriteCodeGuide().run(context=context) guideline = node.instruct_content.json(ensure_ascii=False) return guideline diff --git a/metagpt/utils/mermaid.py b/metagpt/utils/mermaid.py index eb85a3f90..636ec4598 100644 --- a/metagpt/utils/mermaid.py +++ b/metagpt/utils/mermaid.py @@ -120,6 +120,44 @@ MMC1 = """classDiagram SearchEngine --> Summary Index --> KnowledgeBase""" +MMC1_INC_AND_REFINE = """classDiagram + class Main { + -SearchEngine search_engine + +main() str + +newMethod() str # Incremental change + } + class SearchEngine { + -Index index + -Ranking ranking + -Summary summary + +search(query: str) str + +newMethod() str # Incremental change + } + class Index { + -KnowledgeBase knowledge_base + +create_index(data: dict) + +query_index(query: str) list + +newMethod() list # Incremental change + } + class Ranking { + +rank_results(results: list) list + +newMethod() list # Incremental change + } + class Summary { + +summarize_results(results: list) str + +newMethod() str # Incremental change + } + class KnowledgeBase { + +update(data: dict) + +fetch_data(query: str) dict + +newMethod() # Incremental change + } + Main --> SearchEngine + SearchEngine --> Index + SearchEngine --> Ranking + SearchEngine --> Summary + Index --> KnowledgeBase""" + MMC2 = """sequenceDiagram participant M as Main participant SE as SearchEngine From 6d9dfa73aa155f2b18a67a66e8d2fc1036081df4 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 2 Jan 2024 18:24:02 +0800 Subject: [PATCH 248/637] Fallback to a version that only uses "Refine" and update the prompt for ActionNode --- metagpt/actions/design_api_an.py | 19 ++++++++++++----- metagpt/actions/project_management_an.py | 7 ++++--- metagpt/actions/write_code_guide_an.py | 2 +- metagpt/actions/write_prd_an.py | 6 ++---- metagpt/utils/mermaid.py | 26 ++++++++++++++++++++++++ 5 files changed, 47 insertions(+), 13 deletions(-) diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index 740348481..5f17d4656 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -9,7 +9,7 @@ from typing import List from metagpt.actions.action_node import ActionNode from metagpt.logs import logger -from metagpt.utils.mermaid import MMC1, MMC1_INC_AND_REFINE, MMC2 +from metagpt.utils.mermaid import MMC1, MMC1_INC_AND_REFINE, MMC2, MMC2_INC IMPLEMENTATION_APPROACH = ActionNode( key="Implementation approach", @@ -35,7 +35,6 @@ REFINE_IMPLEMENTATION_APPROACH = ActionNode( example="We will refine ...", ) - PROJECT_NAME = ActionNode( key="Project name", expected_type=str, instruction="The project name with underline", example="game_2048" ) @@ -51,7 +50,8 @@ REFINE_FILE_LIST = ActionNode( key="File List", expected_type=List[str], instruction="Update and expand the original file list, including only relative paths. " - "Ensure that the refined file list reflects the evolving structure of the project due to incremental development.", + "Ensure that the refined file list reflects the evolving structure of the project due to incremental development." + "Only output filename!Do not include comments in the list.", example=["main.py", "game.py", "utils.py", "new_feature.py"], ) @@ -93,6 +93,15 @@ PROGRAM_CALL_FLOW = ActionNode( example=MMC2, ) +REFINE_PROGRAM_CALL_FLOW = ActionNode( + key="Program call flow", + expected_type=str, + instruction="Extend the existing sequenceDiagram code syntax with detailed information, accurately covering the" + "CRUD and initialization of each object. Ensure correct syntax usage and reflect the incremental changes introduced" + "in the classes and API defined above.Retain content unrelated to incremental development for coherence and clarity", + example=MMC2_INC, +) + ANYTHING_UNCLEAR = ActionNode( key="Anything UNCLEAR", expected_type=str, @@ -134,14 +143,14 @@ NODES = [ ANYTHING_UNCLEAR, ] -INC_NODES = [INC_IMPLEMENTATION_APPROACH, INC_DATA_STRUCTURES_AND_INTERFACES] +INC_NODES = [INC_IMPLEMENTATION_APPROACH, INC_DATA_STRUCTURES_AND_INTERFACES, REFINE_PROGRAM_CALL_FLOW] REFINE_NODES = [ REFINE_IMPLEMENTATION_APPROACH, # PROJECT_NAME, REFINE_FILE_LIST, REFINE_DATA_STRUCTURES_AND_INTERFACES, - PROGRAM_CALL_FLOW, + REFINE_PROGRAM_CALL_FLOW, ANYTHING_UNCLEAR, ] diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index a13ce2dcd..27d238e9f 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -70,8 +70,9 @@ TASK_LIST = ActionNode( INC_TASK_LIST = ActionNode( key="Incremental Task list", expected_type=List[str], - instruction="Break down the incremental development tasks into a prioritized list of filenames. " - "Organize the tasks based on dependency order, ensuring a systematic and efficient implementation.", + instruction="Break down the incremental development tasks into a prioritized list of filenames." + "Organize the tasks based on dependency order, ensuring a systematic and efficient implementation." + "Only output filename! Do not include comments in the list ", example=["new_feature.py", "utils.py", "main.py"], ) @@ -80,7 +81,7 @@ REFINE_TASK_LIST = ActionNode( expected_type=List[str], instruction="Review and refine the combined task list after the merger of Legacy Content and Incremental Content. " "Ensure that tasks are organized in a logical and prioritized order, considering dependencies for a streamlined and" - " efficient development process.", + " efficient development process. Only output filename! Do not include comments in the list", example=["game.py", "utils.py", "new_feature.py", "main.py"], ) diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guide_an.py index 5b0a438c3..b21e66098 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guide_an.py @@ -149,7 +149,6 @@ Role: You are a professional engineer; The main goal is to complete incremental # Instruction: Based on the context, follow "Format example", write or rewrite code. ## Write/Rewrite Code: Only write one file {filename}, write or rewrite complete code using triple quotes based on the following attentions and context. -### Important Attention: If Legacy Code files contain "{filename} to be rewritten", you are required to merge the Incremental Change into the {filename} file and retain any content unrelated to incremental development to maintain clarity and coherence, when rewriting "{filename} to be rewritten". 1. Only One file: do your best to implement THIS ONLY ONE FILE. 2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets. 3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import. @@ -158,6 +157,7 @@ Role: You are a professional engineer; The main goal is to complete incremental 6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 7. Before using a external variable/module, make sure you import it first. 8. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. +9. Attention: If Legacy Code files contain "{filename} to be rewritten", you are required to merge the Incremental Change into the {filename} file when rewriting "{filename} to be rewritten". """ CODE_GUIDE_CONTEXT_EXAMPLE = """ diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index 5ba694b0f..79046bb7d 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -172,10 +172,8 @@ ANYTHING_UNCLEAR = ActionNode( ISSUE_TYPE = ActionNode( key="issue_type", expected_type=str, - instruction="Answer BUG/REFINE/OVERHAUL. If it is a bugfix, answer BUG;" - "if it is a minor improvement, answer REFINE;" - "if it is a major overhaul, answer OVERHAUL that most likely not answer in most cases.", - example="REFINE", + instruction="Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement", + example="BUG", ) IS_RELATIVE = ActionNode( diff --git a/metagpt/utils/mermaid.py b/metagpt/utils/mermaid.py index 636ec4598..d1cd1b328 100644 --- a/metagpt/utils/mermaid.py +++ b/metagpt/utils/mermaid.py @@ -176,6 +176,32 @@ MMC2 = """sequenceDiagram S-->>SE: return summary SE-->>M: return summary""" +MMC2_INC = """sequenceDiagram + participant M as Main + participant SE as SearchEngine + participant I as Index + participant R as Ranking + participant S as Summary + participant KB as KnowledgeBase + M->>SE: search(query) + SE->>I: query_index(query) + I->>KB: fetch_data(query) + KB-->>I: return data + I-->>SE: return results + SE->>R: rank_results(results) + R-->>SE: return ranked_results + SE->>S: summarize_results(ranked_results) + S-->>SE: return summary + SE-->>M: return summary + M->>SE: newMethod() # Incremental change + SE->>I: newMethod() # Incremental change + I->>KB: newMethod() # Incremental change + KB-->>I: newMethod() # Incremental change + SE->>R: newMethod() # Incremental change + R-->>SE: newMethod() # Incremental change + SE->>S: newMethod() # Incremental change + S-->>SE: newMethod() # Incremental change + SE-->>M: newMethod() # Incremental change""" if __name__ == "__main__": loop = asyncio.new_event_loop() From 3a819ad5762b59c969457bb3881a186e643e2216 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 2 Jan 2024 19:20:46 +0800 Subject: [PATCH 249/637] update write_code.py --- metagpt/actions/write_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index ce63968b1..33ae2e5b3 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -179,7 +179,7 @@ class WriteCode(Action): doc = await old_file_repo.get(filename=filename) # 使用原始代码 else: continue - codes.insert(0, f"-----Now, {filename} need to be rewritten\n```{doc.content}```\n=====") + codes.insert(0, f"-----Now, {filename} to be rewritten\n```{doc.content}```\n=====") else: doc = await src_file_repo.get(filename=filename) # 使用先前生成的代码 From 11faaf0651bf53ae59b19bf6e0c6672eb1f87a92 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 2 Jan 2024 22:57:42 +0800 Subject: [PATCH 250/637] Modify style and format --- metagpt/actions/design_api.py | 8 +- metagpt/actions/design_api_an.py | 69 ++++++++--------- metagpt/actions/project_management.py | 4 +- metagpt/actions/project_management_an.py | 51 +++++++------ metagpt/actions/write_code.py | 6 +- ...guide_an.py => write_code_guideline_an.py} | 43 +++++------ metagpt/actions/write_prd.py | 4 +- metagpt/actions/write_prd_an.py | 75 ++++++++----------- metagpt/roles/engineer.py | 9 ++- metagpt/utils/mermaid.py | 4 +- 10 files changed, 130 insertions(+), 143 deletions(-) rename metagpt/actions/{write_code_guide_an.py => write_code_guideline_an.py} (87%) diff --git a/metagpt/actions/design_api.py b/metagpt/actions/design_api.py index 0cc3395ed..082474098 100644 --- a/metagpt/actions/design_api.py +++ b/metagpt/actions/design_api.py @@ -16,7 +16,7 @@ from typing import Optional from pydantic import Field from metagpt.actions import Action, ActionOutput -from metagpt.actions.design_api_an import DESIGN_API_NODE, REFINE_DESIGN_NODES +from metagpt.actions.design_api_an import DESIGN_API_NODE, REFINED_DESIGN_NODES from metagpt.config import CONFIG from metagpt.const import ( DATA_API_DESIGN_FILE_REPO, @@ -87,7 +87,7 @@ class WriteDesign(Action): async def _merge(self, prd_doc, system_design_doc, schema=CONFIG.prompt_schema): context = NEW_REQ_TEMPLATE.format(old_design=system_design_doc.content, context=prd_doc.content) - node = await REFINE_DESIGN_NODES.fill(context=context, llm=self.llm, schema=schema) + node = await REFINED_DESIGN_NODES.fill(context=context, llm=self.llm, schema=schema) system_design_doc.content = node.instruct_content.json(ensure_ascii=False) return system_design_doc @@ -114,7 +114,7 @@ class WriteDesign(Action): @staticmethod async def _save_data_api_design(design_doc): m = json.loads(design_doc.content) - data_api_design = m.get("Data structures and interfaces") + data_api_design = m.get("Data structures and interfaces") or m.get("Refined Data structures and interfaces") if not data_api_design: return pathname = CONFIG.git_repo.workdir / DATA_API_DESIGN_FILE_REPO / Path(design_doc.filename).with_suffix("") @@ -124,7 +124,7 @@ class WriteDesign(Action): @staticmethod async def _save_seq_flow(design_doc): m = json.loads(design_doc.content) - seq_flow = m.get("Program call flow") + seq_flow = m.get("Program call flow") or m.get("Refined Program call flow") if not seq_flow: return pathname = CONFIG.git_repo.workdir / Path(SEQ_FLOW_FILE_REPO) / Path(design_doc.filename).with_suffix("") diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index 5f17d4656..fcfaff9c3 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -9,7 +9,7 @@ from typing import List from metagpt.actions.action_node import ActionNode from metagpt.logs import logger -from metagpt.utils.mermaid import MMC1, MMC1_INC_AND_REFINE, MMC2, MMC2_INC +from metagpt.utils.mermaid import MMC1, MMC1_REFINE, MMC2, MMC2_REFINE IMPLEMENTATION_APPROACH = ActionNode( key="Implementation approach", @@ -18,7 +18,7 @@ IMPLEMENTATION_APPROACH = ActionNode( example="We will ...", ) -INC_IMPLEMENTATION_APPROACH = ActionNode( +INCREMENTAL_IMPLEMENTATION_APPROACH = ActionNode( key="Incremental Implementation approach", expected_type=str, instruction="Analyze the challenging aspects of the requirements and select a suitable open-source framework. " @@ -26,12 +26,12 @@ INC_IMPLEMENTATION_APPROACH = ActionNode( example="we will ...", ) -REFINE_IMPLEMENTATION_APPROACH = ActionNode( - key="Implementation Approach", +REFINED_IMPLEMENTATION_APPROACH = ActionNode( + key="Refined Implementation Approach", expected_type=str, - instruction="Update and extend the original implementation approach to reflect the evolving challenges and requirements " - "due to incremental development. Provide detailed strategies for incremental steps in the implementation process." - "etain any content unrelated to incremental development for coherence and clarity.", + instruction="Update and extend the original implementation approach to reflect the evolving challenges and " + "requirements due to incremental development. Provide detailed strategies for incremental steps in the " + "implementation process. Retain any content unrelated to incremental development for coherence and clarity.", example="We will refine ...", ) @@ -46,13 +46,13 @@ FILE_LIST = ActionNode( example=["main.py", "game.py"], ) -REFINE_FILE_LIST = ActionNode( - key="File List", +REFINED_FILE_LIST = ActionNode( + key="Refined File List", expected_type=List[str], instruction="Update and expand the original file list, including only relative paths. " "Ensure that the refined file list reflects the evolving structure of the project due to incremental development." - "Only output filename!Do not include comments in the list.", - example=["main.py", "game.py", "utils.py", "new_feature.py"], + "Only output filename! Do not include comments in the list.", + example=["main.py", "game.py", "new_feature.py"], ) DATA_STRUCTURES_AND_INTERFACES = ActionNode( @@ -64,25 +64,25 @@ DATA_STRUCTURES_AND_INTERFACES = ActionNode( example=MMC1, ) -INC_DATA_STRUCTURES_AND_INTERFACES = ActionNode( +INCREMENTAL_DATA_STRUCTURES_AND_INTERFACES = ActionNode( key="Incremental Data structures and interfaces", expected_type=str, instruction="Extend the existing mermaid classDiagram code syntax to incorporate new classes, " "methods (including __init__), and functions with precise type annotations. Clearly delineate additional " "relationships between classes, maintaining adherence to PEP8 standards. Enhance the level of detail in data " "structures, ensuring a comprehensive API design that seamlessly integrates with the existing structure.", - example=MMC1_INC_AND_REFINE, + example=MMC1_REFINE, ) -REFINE_DATA_STRUCTURES_AND_INTERFACES = ActionNode( - key="Data Structures and Interfaces", +REFINED_DATA_STRUCTURES_AND_INTERFACES = ActionNode( + key="Refined Data Structures and Interfaces", expected_type=str, instruction="Update and extend the existing mermaid classDiagram code syntax to incorporate new classes, " "methods (including __init__), and functions with precise type annotations. Delineate additional " "relationships between classes, ensuring clarity and adherence to PEP8 standards. Further enhance the " "detail in data structures for a comprehensive API design that seamlessly integrates with the evolving structure." "Retain any content unrelated to incremental development for coherence and clarity.", - example=MMC1_INC_AND_REFINE, + example=MMC1_REFINE, ) PROGRAM_CALL_FLOW = ActionNode( @@ -93,13 +93,13 @@ PROGRAM_CALL_FLOW = ActionNode( example=MMC2, ) -REFINE_PROGRAM_CALL_FLOW = ActionNode( - key="Program call flow", +REFINED_PROGRAM_CALL_FLOW = ActionNode( + key="Refined Program call flow", expected_type=str, instruction="Extend the existing sequenceDiagram code syntax with detailed information, accurately covering the" "CRUD and initialization of each object. Ensure correct syntax usage and reflect the incremental changes introduced" "in the classes and API defined above.Retain content unrelated to incremental development for coherence and clarity", - example=MMC2_INC, + example=MMC2_REFINE, ) ANYTHING_UNCLEAR = ActionNode( @@ -110,13 +110,13 @@ ANYTHING_UNCLEAR = ActionNode( ) INC_DESIGN_CONTEXT = """ -### Legacy Content +## Legacy Content {old_design} -### New Requirements +## New Requirements {requirements} -### PRD Increment Content +## PRD Increment Content {prd_increment} """ @@ -124,43 +124,44 @@ REFINE_DESIGN_CONTEXT = """ Role: You are a professional Architect tasked with overseeing incremental development. Based on new requirements, review and refine the system design. Integrate existing architecture with incremental design changes, ensuring the refined design encompasses all architectural elements, enhancements, and adjustments. Retain content unrelated to incremental development needs for coherence and clarity. -### New Requirements +# Context +## New Requirements {requirements} -### Legacy Content +## Legacy Content {old_design} -### Design Increment Content +## Design Increment Content {design_increment} """ NODES = [ IMPLEMENTATION_APPROACH, # PROJECT_NAME, - FILE_LIST, + REFINED_FILE_LIST, DATA_STRUCTURES_AND_INTERFACES, PROGRAM_CALL_FLOW, ANYTHING_UNCLEAR, ] -INC_NODES = [INC_IMPLEMENTATION_APPROACH, INC_DATA_STRUCTURES_AND_INTERFACES, REFINE_PROGRAM_CALL_FLOW] +INC_NODES = [INCREMENTAL_IMPLEMENTATION_APPROACH, INCREMENTAL_DATA_STRUCTURES_AND_INTERFACES, REFINED_PROGRAM_CALL_FLOW] REFINE_NODES = [ - REFINE_IMPLEMENTATION_APPROACH, + REFINED_IMPLEMENTATION_APPROACH, # PROJECT_NAME, - REFINE_FILE_LIST, - REFINE_DATA_STRUCTURES_AND_INTERFACES, - REFINE_PROGRAM_CALL_FLOW, + FILE_LIST, + REFINED_DATA_STRUCTURES_AND_INTERFACES, + REFINED_PROGRAM_CALL_FLOW, ANYTHING_UNCLEAR, ] DESIGN_API_NODE = ActionNode.from_children("DesignAPI", NODES) -INC_DESIGN_NODES = ActionNode.from_children("Incremental Design API", INC_NODES) -REFINE_DESIGN_NODES = ActionNode.from_children("Refine Design API", REFINE_NODES) +INCREMENTAL_DESIGN_NODES = ActionNode.from_children("Incremental_Design_API", INC_NODES) +REFINED_DESIGN_NODES = ActionNode.from_children("Refined_Design_API", REFINE_NODES) def main(): - prompt = REFINE_DESIGN_NODES.compile(context="12313") + prompt = REFINED_DESIGN_NODES.compile(context="...") logger.info(prompt) diff --git a/metagpt/actions/project_management.py b/metagpt/actions/project_management.py index 5d2791ea8..f124ba8df 100644 --- a/metagpt/actions/project_management.py +++ b/metagpt/actions/project_management.py @@ -17,7 +17,7 @@ from pydantic import Field from metagpt.actions import ActionOutput from metagpt.actions.action import Action -from metagpt.actions.project_management_an import PM_NODE, REFINE_PM_NODES +from metagpt.actions.project_management_an import PM_NODE, REFINED_PM_NODES from metagpt.config import CONFIG from metagpt.const import ( PACKAGE_REQUIREMENTS_FILENAME, @@ -101,7 +101,7 @@ class WriteTasks(Action): async def _merge(self, system_design_doc, task_doc, schema=CONFIG.prompt_schema) -> Document: context = NEW_REQ_TEMPLATE.format(context=system_design_doc.content, old_tasks=task_doc.content) - node = await REFINE_PM_NODES.fill(context, self.llm, schema) + node = await REFINED_PM_NODES.fill(context, self.llm, schema) task_doc.content = node.instruct_content.json(ensure_ascii=False) return task_doc diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index 27d238e9f..467d26b4e 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -35,18 +35,20 @@ LOGIC_ANALYSIS = ActionNode( ], ) -INC_LOGIC_ANALYSIS = ActionNode( - key="Increment Logic Analysis", +INCREMENTAL_LOGIC_ANALYSIS = ActionNode( + key="Incremental Logic Analysis", expected_type=List[List[str]], - instruction="Provide a list of files with the classes/methods/functions to be implemented or modified incrementally. Include thorough dependency analysis, consider potential impacts on existing code, and document necessary imports.", + instruction="Provide a list of files with the classes/methods/functions to be implemented or modified " + "incrementally. Include thorough dependency analysis, consider potential impacts on existing code, and document" + " necessary imports.", example=[ ["new_feature.py", "Introduces NewFeature class and related functions"], ["utils.py", "Modifies existing utility functions to support incremental changes"], ], ) -REFINE_LOGIC_ANALYSIS = ActionNode( - key="Logic Analysis", +REFINED_LOGIC_ANALYSIS = ActionNode( + key="Refined Logic Analysis", expected_type=List[List[str]], instruction="Review and refine the logic analysis by merging the Legacy Content and Incremental Content. " "Provide a comprehensive list of files with classes/methods/functions to be implemented or modified incrementally. " @@ -67,7 +69,7 @@ TASK_LIST = ActionNode( example=["game.py", "main.py"], ) -INC_TASK_LIST = ActionNode( +INCREMENTAL_TASK_LIST = ActionNode( key="Incremental Task list", expected_type=List[str], instruction="Break down the incremental development tasks into a prioritized list of filenames." @@ -76,8 +78,8 @@ INC_TASK_LIST = ActionNode( example=["new_feature.py", "utils.py", "main.py"], ) -REFINE_TASK_LIST = ActionNode( - key="Task list", +REFINED_TASK_LIST = ActionNode( + key="Refined Task list", expected_type=List[str], instruction="Review and refine the combined task list after the merger of Legacy Content and Incremental Content. " "Ensure that tasks are organized in a logical and prioritized order, considering dependencies for a streamlined and" @@ -100,20 +102,20 @@ SHARED_KNOWLEDGE = ActionNode( example="`game.py` contains functions shared across the project.", ) -INC_SHARED_KNOWLEDGE = ActionNode( - key="Increment Shared Knowledge", +INCREMENTAL_SHARED_KNOWLEDGE = ActionNode( + key="Incremental Shared Knowledge", expected_type=str, instruction="Document any new shared knowledge generated during incremental development. This includes common " "utility functions, configuration variables, or any information vital for team collaboration.", example="`new_module.py` introduces shared utility functions for improved code reusability.", ) -REFINE_SHARED_KNOWLEDGE = ActionNode( - key="Shared Knowledge", +REFINED_SHARED_KNOWLEDGE = ActionNode( + key="Refined Shared Knowledge", expected_type=str, - instruction="Update and expand shared knowledge to reflect any new elements introduced during incremental development. " - "This includes common utility functions, configuration variables, or any information vital for team collaboration." - "Retain any content unrelated to incremental development for coherence and clarity.", + instruction="Update and expand shared knowledge to reflect any new elements introduced during incremental " + "development. This includes common utility functions, configuration variables, or any information vital for team " + "collaboration. Retain any content unrelated to incremental development for coherence and clarity.", example="`new_module.py` enhances shared utility functions for improved code reusability and collaboration.", ) @@ -140,13 +142,14 @@ REFINE_PM_CONTEXT = """ Role: You are a professional Project Manager tasked with overseeing incremental development. Based on New Requirements, refine the project context to account for incremental development. Ensure the context offers a comprehensive overview of the project's evolving scope, covering both legacy content and incremental content. Retain any content unrelated to incremental development. -### New Requirements +# Context +## New Requirements {requirements} -### Legacy Content +## Legacy Content {old_tasks} -### Increment Content +## Increment Content {tasks_increment} """ @@ -160,21 +163,21 @@ NODES = [ ANYTHING_UNCLEAR_PM, ] -INC_NODES = [INC_LOGIC_ANALYSIS, INC_TASK_LIST, INC_SHARED_KNOWLEDGE] +INC_NODES = [INCREMENTAL_LOGIC_ANALYSIS, INCREMENTAL_TASK_LIST, INCREMENTAL_SHARED_KNOWLEDGE] REFINE_NODES = [ REQUIRED_PYTHON_PACKAGES, REQUIRED_OTHER_LANGUAGE_PACKAGES, - REFINE_LOGIC_ANALYSIS, - REFINE_TASK_LIST, + REFINED_LOGIC_ANALYSIS, + REFINED_TASK_LIST, FULL_API_SPEC, - REFINE_SHARED_KNOWLEDGE, + REFINED_SHARED_KNOWLEDGE, ANYTHING_UNCLEAR_PM, ] PM_NODE = ActionNode.from_children("PM_NODE", NODES) -INC_PM_NODES = ActionNode.from_children("Incremental_PM_NODES", INC_NODES) -REFINE_PM_NODES = ActionNode.from_children("Refine_PM_NODES", REFINE_NODES) +INCREMENTAL_PM_NODES = ActionNode.from_children("Incremental_PM_NODES", INC_NODES) +REFINED_PM_NODES = ActionNode.from_children("Refined_PM_NODES", REFINE_NODES) def main(): diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 33ae2e5b3..4183db19d 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -21,7 +21,7 @@ from pydantic import Field from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions.action import Action -from metagpt.actions.write_code_guide_an import WRITE_CODE_INCREMENT_TEMPLATE +from metagpt.actions.write_code_guideline_an import REFINED_TEMPLATE from metagpt.config import CONFIG from metagpt.const import ( BUGFIX_FILENAME, @@ -127,7 +127,7 @@ class WriteCode(Action): code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename) if guideline: - prompt = WRITE_CODE_INCREMENT_TEMPLATE.format( + prompt = REFINED_TEMPLATE.format( requirement=requirement_doc.content if requirement_doc else "", guideline=guideline, design=coding_context.design_doc.content if coding_context.design_doc else "", @@ -164,7 +164,7 @@ class WriteCode(Action): if not task_doc.content: task_doc.content = FileRepository.get_file(filename=task_doc.filename, relative_path=TASK_FILE_REPO) m = json.loads(task_doc.content) - code_filenames = m.get("Task list", []) + code_filenames = m.get("Task list", []) if mode == "normal" else m.get("Refined Task list", []) codes = [] src_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.src_workspace) diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guideline_an.py similarity index 87% rename from metagpt/actions/write_code_guide_an.py rename to metagpt/actions/write_code_guideline_an.py index b21e66098..d68b02d38 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -7,18 +7,13 @@ """ import asyncio -from pydantic import Field - from metagpt.actions.action import Action from metagpt.actions.action_node import ActionNode -from metagpt.llm import LLM -from metagpt.provider.base_gpt_api import BaseGPTAPI -from metagpt.schema import Document GUIDELINE = ActionNode( - key="Code Guideline", + key="Guideline", expected_type=list[str], - instruction="Developing comprehensive and incremental software development plans while providing detailed code guidance.", + instruction="Developing comprehensive and incremental development plans while providing detailed code guideline.", example=[ "Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Implement robust error handling for the division operation to mitigate potential issues related to division by zero.", "Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Ensure seamless integration with the overall application architecture and maintain consistency with coding standards.", @@ -28,7 +23,8 @@ GUIDELINE = ActionNode( INCREMENTAL_CHANGE = ActionNode( key="Incremental Change", expected_type=str, - instruction="Write Incremental Change by making a code draft that how to implement incremental development including detailed steps based on the context.", + instruction="Write Incremental Change by making a code draft that how to implement incremental development " + "including detailed steps based on the context.", example="""- calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Implement robust error handling for the division operation to mitigate potential issues related to division by zero. ```python ## calculator.py @@ -81,8 +77,8 @@ if __name__ == '__main__': ```""", ) -CODE_GUIDE_CONTEXT = """ -### NOTICE +CODE_GUIDELINE_CONTEXT = """ +NOTICE Role: You are a professional software engineer, and your main task is to craft comprehensive incremental development plans and provide detailed code guidance with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". 1. Determine the scope of responsibilities of each file and what classes and methods need to be implemented. 2. Import all referenced classes. @@ -93,20 +89,21 @@ Role: You are a professional software engineer, and your main task is to craft c 7. Examine the code closely to find and fix errors, and confirm that the logic is sound to ensure smooth user interaction while meeting all specified requirements. 8. Attention: Code files in the task list may have a different number of files compared to legacy code files. This requires integrating legacy code files that do not appear in the task list into the code files of the task list. Therefore, when writing code guidance and incremental changes for the code files in the task list, also include how to seamlessly merge and adjust legacy code files. -### Requirement +# Context +## Requirement {requirement} -### Design +## Design {design} -### Tasks +## Tasks {tasks} -### Legacy Code +## Legacy Code {code} """ -WRITE_CODE_INCREMENT_TEMPLATE = """ +REFINED_TEMPLATE = """ NOTICE Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and Incremental Change, ensuring the integration of new features. @@ -161,7 +158,7 @@ Role: You are a professional engineer; The main goal is to complete incremental """ CODE_GUIDE_CONTEXT_EXAMPLE = """ -### Legacy Code +# Legacy Code ## main.py from flask import Flask, request, jsonify @@ -194,21 +191,17 @@ class Calculator: GUIDE_NODES = [INCREMENTAL_CHANGE] -WRITE_CODE_GUIDE_NODE = ActionNode.from_children("WriteCodeGuide", GUIDE_NODES) +WRITE_CODE_GUIDELINE_NODE = ActionNode.from_children("WriteCodeGuideline", GUIDE_NODES) -class WriteCodeGuide(Action): - name: str = "WriteCodeGuide" - context: Document = Field(default_factory=Document) - llm: BaseGPTAPI = Field(default_factory=LLM) - +class WriteCodeGuideline(Action): async def run(self, context): - return await WRITE_CODE_GUIDE_NODE.fill(context=context, llm=self.llm, schema="json") + return await WRITE_CODE_GUIDELINE_NODE.fill(context=context, llm=self.llm, schema="json") def main(): - action = WriteCodeGuide() - return asyncio.run(action.run(CODE_GUIDE_CONTEXT)) + action = WriteCodeGuideline() + return asyncio.run(action.run(CODE_GUIDELINE_CONTEXT)) if __name__ == "__main__": diff --git a/metagpt/actions/write_prd.py b/metagpt/actions/write_prd.py index dc1af0735..a070e7a96 100644 --- a/metagpt/actions/write_prd.py +++ b/metagpt/actions/write_prd.py @@ -24,7 +24,7 @@ from metagpt.actions.action_node import ActionNode from metagpt.actions.fix_bug import FixBug from metagpt.actions.write_prd_an import ( REFINE_PRD_NODE, - REFINE_PRD_SIMPLE_CONTEXT, + REFINE_PRD_TEMPLATE, WP_IS_RELATIVE_NODE, WP_ISSUE_TYPE_NODE, WRITE_PRD_NODE, @@ -139,7 +139,7 @@ class WritePRD(Action): CONFIG.project_name = Path(CONFIG.project_path).name project_name = CONFIG.project_name if CONFIG.project_name else "" - prompt = REFINE_PRD_SIMPLE_CONTEXT.format( + prompt = REFINE_PRD_TEMPLATE.format( requirements=new_requirement_doc.content, old_prd=prd_doc.content, project_name=project_name, diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index 79046bb7d..f645225c7 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -31,8 +31,8 @@ ORIGINAL_REQUIREMENTS = ActionNode( example="Create a 2048 game", ) -REFINE_REQUIREMENTS = ActionNode( - key="Original Requirements", +REFINED_REQUIREMENTS = ActionNode( + key="Refined Requirements", expected_type=str, instruction="Update and expand the original user's requirements to reflect the evolving needs of the project." "Retain any content unrelated to incremental development", @@ -46,13 +46,6 @@ PROJECT_NAME = ActionNode( example="game_2048", ) -REFINE_PROJECT_NAME = ActionNode( - key="Project Name", - expected_type=str, - instruction="Update the project name based on the context.", - example="game_2048_new", -) - PRODUCT_GOALS = ActionNode( key="Product Goals", expected_type=List[str], @@ -60,8 +53,8 @@ PRODUCT_GOALS = ActionNode( example=["Create an engaging user experience", "Improve accessibility, be responsive", "More beautiful UI"], ) -REFINE_PRODUCT_GOALS = ActionNode( - key="Product Goals", +REFINED_PRODUCT_GOALS = ActionNode( + key="Refined Product Goals", expected_type=List[str], instruction="Update and expand the original product goals to reflect the evolving needs due to incremental " "development.Ensure that the refined goals align with the current project direction and contribute to its success." @@ -86,11 +79,11 @@ USER_STORIES = ActionNode( ], ) -REFINE_USER_STORIES = ActionNode( - key="User Stories", +REFINED_USER_STORIES = ActionNode( + key="Refined User Stories", expected_type=List[str], instruction="Update and expand the original scenario-based user stories to reflect the evolving needs due to " - "incremental development, no less than 7. Ensure that the refined user stories capture incremental features and " + "incremental development, no less than 5. Ensure that the refined user stories capture incremental features and " "improvements. Retain any content unrelated to incremental development", example=[ "As a player, I want to choose difficulty levels to challenge my skills", @@ -140,6 +133,14 @@ REQUIREMENT_ANALYSIS = ActionNode( example="", ) +INCREMENTAL_REQUIREMENT_ANALYSIS = ActionNode( + key="Incremental Requirement Analysis", + expected_type=List[str], + instruction="Propose the comprehensive incremental development requirement analysis on new features and enhanced " + "features for New Requirements.", + example=["Require add/update/modify ..."], +) + REQUIREMENT_POOL = ActionNode( key="Requirement Pool", expected_type=List[List[str]], @@ -147,8 +148,8 @@ REQUIREMENT_POOL = ActionNode( example=[["P0", "The main code ..."], ["P0", "The game algorithm ..."]], ) -REFINE_REQUIREMENT_POOL = ActionNode( - key="Requirement Pool", +REFINED_REQUIREMENT_POOL = ActionNode( + key="Refined Requirement Pool", expected_type=List[List[str]], instruction="List no less than 7 requirements with their priority (P0, P1, P2). " "Cover both legacy content and incremental content. Retain any content unrelated to incremental development", @@ -192,25 +193,18 @@ REFINE_PRD_CONTEXT = """ Role: You are a professional Product Manager tasked with overseeing incremental development. Based on New Requirements, output a New PRD that seamlessly integrates both the Legacy Content and the Incremental Content. Ensure the resulting document captures the complete scope of features, enhancements, and retain content unrelated to incremental development needs for coherence and clarity. -### New Project Name -{project_name} - -### New Requirements +# Context +## New Requirements {requirements} -### Legacy Content +## Legacy Content {old_prd} -### PRD Incremental Content +## PRD Incremental Content {prd_increment} - -### Search Information -- """ -REFINE_PRD_SIMPLE_CONTEXT = """ -You are a professional Product Manager tasked with overseeing incremental development. - +REFINE_PRD_TEMPLATE = """ ### New Project Name {project_name} @@ -224,13 +218,6 @@ You are a professional Product Manager tasked with overseeing incremental develo - """ -INCREMENTAL_DEVELOPMENT_ANALYSIS = ActionNode( - key="Requirement Analysis", - expected_type=List[str], - instruction="Propose the comprehensive incremental development requirement analysis on new features and enhanced features for New Requirements.", - example=["Require add/update/modify ..."], -) - NODES = [ LANGUAGE, @@ -250,30 +237,30 @@ NODES = [ REFINE_NODES = [ LANGUAGE, PROGRAMMING_LANGUAGE, - REFINE_REQUIREMENTS, - REFINE_PROJECT_NAME, - REFINE_PRODUCT_GOALS, - REFINE_USER_STORIES, + REFINED_REQUIREMENTS, + PROJECT_NAME, + REFINED_PRODUCT_GOALS, + REFINED_USER_STORIES, COMPETITIVE_ANALYSIS, COMPETITIVE_QUADRANT_CHART, - INCREMENTAL_DEVELOPMENT_ANALYSIS, - REFINE_REQUIREMENT_POOL, + INCREMENTAL_REQUIREMENT_ANALYSIS, + REFINED_REQUIREMENT_POOL, UI_DESIGN_DRAFT, ANYTHING_UNCLEAR, ] -INCREMENT_PRD_NODES = [INCREMENTAL_DEVELOPMENT_ANALYSIS, REQUIREMENT_POOL] +INCREMENT_PRD_NODES = [INCREMENTAL_REQUIREMENT_ANALYSIS, REQUIREMENT_POOL] WRITE_PRD_NODE = ActionNode.from_children("WritePRD", NODES) REFINE_PRD_NODE = ActionNode.from_children("RefinePRD", REFINE_NODES) -INCREMENT_NODE = ActionNode.from_children("IncrementPRD", INCREMENT_PRD_NODES) +INCREMENTAL_PRD_NODE = ActionNode.from_children("IncrementalPRD", INCREMENT_PRD_NODES) WP_ISSUE_TYPE_NODE = ActionNode.from_children("WP_ISSUE_TYPE", [ISSUE_TYPE, REASON]) WP_IS_RELATIVE_NODE = ActionNode.from_children("WP_IS_RELATIVE", [IS_RELATIVE, REASON]) def main(): # prompt = WRITE_PRD_NODE.compile(context="") - prompt = INCREMENT_NODE.compile(context=REFINE_PRD_CONTEXT) + prompt = INCREMENTAL_PRD_NODE.compile(context=REFINE_PRD_CONTEXT) logger.info(prompt) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 4aa8f209d..d3584e987 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -28,7 +28,10 @@ from typing import Set from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks from metagpt.actions.fix_bug import FixBug from metagpt.actions.summarize_code import SummarizeCode -from metagpt.actions.write_code_guide_an import CODE_GUIDE_CONTEXT, WriteCodeGuide +from metagpt.actions.write_code_guideline_an import ( + CODE_GUIDELINE_CONTEXT, + WriteCodeGuideline, +) from metagpt.config import CONFIG from metagpt.const import ( CODE_SUMMARIES_FILE_REPO, @@ -323,8 +326,8 @@ class Engineer(Role): tasks = "\n".join([doc.content for doc in tasks]) old_codes = await self.get_old_codes() - context = CODE_GUIDE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) - node = await WriteCodeGuide().run(context=context) + context = CODE_GUIDELINE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) + node = await WriteCodeGuideline().run(context=context) guideline = node.instruct_content.json(ensure_ascii=False) return guideline diff --git a/metagpt/utils/mermaid.py b/metagpt/utils/mermaid.py index d1cd1b328..76adbf5a7 100644 --- a/metagpt/utils/mermaid.py +++ b/metagpt/utils/mermaid.py @@ -120,7 +120,7 @@ MMC1 = """classDiagram SearchEngine --> Summary Index --> KnowledgeBase""" -MMC1_INC_AND_REFINE = """classDiagram +MMC1_REFINE = """classDiagram class Main { -SearchEngine search_engine +main() str @@ -176,7 +176,7 @@ MMC2 = """sequenceDiagram S-->>SE: return summary SE-->>M: return summary""" -MMC2_INC = """sequenceDiagram +MMC2_REFINE = """sequenceDiagram participant M as Main participant SE as SearchEngine participant I as Index From a24bfdc6c7f98c9d22530701fbb0bb04380fad00 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 2 Jan 2024 22:57:42 +0800 Subject: [PATCH 251/637] Modify style and format --- metagpt/actions/design_api.py | 8 +- metagpt/actions/design_api_an.py | 69 ++++++++--------- metagpt/actions/project_management.py | 4 +- metagpt/actions/project_management_an.py | 51 +++++++------ metagpt/actions/write_code.py | 6 +- ...guide_an.py => write_code_guideline_an.py} | 43 +++++------ metagpt/actions/write_prd.py | 4 +- metagpt/actions/write_prd_an.py | 75 ++++++++----------- metagpt/roles/engineer.py | 11 ++- metagpt/utils/mermaid.py | 4 +- 10 files changed, 131 insertions(+), 144 deletions(-) rename metagpt/actions/{write_code_guide_an.py => write_code_guideline_an.py} (87%) diff --git a/metagpt/actions/design_api.py b/metagpt/actions/design_api.py index 0cc3395ed..082474098 100644 --- a/metagpt/actions/design_api.py +++ b/metagpt/actions/design_api.py @@ -16,7 +16,7 @@ from typing import Optional from pydantic import Field from metagpt.actions import Action, ActionOutput -from metagpt.actions.design_api_an import DESIGN_API_NODE, REFINE_DESIGN_NODES +from metagpt.actions.design_api_an import DESIGN_API_NODE, REFINED_DESIGN_NODES from metagpt.config import CONFIG from metagpt.const import ( DATA_API_DESIGN_FILE_REPO, @@ -87,7 +87,7 @@ class WriteDesign(Action): async def _merge(self, prd_doc, system_design_doc, schema=CONFIG.prompt_schema): context = NEW_REQ_TEMPLATE.format(old_design=system_design_doc.content, context=prd_doc.content) - node = await REFINE_DESIGN_NODES.fill(context=context, llm=self.llm, schema=schema) + node = await REFINED_DESIGN_NODES.fill(context=context, llm=self.llm, schema=schema) system_design_doc.content = node.instruct_content.json(ensure_ascii=False) return system_design_doc @@ -114,7 +114,7 @@ class WriteDesign(Action): @staticmethod async def _save_data_api_design(design_doc): m = json.loads(design_doc.content) - data_api_design = m.get("Data structures and interfaces") + data_api_design = m.get("Data structures and interfaces") or m.get("Refined Data structures and interfaces") if not data_api_design: return pathname = CONFIG.git_repo.workdir / DATA_API_DESIGN_FILE_REPO / Path(design_doc.filename).with_suffix("") @@ -124,7 +124,7 @@ class WriteDesign(Action): @staticmethod async def _save_seq_flow(design_doc): m = json.loads(design_doc.content) - seq_flow = m.get("Program call flow") + seq_flow = m.get("Program call flow") or m.get("Refined Program call flow") if not seq_flow: return pathname = CONFIG.git_repo.workdir / Path(SEQ_FLOW_FILE_REPO) / Path(design_doc.filename).with_suffix("") diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index 5f17d4656..fcfaff9c3 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -9,7 +9,7 @@ from typing import List from metagpt.actions.action_node import ActionNode from metagpt.logs import logger -from metagpt.utils.mermaid import MMC1, MMC1_INC_AND_REFINE, MMC2, MMC2_INC +from metagpt.utils.mermaid import MMC1, MMC1_REFINE, MMC2, MMC2_REFINE IMPLEMENTATION_APPROACH = ActionNode( key="Implementation approach", @@ -18,7 +18,7 @@ IMPLEMENTATION_APPROACH = ActionNode( example="We will ...", ) -INC_IMPLEMENTATION_APPROACH = ActionNode( +INCREMENTAL_IMPLEMENTATION_APPROACH = ActionNode( key="Incremental Implementation approach", expected_type=str, instruction="Analyze the challenging aspects of the requirements and select a suitable open-source framework. " @@ -26,12 +26,12 @@ INC_IMPLEMENTATION_APPROACH = ActionNode( example="we will ...", ) -REFINE_IMPLEMENTATION_APPROACH = ActionNode( - key="Implementation Approach", +REFINED_IMPLEMENTATION_APPROACH = ActionNode( + key="Refined Implementation Approach", expected_type=str, - instruction="Update and extend the original implementation approach to reflect the evolving challenges and requirements " - "due to incremental development. Provide detailed strategies for incremental steps in the implementation process." - "etain any content unrelated to incremental development for coherence and clarity.", + instruction="Update and extend the original implementation approach to reflect the evolving challenges and " + "requirements due to incremental development. Provide detailed strategies for incremental steps in the " + "implementation process. Retain any content unrelated to incremental development for coherence and clarity.", example="We will refine ...", ) @@ -46,13 +46,13 @@ FILE_LIST = ActionNode( example=["main.py", "game.py"], ) -REFINE_FILE_LIST = ActionNode( - key="File List", +REFINED_FILE_LIST = ActionNode( + key="Refined File List", expected_type=List[str], instruction="Update and expand the original file list, including only relative paths. " "Ensure that the refined file list reflects the evolving structure of the project due to incremental development." - "Only output filename!Do not include comments in the list.", - example=["main.py", "game.py", "utils.py", "new_feature.py"], + "Only output filename! Do not include comments in the list.", + example=["main.py", "game.py", "new_feature.py"], ) DATA_STRUCTURES_AND_INTERFACES = ActionNode( @@ -64,25 +64,25 @@ DATA_STRUCTURES_AND_INTERFACES = ActionNode( example=MMC1, ) -INC_DATA_STRUCTURES_AND_INTERFACES = ActionNode( +INCREMENTAL_DATA_STRUCTURES_AND_INTERFACES = ActionNode( key="Incremental Data structures and interfaces", expected_type=str, instruction="Extend the existing mermaid classDiagram code syntax to incorporate new classes, " "methods (including __init__), and functions with precise type annotations. Clearly delineate additional " "relationships between classes, maintaining adherence to PEP8 standards. Enhance the level of detail in data " "structures, ensuring a comprehensive API design that seamlessly integrates with the existing structure.", - example=MMC1_INC_AND_REFINE, + example=MMC1_REFINE, ) -REFINE_DATA_STRUCTURES_AND_INTERFACES = ActionNode( - key="Data Structures and Interfaces", +REFINED_DATA_STRUCTURES_AND_INTERFACES = ActionNode( + key="Refined Data Structures and Interfaces", expected_type=str, instruction="Update and extend the existing mermaid classDiagram code syntax to incorporate new classes, " "methods (including __init__), and functions with precise type annotations. Delineate additional " "relationships between classes, ensuring clarity and adherence to PEP8 standards. Further enhance the " "detail in data structures for a comprehensive API design that seamlessly integrates with the evolving structure." "Retain any content unrelated to incremental development for coherence and clarity.", - example=MMC1_INC_AND_REFINE, + example=MMC1_REFINE, ) PROGRAM_CALL_FLOW = ActionNode( @@ -93,13 +93,13 @@ PROGRAM_CALL_FLOW = ActionNode( example=MMC2, ) -REFINE_PROGRAM_CALL_FLOW = ActionNode( - key="Program call flow", +REFINED_PROGRAM_CALL_FLOW = ActionNode( + key="Refined Program call flow", expected_type=str, instruction="Extend the existing sequenceDiagram code syntax with detailed information, accurately covering the" "CRUD and initialization of each object. Ensure correct syntax usage and reflect the incremental changes introduced" "in the classes and API defined above.Retain content unrelated to incremental development for coherence and clarity", - example=MMC2_INC, + example=MMC2_REFINE, ) ANYTHING_UNCLEAR = ActionNode( @@ -110,13 +110,13 @@ ANYTHING_UNCLEAR = ActionNode( ) INC_DESIGN_CONTEXT = """ -### Legacy Content +## Legacy Content {old_design} -### New Requirements +## New Requirements {requirements} -### PRD Increment Content +## PRD Increment Content {prd_increment} """ @@ -124,43 +124,44 @@ REFINE_DESIGN_CONTEXT = """ Role: You are a professional Architect tasked with overseeing incremental development. Based on new requirements, review and refine the system design. Integrate existing architecture with incremental design changes, ensuring the refined design encompasses all architectural elements, enhancements, and adjustments. Retain content unrelated to incremental development needs for coherence and clarity. -### New Requirements +# Context +## New Requirements {requirements} -### Legacy Content +## Legacy Content {old_design} -### Design Increment Content +## Design Increment Content {design_increment} """ NODES = [ IMPLEMENTATION_APPROACH, # PROJECT_NAME, - FILE_LIST, + REFINED_FILE_LIST, DATA_STRUCTURES_AND_INTERFACES, PROGRAM_CALL_FLOW, ANYTHING_UNCLEAR, ] -INC_NODES = [INC_IMPLEMENTATION_APPROACH, INC_DATA_STRUCTURES_AND_INTERFACES, REFINE_PROGRAM_CALL_FLOW] +INC_NODES = [INCREMENTAL_IMPLEMENTATION_APPROACH, INCREMENTAL_DATA_STRUCTURES_AND_INTERFACES, REFINED_PROGRAM_CALL_FLOW] REFINE_NODES = [ - REFINE_IMPLEMENTATION_APPROACH, + REFINED_IMPLEMENTATION_APPROACH, # PROJECT_NAME, - REFINE_FILE_LIST, - REFINE_DATA_STRUCTURES_AND_INTERFACES, - REFINE_PROGRAM_CALL_FLOW, + FILE_LIST, + REFINED_DATA_STRUCTURES_AND_INTERFACES, + REFINED_PROGRAM_CALL_FLOW, ANYTHING_UNCLEAR, ] DESIGN_API_NODE = ActionNode.from_children("DesignAPI", NODES) -INC_DESIGN_NODES = ActionNode.from_children("Incremental Design API", INC_NODES) -REFINE_DESIGN_NODES = ActionNode.from_children("Refine Design API", REFINE_NODES) +INCREMENTAL_DESIGN_NODES = ActionNode.from_children("Incremental_Design_API", INC_NODES) +REFINED_DESIGN_NODES = ActionNode.from_children("Refined_Design_API", REFINE_NODES) def main(): - prompt = REFINE_DESIGN_NODES.compile(context="12313") + prompt = REFINED_DESIGN_NODES.compile(context="...") logger.info(prompt) diff --git a/metagpt/actions/project_management.py b/metagpt/actions/project_management.py index 5d2791ea8..f124ba8df 100644 --- a/metagpt/actions/project_management.py +++ b/metagpt/actions/project_management.py @@ -17,7 +17,7 @@ from pydantic import Field from metagpt.actions import ActionOutput from metagpt.actions.action import Action -from metagpt.actions.project_management_an import PM_NODE, REFINE_PM_NODES +from metagpt.actions.project_management_an import PM_NODE, REFINED_PM_NODES from metagpt.config import CONFIG from metagpt.const import ( PACKAGE_REQUIREMENTS_FILENAME, @@ -101,7 +101,7 @@ class WriteTasks(Action): async def _merge(self, system_design_doc, task_doc, schema=CONFIG.prompt_schema) -> Document: context = NEW_REQ_TEMPLATE.format(context=system_design_doc.content, old_tasks=task_doc.content) - node = await REFINE_PM_NODES.fill(context, self.llm, schema) + node = await REFINED_PM_NODES.fill(context, self.llm, schema) task_doc.content = node.instruct_content.json(ensure_ascii=False) return task_doc diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index 27d238e9f..467d26b4e 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -35,18 +35,20 @@ LOGIC_ANALYSIS = ActionNode( ], ) -INC_LOGIC_ANALYSIS = ActionNode( - key="Increment Logic Analysis", +INCREMENTAL_LOGIC_ANALYSIS = ActionNode( + key="Incremental Logic Analysis", expected_type=List[List[str]], - instruction="Provide a list of files with the classes/methods/functions to be implemented or modified incrementally. Include thorough dependency analysis, consider potential impacts on existing code, and document necessary imports.", + instruction="Provide a list of files with the classes/methods/functions to be implemented or modified " + "incrementally. Include thorough dependency analysis, consider potential impacts on existing code, and document" + " necessary imports.", example=[ ["new_feature.py", "Introduces NewFeature class and related functions"], ["utils.py", "Modifies existing utility functions to support incremental changes"], ], ) -REFINE_LOGIC_ANALYSIS = ActionNode( - key="Logic Analysis", +REFINED_LOGIC_ANALYSIS = ActionNode( + key="Refined Logic Analysis", expected_type=List[List[str]], instruction="Review and refine the logic analysis by merging the Legacy Content and Incremental Content. " "Provide a comprehensive list of files with classes/methods/functions to be implemented or modified incrementally. " @@ -67,7 +69,7 @@ TASK_LIST = ActionNode( example=["game.py", "main.py"], ) -INC_TASK_LIST = ActionNode( +INCREMENTAL_TASK_LIST = ActionNode( key="Incremental Task list", expected_type=List[str], instruction="Break down the incremental development tasks into a prioritized list of filenames." @@ -76,8 +78,8 @@ INC_TASK_LIST = ActionNode( example=["new_feature.py", "utils.py", "main.py"], ) -REFINE_TASK_LIST = ActionNode( - key="Task list", +REFINED_TASK_LIST = ActionNode( + key="Refined Task list", expected_type=List[str], instruction="Review and refine the combined task list after the merger of Legacy Content and Incremental Content. " "Ensure that tasks are organized in a logical and prioritized order, considering dependencies for a streamlined and" @@ -100,20 +102,20 @@ SHARED_KNOWLEDGE = ActionNode( example="`game.py` contains functions shared across the project.", ) -INC_SHARED_KNOWLEDGE = ActionNode( - key="Increment Shared Knowledge", +INCREMENTAL_SHARED_KNOWLEDGE = ActionNode( + key="Incremental Shared Knowledge", expected_type=str, instruction="Document any new shared knowledge generated during incremental development. This includes common " "utility functions, configuration variables, or any information vital for team collaboration.", example="`new_module.py` introduces shared utility functions for improved code reusability.", ) -REFINE_SHARED_KNOWLEDGE = ActionNode( - key="Shared Knowledge", +REFINED_SHARED_KNOWLEDGE = ActionNode( + key="Refined Shared Knowledge", expected_type=str, - instruction="Update and expand shared knowledge to reflect any new elements introduced during incremental development. " - "This includes common utility functions, configuration variables, or any information vital for team collaboration." - "Retain any content unrelated to incremental development for coherence and clarity.", + instruction="Update and expand shared knowledge to reflect any new elements introduced during incremental " + "development. This includes common utility functions, configuration variables, or any information vital for team " + "collaboration. Retain any content unrelated to incremental development for coherence and clarity.", example="`new_module.py` enhances shared utility functions for improved code reusability and collaboration.", ) @@ -140,13 +142,14 @@ REFINE_PM_CONTEXT = """ Role: You are a professional Project Manager tasked with overseeing incremental development. Based on New Requirements, refine the project context to account for incremental development. Ensure the context offers a comprehensive overview of the project's evolving scope, covering both legacy content and incremental content. Retain any content unrelated to incremental development. -### New Requirements +# Context +## New Requirements {requirements} -### Legacy Content +## Legacy Content {old_tasks} -### Increment Content +## Increment Content {tasks_increment} """ @@ -160,21 +163,21 @@ NODES = [ ANYTHING_UNCLEAR_PM, ] -INC_NODES = [INC_LOGIC_ANALYSIS, INC_TASK_LIST, INC_SHARED_KNOWLEDGE] +INC_NODES = [INCREMENTAL_LOGIC_ANALYSIS, INCREMENTAL_TASK_LIST, INCREMENTAL_SHARED_KNOWLEDGE] REFINE_NODES = [ REQUIRED_PYTHON_PACKAGES, REQUIRED_OTHER_LANGUAGE_PACKAGES, - REFINE_LOGIC_ANALYSIS, - REFINE_TASK_LIST, + REFINED_LOGIC_ANALYSIS, + REFINED_TASK_LIST, FULL_API_SPEC, - REFINE_SHARED_KNOWLEDGE, + REFINED_SHARED_KNOWLEDGE, ANYTHING_UNCLEAR_PM, ] PM_NODE = ActionNode.from_children("PM_NODE", NODES) -INC_PM_NODES = ActionNode.from_children("Incremental_PM_NODES", INC_NODES) -REFINE_PM_NODES = ActionNode.from_children("Refine_PM_NODES", REFINE_NODES) +INCREMENTAL_PM_NODES = ActionNode.from_children("Incremental_PM_NODES", INC_NODES) +REFINED_PM_NODES = ActionNode.from_children("Refined_PM_NODES", REFINE_NODES) def main(): diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 33ae2e5b3..4183db19d 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -21,7 +21,7 @@ from pydantic import Field from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions.action import Action -from metagpt.actions.write_code_guide_an import WRITE_CODE_INCREMENT_TEMPLATE +from metagpt.actions.write_code_guideline_an import REFINED_TEMPLATE from metagpt.config import CONFIG from metagpt.const import ( BUGFIX_FILENAME, @@ -127,7 +127,7 @@ class WriteCode(Action): code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename) if guideline: - prompt = WRITE_CODE_INCREMENT_TEMPLATE.format( + prompt = REFINED_TEMPLATE.format( requirement=requirement_doc.content if requirement_doc else "", guideline=guideline, design=coding_context.design_doc.content if coding_context.design_doc else "", @@ -164,7 +164,7 @@ class WriteCode(Action): if not task_doc.content: task_doc.content = FileRepository.get_file(filename=task_doc.filename, relative_path=TASK_FILE_REPO) m = json.loads(task_doc.content) - code_filenames = m.get("Task list", []) + code_filenames = m.get("Task list", []) if mode == "normal" else m.get("Refined Task list", []) codes = [] src_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.src_workspace) diff --git a/metagpt/actions/write_code_guide_an.py b/metagpt/actions/write_code_guideline_an.py similarity index 87% rename from metagpt/actions/write_code_guide_an.py rename to metagpt/actions/write_code_guideline_an.py index b21e66098..d68b02d38 100644 --- a/metagpt/actions/write_code_guide_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -7,18 +7,13 @@ """ import asyncio -from pydantic import Field - from metagpt.actions.action import Action from metagpt.actions.action_node import ActionNode -from metagpt.llm import LLM -from metagpt.provider.base_gpt_api import BaseGPTAPI -from metagpt.schema import Document GUIDELINE = ActionNode( - key="Code Guideline", + key="Guideline", expected_type=list[str], - instruction="Developing comprehensive and incremental software development plans while providing detailed code guidance.", + instruction="Developing comprehensive and incremental development plans while providing detailed code guideline.", example=[ "Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Implement robust error handling for the division operation to mitigate potential issues related to division by zero.", "Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Ensure seamless integration with the overall application architecture and maintain consistency with coding standards.", @@ -28,7 +23,8 @@ GUIDELINE = ActionNode( INCREMENTAL_CHANGE = ActionNode( key="Incremental Change", expected_type=str, - instruction="Write Incremental Change by making a code draft that how to implement incremental development including detailed steps based on the context.", + instruction="Write Incremental Change by making a code draft that how to implement incremental development " + "including detailed steps based on the context.", example="""- calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Implement robust error handling for the division operation to mitigate potential issues related to division by zero. ```python ## calculator.py @@ -81,8 +77,8 @@ if __name__ == '__main__': ```""", ) -CODE_GUIDE_CONTEXT = """ -### NOTICE +CODE_GUIDELINE_CONTEXT = """ +NOTICE Role: You are a professional software engineer, and your main task is to craft comprehensive incremental development plans and provide detailed code guidance with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". 1. Determine the scope of responsibilities of each file and what classes and methods need to be implemented. 2. Import all referenced classes. @@ -93,20 +89,21 @@ Role: You are a professional software engineer, and your main task is to craft c 7. Examine the code closely to find and fix errors, and confirm that the logic is sound to ensure smooth user interaction while meeting all specified requirements. 8. Attention: Code files in the task list may have a different number of files compared to legacy code files. This requires integrating legacy code files that do not appear in the task list into the code files of the task list. Therefore, when writing code guidance and incremental changes for the code files in the task list, also include how to seamlessly merge and adjust legacy code files. -### Requirement +# Context +## Requirement {requirement} -### Design +## Design {design} -### Tasks +## Tasks {tasks} -### Legacy Code +## Legacy Code {code} """ -WRITE_CODE_INCREMENT_TEMPLATE = """ +REFINED_TEMPLATE = """ NOTICE Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and Incremental Change, ensuring the integration of new features. @@ -161,7 +158,7 @@ Role: You are a professional engineer; The main goal is to complete incremental """ CODE_GUIDE_CONTEXT_EXAMPLE = """ -### Legacy Code +# Legacy Code ## main.py from flask import Flask, request, jsonify @@ -194,21 +191,17 @@ class Calculator: GUIDE_NODES = [INCREMENTAL_CHANGE] -WRITE_CODE_GUIDE_NODE = ActionNode.from_children("WriteCodeGuide", GUIDE_NODES) +WRITE_CODE_GUIDELINE_NODE = ActionNode.from_children("WriteCodeGuideline", GUIDE_NODES) -class WriteCodeGuide(Action): - name: str = "WriteCodeGuide" - context: Document = Field(default_factory=Document) - llm: BaseGPTAPI = Field(default_factory=LLM) - +class WriteCodeGuideline(Action): async def run(self, context): - return await WRITE_CODE_GUIDE_NODE.fill(context=context, llm=self.llm, schema="json") + return await WRITE_CODE_GUIDELINE_NODE.fill(context=context, llm=self.llm, schema="json") def main(): - action = WriteCodeGuide() - return asyncio.run(action.run(CODE_GUIDE_CONTEXT)) + action = WriteCodeGuideline() + return asyncio.run(action.run(CODE_GUIDELINE_CONTEXT)) if __name__ == "__main__": diff --git a/metagpt/actions/write_prd.py b/metagpt/actions/write_prd.py index dc1af0735..a070e7a96 100644 --- a/metagpt/actions/write_prd.py +++ b/metagpt/actions/write_prd.py @@ -24,7 +24,7 @@ from metagpt.actions.action_node import ActionNode from metagpt.actions.fix_bug import FixBug from metagpt.actions.write_prd_an import ( REFINE_PRD_NODE, - REFINE_PRD_SIMPLE_CONTEXT, + REFINE_PRD_TEMPLATE, WP_IS_RELATIVE_NODE, WP_ISSUE_TYPE_NODE, WRITE_PRD_NODE, @@ -139,7 +139,7 @@ class WritePRD(Action): CONFIG.project_name = Path(CONFIG.project_path).name project_name = CONFIG.project_name if CONFIG.project_name else "" - prompt = REFINE_PRD_SIMPLE_CONTEXT.format( + prompt = REFINE_PRD_TEMPLATE.format( requirements=new_requirement_doc.content, old_prd=prd_doc.content, project_name=project_name, diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index 79046bb7d..f645225c7 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -31,8 +31,8 @@ ORIGINAL_REQUIREMENTS = ActionNode( example="Create a 2048 game", ) -REFINE_REQUIREMENTS = ActionNode( - key="Original Requirements", +REFINED_REQUIREMENTS = ActionNode( + key="Refined Requirements", expected_type=str, instruction="Update and expand the original user's requirements to reflect the evolving needs of the project." "Retain any content unrelated to incremental development", @@ -46,13 +46,6 @@ PROJECT_NAME = ActionNode( example="game_2048", ) -REFINE_PROJECT_NAME = ActionNode( - key="Project Name", - expected_type=str, - instruction="Update the project name based on the context.", - example="game_2048_new", -) - PRODUCT_GOALS = ActionNode( key="Product Goals", expected_type=List[str], @@ -60,8 +53,8 @@ PRODUCT_GOALS = ActionNode( example=["Create an engaging user experience", "Improve accessibility, be responsive", "More beautiful UI"], ) -REFINE_PRODUCT_GOALS = ActionNode( - key="Product Goals", +REFINED_PRODUCT_GOALS = ActionNode( + key="Refined Product Goals", expected_type=List[str], instruction="Update and expand the original product goals to reflect the evolving needs due to incremental " "development.Ensure that the refined goals align with the current project direction and contribute to its success." @@ -86,11 +79,11 @@ USER_STORIES = ActionNode( ], ) -REFINE_USER_STORIES = ActionNode( - key="User Stories", +REFINED_USER_STORIES = ActionNode( + key="Refined User Stories", expected_type=List[str], instruction="Update and expand the original scenario-based user stories to reflect the evolving needs due to " - "incremental development, no less than 7. Ensure that the refined user stories capture incremental features and " + "incremental development, no less than 5. Ensure that the refined user stories capture incremental features and " "improvements. Retain any content unrelated to incremental development", example=[ "As a player, I want to choose difficulty levels to challenge my skills", @@ -140,6 +133,14 @@ REQUIREMENT_ANALYSIS = ActionNode( example="", ) +INCREMENTAL_REQUIREMENT_ANALYSIS = ActionNode( + key="Incremental Requirement Analysis", + expected_type=List[str], + instruction="Propose the comprehensive incremental development requirement analysis on new features and enhanced " + "features for New Requirements.", + example=["Require add/update/modify ..."], +) + REQUIREMENT_POOL = ActionNode( key="Requirement Pool", expected_type=List[List[str]], @@ -147,8 +148,8 @@ REQUIREMENT_POOL = ActionNode( example=[["P0", "The main code ..."], ["P0", "The game algorithm ..."]], ) -REFINE_REQUIREMENT_POOL = ActionNode( - key="Requirement Pool", +REFINED_REQUIREMENT_POOL = ActionNode( + key="Refined Requirement Pool", expected_type=List[List[str]], instruction="List no less than 7 requirements with their priority (P0, P1, P2). " "Cover both legacy content and incremental content. Retain any content unrelated to incremental development", @@ -192,25 +193,18 @@ REFINE_PRD_CONTEXT = """ Role: You are a professional Product Manager tasked with overseeing incremental development. Based on New Requirements, output a New PRD that seamlessly integrates both the Legacy Content and the Incremental Content. Ensure the resulting document captures the complete scope of features, enhancements, and retain content unrelated to incremental development needs for coherence and clarity. -### New Project Name -{project_name} - -### New Requirements +# Context +## New Requirements {requirements} -### Legacy Content +## Legacy Content {old_prd} -### PRD Incremental Content +## PRD Incremental Content {prd_increment} - -### Search Information -- """ -REFINE_PRD_SIMPLE_CONTEXT = """ -You are a professional Product Manager tasked with overseeing incremental development. - +REFINE_PRD_TEMPLATE = """ ### New Project Name {project_name} @@ -224,13 +218,6 @@ You are a professional Product Manager tasked with overseeing incremental develo - """ -INCREMENTAL_DEVELOPMENT_ANALYSIS = ActionNode( - key="Requirement Analysis", - expected_type=List[str], - instruction="Propose the comprehensive incremental development requirement analysis on new features and enhanced features for New Requirements.", - example=["Require add/update/modify ..."], -) - NODES = [ LANGUAGE, @@ -250,30 +237,30 @@ NODES = [ REFINE_NODES = [ LANGUAGE, PROGRAMMING_LANGUAGE, - REFINE_REQUIREMENTS, - REFINE_PROJECT_NAME, - REFINE_PRODUCT_GOALS, - REFINE_USER_STORIES, + REFINED_REQUIREMENTS, + PROJECT_NAME, + REFINED_PRODUCT_GOALS, + REFINED_USER_STORIES, COMPETITIVE_ANALYSIS, COMPETITIVE_QUADRANT_CHART, - INCREMENTAL_DEVELOPMENT_ANALYSIS, - REFINE_REQUIREMENT_POOL, + INCREMENTAL_REQUIREMENT_ANALYSIS, + REFINED_REQUIREMENT_POOL, UI_DESIGN_DRAFT, ANYTHING_UNCLEAR, ] -INCREMENT_PRD_NODES = [INCREMENTAL_DEVELOPMENT_ANALYSIS, REQUIREMENT_POOL] +INCREMENT_PRD_NODES = [INCREMENTAL_REQUIREMENT_ANALYSIS, REQUIREMENT_POOL] WRITE_PRD_NODE = ActionNode.from_children("WritePRD", NODES) REFINE_PRD_NODE = ActionNode.from_children("RefinePRD", REFINE_NODES) -INCREMENT_NODE = ActionNode.from_children("IncrementPRD", INCREMENT_PRD_NODES) +INCREMENTAL_PRD_NODE = ActionNode.from_children("IncrementalPRD", INCREMENT_PRD_NODES) WP_ISSUE_TYPE_NODE = ActionNode.from_children("WP_ISSUE_TYPE", [ISSUE_TYPE, REASON]) WP_IS_RELATIVE_NODE = ActionNode.from_children("WP_IS_RELATIVE", [IS_RELATIVE, REASON]) def main(): # prompt = WRITE_PRD_NODE.compile(context="") - prompt = INCREMENT_NODE.compile(context=REFINE_PRD_CONTEXT) + prompt = INCREMENTAL_PRD_NODE.compile(context=REFINE_PRD_CONTEXT) logger.info(prompt) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 4aa8f209d..fe2d369cb 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -28,7 +28,10 @@ from typing import Set from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks from metagpt.actions.fix_bug import FixBug from metagpt.actions.summarize_code import SummarizeCode -from metagpt.actions.write_code_guide_an import CODE_GUIDE_CONTEXT, WriteCodeGuide +from metagpt.actions.write_code_guideline_an import ( + CODE_GUIDELINE_CONTEXT, + WriteCodeGuideline, +) from metagpt.config import CONFIG from metagpt.const import ( CODE_SUMMARIES_FILE_REPO, @@ -91,7 +94,7 @@ class Engineer(Role): @staticmethod def _parse_tasks(task_msg: Document) -> list[str]: m = json.loads(task_msg.content) - return m.get("Task list") + return m.get("Task list") or m.get("Refined Task list") async def _act_sp_with_cr(self, review=False, guideline="") -> Set[str]: changed_files = set() @@ -323,8 +326,8 @@ class Engineer(Role): tasks = "\n".join([doc.content for doc in tasks]) old_codes = await self.get_old_codes() - context = CODE_GUIDE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) - node = await WriteCodeGuide().run(context=context) + context = CODE_GUIDELINE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) + node = await WriteCodeGuideline().run(context=context) guideline = node.instruct_content.json(ensure_ascii=False) return guideline diff --git a/metagpt/utils/mermaid.py b/metagpt/utils/mermaid.py index d1cd1b328..76adbf5a7 100644 --- a/metagpt/utils/mermaid.py +++ b/metagpt/utils/mermaid.py @@ -120,7 +120,7 @@ MMC1 = """classDiagram SearchEngine --> Summary Index --> KnowledgeBase""" -MMC1_INC_AND_REFINE = """classDiagram +MMC1_REFINE = """classDiagram class Main { -SearchEngine search_engine +main() str @@ -176,7 +176,7 @@ MMC2 = """sequenceDiagram S-->>SE: return summary SE-->>M: return summary""" -MMC2_INC = """sequenceDiagram +MMC2_REFINE = """sequenceDiagram participant M as Main participant SE as SearchEngine participant I as Index From b19995f083288863b8f02c474da4fca63e265665 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 3 Jan 2024 14:19:14 +0800 Subject: [PATCH 252/637] Added some test cases --- metagpt/actions/design_api_an.py | 7 +- metagpt/actions/project_management_an.py | 4 +- metagpt/actions/write_code.py | 4 +- metagpt/actions/write_code_guideline_an.py | 328 ++++++++++++++++-- metagpt/actions/write_prd_an.py | 20 +- tests/conftest.py | 3 +- tests/metagpt/actions/test_design_api_an.py | 102 ++++++ .../actions/test_project_management_an.py | 141 ++++++++ .../actions/test_write_code_guideline_an.py | 112 ++++++ tests/metagpt/actions/test_write_prd_an.py | 85 +++++ 10 files changed, 757 insertions(+), 49 deletions(-) create mode 100644 tests/metagpt/actions/test_design_api_an.py create mode 100644 tests/metagpt/actions/test_project_management_an.py create mode 100644 tests/metagpt/actions/test_write_code_guideline_an.py create mode 100644 tests/metagpt/actions/test_write_prd_an.py diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index fcfaff9c3..bcfaf0bfb 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -120,7 +120,7 @@ INC_DESIGN_CONTEXT = """ {prd_increment} """ -REFINE_DESIGN_CONTEXT = """ +MERGE_DESIGN_CONTEXT = """ Role: You are a professional Architect tasked with overseeing incremental development. Based on new requirements, review and refine the system design. Integrate existing architecture with incremental design changes, ensuring the refined design encompasses all architectural elements, enhancements, and adjustments. Retain content unrelated to incremental development needs for coherence and clarity. @@ -148,7 +148,6 @@ INC_NODES = [INCREMENTAL_IMPLEMENTATION_APPROACH, INCREMENTAL_DATA_STRUCTURES_AN REFINE_NODES = [ REFINED_IMPLEMENTATION_APPROACH, - # PROJECT_NAME, FILE_LIST, REFINED_DATA_STRUCTURES_AND_INTERFACES, REFINED_PROGRAM_CALL_FLOW, @@ -161,7 +160,9 @@ REFINED_DESIGN_NODES = ActionNode.from_children("Refined_Design_API", REFINE_NOD def main(): - prompt = REFINED_DESIGN_NODES.compile(context="...") + prompt = DESIGN_API_NODE.compile(context="") + logger.info(prompt) + prompt = REFINED_DESIGN_NODES.compile(context="") logger.info(prompt) diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index 467d26b4e..a970c05a3 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -138,7 +138,7 @@ INC_PM_CONTEXT = """ {design_increment} """ -REFINE_PM_CONTEXT = """ +MERGE_PM_CONTEXT = """ Role: You are a professional Project Manager tasked with overseeing incremental development. Based on New Requirements, refine the project context to account for incremental development. Ensure the context offers a comprehensive overview of the project's evolving scope, covering both legacy content and incremental content. Retain any content unrelated to incremental development. @@ -183,6 +183,8 @@ REFINED_PM_NODES = ActionNode.from_children("Refined_PM_NODES", REFINE_NODES) def main(): prompt = PM_NODE.compile(context="") logger.info(prompt) + prompt = REFINED_PM_NODES.compile(context="") + logger.info(prompt) if __name__ == "__main__": diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 4183db19d..0ff43905c 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -21,7 +21,7 @@ from pydantic import Field from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions.action import Action -from metagpt.actions.write_code_guideline_an import REFINED_TEMPLATE +from metagpt.actions.write_code_guideline_an import REFINED_CODE_TEMPLATE from metagpt.config import CONFIG from metagpt.const import ( BUGFIX_FILENAME, @@ -127,7 +127,7 @@ class WriteCode(Action): code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename) if guideline: - prompt = REFINED_TEMPLATE.format( + prompt = REFINED_CODE_TEMPLATE.format( requirement=requirement_doc.content if requirement_doc else "", guideline=guideline, design=coding_context.design_doc.content if coding_context.design_doc else "", diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index d68b02d38..ee6dc81b7 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -103,7 +103,291 @@ Role: You are a professional software engineer, and your main task is to craft c {code} """ -REFINED_TEMPLATE = """ +CODE_GUIDELINE_CONTEXT_EXAMPLE = """ +NOTICE +Role: You are a professional software engineer, and your main task is to craft comprehensive incremental development plans and provide detailed code guidance with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". +1. Determine the scope of responsibilities of each file and what classes and methods need to be implemented. +2. Import all referenced classes. +3. Implement all methods. +4. Add necessary explanation to all methods. +5. Ensure there are no potential bugs. +6. Confirm that the entire project conforms to the tasks proposed by the user. +7. Examine the code closely to find and fix errors, and confirm that the logic is sound to ensure smooth user interaction while meeting all specified requirements. +8. Attention: Code files in the task list may have a different number of files compared to legacy code files. This requires integrating legacy code files that do not appear in the task list into the code files of the task list. Therefore, when writing code guidance and incremental changes for the code files in the task list, also include how to seamlessly merge and adjust legacy code files. + +# Context +## Requirement +Add subtraction, multiplication and division operations to the calculator. +The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator + +## Design +{ + "Refined Implementation Approach": "To accommodate the new requirements, we will extend the existing Python-based calculator application. We will enhance the Tkinter-based UI to include buttons for subtraction, multiplication, and division, alongside the existing addition functionality. We will also implement input validation to handle edge cases such as division by zero. The architecture will be modular, with separate components for the UI, calculation logic, and error handling to maintain simplicity and facilitate future enhancements such as a history feature.", + "File list": [ + "main.py", + "calculator.py", + "interface.py", + "operations.py" + ], + "Refined Data Structures and Interfaces": "classDiagram\n class CalculatorApp {\n +main() None\n }\n class Calculator {\n -result float\n +add(number1: float, number2: float) float\n +subtract(number1: float, number2: float) float\n +multiply(number1: float, number2: float) float\n +divide(number1: float, number2: float) float\n +clear() None\n }\n class Interface {\n -calculator Calculator\n +start() None\n +display_result(result: float) None\n +get_input() float\n +show_error(message: str) None\n +update_operation(operation: str) None\n }\n class Operations {\n +perform_operation(operation: str, number1: float, number2: float) float\n }\n CalculatorApp --> Interface\n Interface --> Calculator\n Calculator --> Operations", + "Refined Program call flow": "sequenceDiagram\n participant CA as CalculatorApp\n participant I as Interface\n participant C as Calculator\n participant O as Operations\n CA->>I: start()\n I->>I: get_input()\n I->>I: update_operation(operation)\n loop For Each Operation\n I->>C: perform_operation(operation, number1, number2)\n C->>O: perform_operation(operation, number1, number2)\n O-->>C: return result\n C-->>I: return result\n I->>I: display_result(result)\n end\n I->>I: show_error(message)", + "Anything UNCLEAR": "The requirement for a history feature is mentioned but not prioritized. It is unclear whether this should be implemented now or in the future. Additionally, there is no specification on the limit to the size of the numbers or the number of operations that can be performed in sequence. These aspects will need clarification for complete implementation." +} + +## Tasks +{ + "Required Python packages": [ + "tkinter" + ], + "Required Other language third-party packages": [ + "No third-party dependencies required" + ], + "Refined Logic Analysis": [ + [ + "main.py", + "Entry point of the application, creates an instance of the Interface class and starts the application." + ], + [ + "calculator.py", + "Contains the Calculator class with add, subtract, multiply, divide and clear methods for performing arithmetic operations." + ], + [ + "interface.py", + "Contains the Interface class responsible for the GUI, interacts with Calculator for the logic and displays results or errors." + ], + [ + "operations.py", + "Contains the Operations class with perform_operation method that delegates the arithmetic operation based on the operation argument." + ] + ], + "Refined Task list": [ + "operations.py", + "calculator.py", + "interface.py", + "main.py" + ], + "Full API spec": "", + "Refined Shared Knowledge": "`interface.py` will use the Calculator class from `calculator.py` to perform operations and display results. `main.py` will be the starting point that initializes the Interface. `calculator.py` will now also interact with `operations.py` to perform the arithmetic operations.", + "Anything UNCLEAR": "The requirement for a history feature is mentioned but not prioritized. It is unclear whether this should be implemented now or in the future. Additionally, there is no specification on the limit to the size of the numbers or the number of operations that can be performed in sequence. These aspects will need clarification for complete implementation." +} + +## Legacy Code +{code} +""" + +CODE_GUIDELINE_SCRIPT_EXAMPLE = """ +----- calculator.py +```## calculator.py + +class Calculator: + def __init__(self): + self.result = 0.0 # Default value for the result + + def add(self, number1: float, number2: float) -> float: + ''' + Adds two numbers and returns the result. + + Args: + number1 (float): The first number to add. + number2 (float): The second number to add. + + Returns: + float: The sum of number1 and number2. + ''' + self.result = number1 + number2 + return self.result + + def clear(self) -> None: + ''' + Clears the result to its default value. + ''' + self.result = 0.0 +``` + +---- interface.py +```## interface.py +import tkinter as tk +from calculator import Calculator + +class Interface: + def __init__(self): + self.calculator = Calculator() + self.root = tk.Tk() + self.root.title("Calculator") + self.create_widgets() + + def create_widgets(self): + self.result_var = tk.StringVar() + self.result_display = tk.Entry(self.root, textvariable=self.result_var, state='readonly', justify='right', font=('Arial', 24)) + self.result_display.grid(row=0, column=0, columnspan=4, sticky='nsew') + + self.entry_number1 = tk.Entry(self.root, justify='right', font=('Arial', 18)) + self.entry_number1.grid(row=1, column=0, columnspan=2, sticky='nsew') + + self.entry_number2 = tk.Entry(self.root, justify='right', font=('Arial', 18)) + self.entry_number2.grid(row=1, column=2, columnspan=2, sticky='nsew') + + self.add_button = tk.Button(self.root, text='+', command=self.add, font=('Arial', 18)) + self.add_button.grid(row=2, column=0, sticky='nsew') + + self.clear_button = tk.Button(self.root, text='C', command=self.clear, font=('Arial', 18)) + self.clear_button.grid(row=2, column=1, sticky='nsew') + + self.quit_button = tk.Button(self.root, text='Quit', command=self.root.quit, font=('Arial', 18)) + self.quit_button.grid(row=2, column=2, columnspan=2, sticky='nsew') + + self.root.grid_rowconfigure(1, weight=1) + self.root.grid_columnconfigure(0, weight=1) + + def start(self): + self.root.mainloop() + + def display_result(self, result: float): + self.result_var.set(str(result)) + + def get_input(self): + try: + number1 = float(self.entry_number1.get()) + number2 = float(self.entry_number2.get()) + return number1, number2 + except ValueError: + self.show_error("Invalid input! Please enter valid numbers.") + return None, None + + def add(self): + number1, number2 = self.get_input() + if number1 is not None and number2 is not None: + result = self.calculator.add(number1, number2) + self.display_result(result) + + def clear(self): + self.entry_number1.delete(0, tk.END) + self.entry_number2.delete(0, tk.END) + self.result_var.set("") + + def show_error(self, message: str): + tk.messagebox.showerror("Error", message) + +# This code is meant to be used as a module and not as a standalone script. +# The Interface class will be instantiated and started by the main.py file. +``` + +---- main.py +```## main.py +from interface import Interface + + +class CalculatorApp: + @staticmethod + def main(): + interface = Interface() + interface.start() + + +if __name__ == "__main__": + CalculatorApp.main() +``` +""" + +REFINE_CODE_SCRIPT_EXAMPLE = """ +----- calculator.py +```## calculator.py + +class Calculator: + def __init__(self): + self.result = 0.0 # Default value for the result + + def add(self, number1: float, number2: float) -> float: + ''' + Adds two numbers and returns the result. + + Args: + number1 (float): The first number to add. + number2 (float): The second number to add. + + Returns: + float: The sum of number1 and number2. + ''' + self.result = number1 + number2 + return self.result + + def clear(self) -> None: + ''' + Clears the result to its default value. + ''' + self.result = 0.0 +``` + +---- Now, interface.py to be rewritten +```## interface.py +import tkinter as tk +from calculator import Calculator + +class Interface: + def __init__(self): + self.calculator = Calculator() + self.root = tk.Tk() + self.root.title("Calculator") + self.create_widgets() + + def create_widgets(self): + self.result_var = tk.StringVar() + self.result_display = tk.Entry(self.root, textvariable=self.result_var, state='readonly', justify='right', font=('Arial', 24)) + self.result_display.grid(row=0, column=0, columnspan=4, sticky='nsew') + + self.entry_number1 = tk.Entry(self.root, justify='right', font=('Arial', 18)) + self.entry_number1.grid(row=1, column=0, columnspan=2, sticky='nsew') + + self.entry_number2 = tk.Entry(self.root, justify='right', font=('Arial', 18)) + self.entry_number2.grid(row=1, column=2, columnspan=2, sticky='nsew') + + self.add_button = tk.Button(self.root, text='+', command=self.add, font=('Arial', 18)) + self.add_button.grid(row=2, column=0, sticky='nsew') + + self.clear_button = tk.Button(self.root, text='C', command=self.clear, font=('Arial', 18)) + self.clear_button.grid(row=2, column=1, sticky='nsew') + + self.quit_button = tk.Button(self.root, text='Quit', command=self.root.quit, font=('Arial', 18)) + self.quit_button.grid(row=2, column=2, columnspan=2, sticky='nsew') + + self.root.grid_rowconfigure(1, weight=1) + self.root.grid_columnconfigure(0, weight=1) + + def start(self): + self.root.mainloop() + + def display_result(self, result: float): + self.result_var.set(str(result)) + + def get_input(self): + try: + number1 = float(self.entry_number1.get()) + number2 = float(self.entry_number2.get()) + return number1, number2 + except ValueError: + self.show_error("Invalid input! Please enter valid numbers.") + return None, None + + def add(self): + number1, number2 = self.get_input() + if number1 is not None and number2 is not None: + result = self.calculator.add(number1, number2) + self.display_result(result) + + def clear(self): + self.entry_number1.delete(0, tk.END) + self.entry_number2.delete(0, tk.END) + self.result_var.set("") + + def show_error(self, message: str): + tk.messagebox.showerror("Error", message) + +# This code is meant to be used as a module and not as a standalone script. +# The Interface class will be instantiated and started by the main.py file. +``` +""" + +REFINED_CODE_TEMPLATE = """ NOTICE Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and Incremental Change, ensuring the integration of new features. @@ -157,38 +441,6 @@ Role: You are a professional engineer; The main goal is to complete incremental 9. Attention: If Legacy Code files contain "{filename} to be rewritten", you are required to merge the Incremental Change into the {filename} file when rewriting "{filename} to be rewritten". """ -CODE_GUIDE_CONTEXT_EXAMPLE = """ -# Legacy Code -## main.py - -from flask import Flask, request, jsonify -from calculator import Calculator - -app = Flask(__name__) -calculator = Calculator() - -@app.route('/add_numbers', methods=['POST']) -def add_numbers(): - data = request.get_json() - num1 = data.get('num1', 0) - num2 = data.get('num2', 0) - result = calculator.add_numbers(num1, num2) - return jsonify({'result': result}), 200 - -if __name__ == '__main__': - app.run() - -## calculator.py - -class Calculator: - def __init__(self, num1: int = 0, num2: int = 0): - self.num1 = num1 - self.num2 = num2 - - def add_numbers(self, num1: int, num2: int) -> int: - return num1 + num2 -""" - GUIDE_NODES = [INCREMENTAL_CHANGE] WRITE_CODE_GUIDELINE_NODE = ActionNode.from_children("WriteCodeGuideline", GUIDE_NODES) @@ -199,10 +451,12 @@ class WriteCodeGuideline(Action): return await WRITE_CODE_GUIDELINE_NODE.fill(context=context, llm=self.llm, schema="json") -def main(): - action = WriteCodeGuideline() - return asyncio.run(action.run(CODE_GUIDELINE_CONTEXT)) +async def main(): + write_code_guideline = WriteCodeGuideline() + node = await write_code_guideline.run(CODE_GUIDELINE_CONTEXT_EXAMPLE.format(code=CODE_GUIDELINE_SCRIPT_EXAMPLE)) + guideline = node.instruct_content.json(ensure_ascii=False) + print(guideline) if __name__ == "__main__": - main() + asyncio.run(main()) diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index f645225c7..33008d3fe 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -141,6 +141,15 @@ INCREMENTAL_REQUIREMENT_ANALYSIS = ActionNode( example=["Require add/update/modify ..."], ) +REFINED_REQUIREMENT_ANALYSIS = ActionNode( + key="Refined Requirement Analysis", + expected_type=List[str], + instruction="Review and refine the existing requirement analysis to align with the evolving needs of the project " + "due to incremental development. Ensure the analysis comprehensively covers the new features and enhancements " + "required for the refined project scope.", + example=["Require add/update/modify ..."], +) + REQUIREMENT_POOL = ActionNode( key="Requirement Pool", expected_type=List[List[str]], @@ -151,7 +160,7 @@ REQUIREMENT_POOL = ActionNode( REFINED_REQUIREMENT_POOL = ActionNode( key="Refined Requirement Pool", expected_type=List[List[str]], - instruction="List no less than 7 requirements with their priority (P0, P1, P2). " + instruction="List no less than 5 requirements with their priority (P0, P1, P2). " "Cover both legacy content and incremental content. Retain any content unrelated to incremental development", example=[["P0", "The main code ..."], ["P0", "The game algorithm ..."]], ) @@ -189,7 +198,7 @@ REASON = ActionNode( ) -REFINE_PRD_CONTEXT = """ +INCREMENTAL_PRD_CONTEXT = """ Role: You are a professional Product Manager tasked with overseeing incremental development. Based on New Requirements, output a New PRD that seamlessly integrates both the Legacy Content and the Incremental Content. Ensure the resulting document captures the complete scope of features, enhancements, and retain content unrelated to incremental development needs for coherence and clarity. @@ -243,7 +252,7 @@ REFINE_NODES = [ REFINED_USER_STORIES, COMPETITIVE_ANALYSIS, COMPETITIVE_QUADRANT_CHART, - INCREMENTAL_REQUIREMENT_ANALYSIS, + REFINED_REQUIREMENT_ANALYSIS, REFINED_REQUIREMENT_POOL, UI_DESIGN_DRAFT, ANYTHING_UNCLEAR, @@ -259,8 +268,9 @@ WP_IS_RELATIVE_NODE = ActionNode.from_children("WP_IS_RELATIVE", [IS_RELATIVE, R def main(): - # prompt = WRITE_PRD_NODE.compile(context="") - prompt = INCREMENTAL_PRD_NODE.compile(context=REFINE_PRD_CONTEXT) + prompt = WRITE_PRD_NODE.compile(context="") + logger.info(prompt) + prompt = REFINE_PRD_NODE.compile(context="") logger.info(prompt) diff --git a/tests/conftest.py b/tests/conftest.py index b22e43e79..50fbf556c 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -86,7 +86,8 @@ def loguru_caplog(caplog): # init & dispose git repo -@pytest.fixture(scope="session", autouse=True) +# @pytest.fixture(scope="session", autouse=True) +@pytest.fixture(scope="session") def setup_and_teardown_git_repo(request): CONFIG.git_repo = GitRepository(local_path=DEFAULT_WORKSPACE_ROOT / "unittest") diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py new file mode 100644 index 000000000..e50288640 --- /dev/null +++ b/tests/metagpt/actions/test_design_api_an.py @@ -0,0 +1,102 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/03 +@Author : mannaandpoem +@File : test_design_api_an.py.py +""" +import pytest + +from metagpt.actions.design_api_an import REFINED_DESIGN_NODES +from metagpt.provider import OpenAIGPTAPI + +CONTEXT = """ +### Legacy Content +{ + "Implementation approach": "We will use Python with the Pygame library to develop the core mechanics of the 2048 game. The game will feature a simple and intuitive user interface, score tracking with high score memory, and an undo move feature. We'll ensure the game has visually appealing graphics and animations while maintaining a minimalist design. The undo feature will allow a single move to be undone without affecting the score, to keep the implementation straightforward.", + "File list": [ + "main.py", + "game.py", + "ui.py", + "constants.py" + ], + "Data structures and interfaces": "classDiagram\n class Main {\n +pygame: PygameInstance\n +game: Game\n +run() void\n }\n class Game {\n -grid: list\n -current_score: int\n -high_score: int\n -last_move: list\n +move(direction: str) bool\n +undo() bool\n +check_game_over() bool\n +reset_game() void\n }\n class UI {\n -screen: PygameSurface\n -font: PygameFont\n +draw_grid(grid: list) void\n +display_score(current_score: int, high_score: int) void\n +show_game_over() void\n +show_undo_button() void\n }\n class Constants {\n +GRID_SIZE: int\n +WINDOW_WIDTH: int\n +WINDOW_HEIGHT: int\n +BACKGROUND_COLOR: tuple\n +TILE_COLORS: dict\n }\n Main --> Game\n Main --> UI\n Game --> Constants\n UI --> Constants", + "Program call flow": "sequenceDiagram\n participant M as Main\n participant G as Game\n participant U as UI\n M->>G: create instance\n M->>U: create instance\n loop game loop\n M->>U: draw_grid(G.grid)\n M->>U: display_score(G.current_score, G.high_score)\n M->>U: show_undo_button()\n M->>G: move(direction)\n alt if move is valid\n G-->>M: return true\n else if move is invalid\n G-->>M: return false\n end\n alt if undo is triggered\n M->>G: undo()\n G-->>M: return true\n else no undo\n G-->>M: return false\n end\n alt if game over\n M->>U: show_game_over()\n M->>G: reset_game()\n end\n end", + "Anything UNCLEAR": "The specifics of the scoring system and how the high score is stored and retrieved need to be clarified. Additionally, the exact graphical assets and animations for the game are not specified." +} + +### New Requirements +{ + "Language": "en_us", + "Programming Language": "Python", + "Refined Requirements": "Update the py2048_game to have a larger 8x8 grid and a new winning score target of 4096, while maintaining the core mechanics and user-friendly interface of the original 2048 game.", + "Project Name": "py2048_game", + "Refined Product Goals": [ + "Develop an enhanced version of the 2048 game with a larger grid and higher score target to provide a new challenge to players", + "Ensure the game remains visually appealing and maintains a consistent theme with the added complexity", + "Implement smooth and responsive game controls suitable for an 8x8 grid interface" + ], + "Refined User Stories": [ + "As a player, I want to experience a clear and simple interface on an 8x8 grid so that I can focus on the gameplay", + "As a player, I want to aim for a higher score target of 4096 to challenge my skills further", + "As a player, I want to see my current and high scores to track my progress on the new larger grid", + "As a player, I want the option to undo my last move to improve my strategy on the 8x8 grid", + "As a player, I want the game to perform smoothly despite the increased complexity of the larger grid" + ], + "Competitive Analysis": [ + "2048 Original: Classic gameplay with minimalistic design, but lacks modern features", + "2048 by Gabriele Cirulli: Open-source version with clean UI, but no additional features", + "2048 Hex: Unique hexagon board, providing a different challenge", + "2048 Multiplayer: Allows playing against others, but the interface is cluttered", + "2048 with AI: Includes AI challenge mode, but the AI is often too difficult for casual players", + "2048.io: Combines 2048 gameplay with .io style, though it can be overwhelming for new players", + "2048 Animated: Features animations, but has performance issues on some devices" + ], + "Competitive Quadrant Chart": "quadrantChart\n title \"2048 Game Market Positioning\"\n x-axis \"Basic Features\" --> \"Advanced Features\"\n y-axis \"Low User Engagement\" --> \"High User Engagement\"\n quadrant-1 \"Niche Innovators\"\n quadrant-2 \"Market Leaders\"\n quadrant-3 \"Emerging Contenders\"\n quadrant-4 \"Falling Behind\"\n \"2048 Original\": [0.2, 0.7]\n \"2048 by Gabriele Cirulli\": [0.3, 0.8]\n \"2048 Hex\": [0.5, 0.4]\n \"2048 Multiplayer\": [0.6, 0.6]\n \"2048 with AI\": [0.7, 0.5]\n \"2048.io\": [0.4, 0.3]\n \"2048 Animated\": [0.3, 0.2]\n \"Our Target Product\": [0.9, 0.9]", + "Incremental Requirement Analysis": [ + "Adjust the game logic to accommodate an 8x8 grid while ensuring performance remains optimal", + "Update the UI to fit the larger grid and include visual cues for the new score target", + "Enhance the scoring system to support the new target of 4096", + "Ensure the undo feature is adapted to work with the larger grid and increased game complexity", + "Test the game thoroughly to maintain a smooth and responsive experience on the new 8x8 grid" + ], + "Refined Requirement Pool": [ + [ + "P0", + "Expand the game grid to 8x8 and adjust the core mechanics accordingly" + ], + [ + "P0", + "Increase the score target to 4096 and update the scoring system" + ], + [ + "P1", + "Redesign the user interface to accommodate the larger grid size" + ], + [ + "P1", + "Ensure the undo move feature is compatible with the new grid and score target" + ], + [ + "P2", + "Optimize game performance for the increased complexity of an 8x8 grid" + ], + [ + "P2", + "Maintain a visually appealing and consistent theme with the updated game features" + ] + ], + "UI Design draft": "The UI will be updated to feature an 8x8 grid while maintaining a minimalist design. The main game screen will display the larger game grid, current score, high score, and an undo button. The color scheme and transitions will be adapted to ensure clarity and pleasant aesthetics despite the increased grid size.", + "Anything UNCLEAR": "The specifics of how the undo feature should work with the larger grid size and whether there should be any limitations on its use need to be clarified." +} +""" + + +llm = OpenAIGPTAPI() + + +@pytest.mark.asyncio +async def test_write_design_an(): + node = await REFINED_DESIGN_NODES.fill(CONTEXT, llm) + assert node.instruct_content + assert "Refined Data Structures and Interfaces" in node.instruct_content.json(ensure_ascii=False) diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py new file mode 100644 index 000000000..86c5e3685 --- /dev/null +++ b/tests/metagpt/actions/test_project_management_an.py @@ -0,0 +1,141 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/03 +@Author : mannaandpoem +@File : test_project_management_an.py.py +""" +import pytest + +from metagpt.actions.project_management_an import REFINED_PM_NODES +from metagpt.provider import OpenAIGPTAPI + +CONTEXT = """ +### Legacy Content +{ + "Required Python packages": [ + "pygame==2.0.1" + ], + "Required Other language third-party packages": [ + "No third-party dependencies required" + ], + "Logic Analysis": [ + [ + "constants.py", + "Contains all the constants like GRID_SIZE, WINDOW_WIDTH, WINDOW_HEIGHT, BACKGROUND_COLOR, TILE_COLORS" + ], + [ + "game.py", + "Contains Game class with methods for game logic such as move, undo, check_game_over, and reset_game" + ], + [ + "ui.py", + "Contains UI class responsible for drawing the grid, displaying scores, showing game over, and undo button" + ], + [ + "main.py", + "Contains Main class which initializes the game loop and orchestrates the interactions between Game and UI classes" + ] + ], + "Task list": [ + "constants.py", + "game.py", + "ui.py", + "main.py" + ], + "Full API spec": "", + "Shared Knowledge": "`constants.py` contains constants shared across `game.py` and `ui.py`. The Main class in `main.py` acts as the controller orchestrating the game flow and UI updates.", + "Anything UNCLEAR": "The specifics of the scoring system and how the high score is stored and retrieved need to be clarified. Additionally, the exact graphical assets and animations for the game are not specified." +} + +### New Requirements +{ + "Refined Implementation Approach": "We will refine our implementation approach to accommodate the new 8x8 grid and the increased winning score target of 4096. This will involve optimizing the game's core logic to handle the larger grid size efficiently and updating the scoring system. We will also enhance the UI to ensure it remains user-friendly and visually appealing with the new grid. The undo feature will be adapted to work seamlessly with the increased complexity of the game.", + "File list": [ + "main.py", + "game.py", + "ui.py", + "constants.py" + ], + "Refined Data Structures and Interfaces": "classDiagram + class Main { + +pygame: PygameInstance + +game: Game + +ui: UI + +run() void + } + class Game { + -grid: list + -current_score: int + -high_score: int + -last_move: list + -target_score: int + +__init__(grid_size: int, target_score: int) + +move(direction: str) bool + +undo() bool + +check_game_over() bool + +reset_game() void + +update_score(value: int) void + } + class UI { + -screen: PygameSurface + -font: PygameFont + +__init__(screen_size: tuple, grid_size: int) + +draw_grid(grid: list) void + +display_score(current_score: int, high_score: int) void + +show_game_over() void + +show_undo_button() void + +update_ui_for_larger_grid() void + } + class Constants { + +GRID_SIZE: int = 8 + +TARGET_SCORE: int = 4096 + +WINDOW_WIDTH: int + +WINDOW_HEIGHT: int + +BACKGROUND_COLOR: tuple + +TILE_COLORS: dict + } + Main --> Game + Main --> UI + Game --> Constants + UI --> Constants", + "Refined Program call flow": "sequenceDiagram + participant M as Main + participant G as Game + participant U as UI + M->>G: create instance(grid_size: Constants.GRID_SIZE, target_score: Constants.TARGET_SCORE) + M->>U: create instance(screen_size: (Constants.WINDOW_WIDTH, Constants.WINDOW_HEIGHT), grid_size: Constants.GRID_SIZE) + loop game loop + M->>U: draw_grid(G.grid) + M->>U: display_score(G.current_score, G.high_score) + M->>U: show_undo_button() + M->>G: move(direction) + alt if move is valid + G-->>M: return true + M->>G: update_score(value) + else if move is invalid + G-->>M: return false + end + alt if undo is triggered + M->>G: undo() + G-->>M: return true + else no undo + G-->>M: return false + end + alt if game over + M->>U: show_game_over() + M->>G: reset_game() + end + end", + "Anything UNCLEAR": "It remains unclear if the undo feature should have limitations on its use, such as a maximum number of undos per game or if it should be available without restriction. Further clarification on this aspect would be beneficial." +} +""" + +llm = OpenAIGPTAPI() + + +@pytest.mark.asyncio +async def test_project_management_an(): + node = await REFINED_PM_NODES.fill(CONTEXT, llm) + assert node.instruct_content + assert "Refined Logic Analysis" in node.instruct_content.json(ensure_ascii=False) diff --git a/tests/metagpt/actions/test_write_code_guideline_an.py b/tests/metagpt/actions/test_write_code_guideline_an.py new file mode 100644 index 000000000..c9fb78e0e --- /dev/null +++ b/tests/metagpt/actions/test_write_code_guideline_an.py @@ -0,0 +1,112 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/03 +@Author : mannaandpoem +@File : test_write_code_guideline_an.py.py +""" +import pytest + +from metagpt.actions import WriteCode +from metagpt.actions.write_code_guideline_an import ( + CODE_GUIDELINE_CONTEXT, + CODE_GUIDELINE_SCRIPT_EXAMPLE, + REFINE_CODE_SCRIPT_EXAMPLE, + REFINED_CODE_TEMPLATE, + WriteCodeGuideline, +) +from metagpt.provider import OpenAIGPTAPI + +REQUIREMENT_EXAMPLE = """Add subtraction, multiplication and division operations to the calculator. +The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator +""" + +DESIGN_EXAMPLE = """ +{ + "Refined Implementation Approach": "To accommodate the new requirements, we will extend the existing Python-based calculator application. We will enhance the Tkinter-based UI to include buttons for subtraction, multiplication, and division, alongside the existing addition functionality. We will also implement input validation to handle edge cases such as division by zero. The architecture will be modular, with separate components for the UI, calculation logic, and error handling to maintain simplicity and facilitate future enhancements such as a history feature.", + "File list": [ + "main.py", + "calculator.py", + "interface.py", + "operations.py" + ], + "Refined Data Structures and Interfaces": "classDiagram\n class CalculatorApp {\n +main() None\n }\n class Calculator {\n -result float\n +add(number1: float, number2: float) float\n +subtract(number1: float, number2: float) float\n +multiply(number1: float, number2: float) float\n +divide(number1: float, number2: float) float\n +clear() None\n }\n class Interface {\n -calculator Calculator\n +start() None\n +display_result(result: float) None\n +get_input() float\n +show_error(message: str) None\n +update_operation(operation: str) None\n }\n class Operations {\n +perform_operation(operation: str, number1: float, number2: float) float\n }\n CalculatorApp --> Interface\n Interface --> Calculator\n Calculator --> Operations", + "Refined Program call flow": "sequenceDiagram\n participant CA as CalculatorApp\n participant I as Interface\n participant C as Calculator\n participant O as Operations\n CA->>I: start()\n I->>I: get_input()\n I->>I: update_operation(operation)\n loop For Each Operation\n I->>C: perform_operation(operation, number1, number2)\n C->>O: perform_operation(operation, number1, number2)\n O-->>C: return result\n C-->>I: return result\n I->>I: display_result(result)\n end\n I->>I: show_error(message)", + "Anything UNCLEAR": "The requirement for a history feature is mentioned but not prioritized. It is unclear whether this should be implemented now or in the future. Additionally, there is no specification on the limit to the size of the numbers or the number of operations that can be performed in sequence. These aspects will need clarification for complete implementation." +} +""" + +TASKS_EXAMPLE = """ +{ + "Required Python packages": [ + "tkinter" + ], + "Required Other language third-party packages": [ + "No third-party dependencies required" + ], + "Refined Logic Analysis": [ + [ + "main.py", + "Entry point of the application, creates an instance of the Interface class and starts the application." + ], + [ + "calculator.py", + "Contains the Calculator class with add, subtract, multiply, divide and clear methods for performing arithmetic operations." + ], + [ + "interface.py", + "Contains the Interface class responsible for the GUI, interacts with Calculator for the logic and displays results or errors." + ], + [ + "operations.py", + "Contains the Operations class with perform_operation method that delegates the arithmetic operation based on the operation argument." + ] + ], + "Refined Task list": [ + "operations.py", + "calculator.py", + "interface.py", + "main.py" + ], + "Full API spec": "", + "Refined Shared Knowledge": "`interface.py` will use the Calculator class from `calculator.py` to perform operations and display results. `main.py` will be the starting point that initializes the Interface. `calculator.py` will now also interact with `operations.py` to perform the arithmetic operations.", + "Anything UNCLEAR": "The requirement for a history feature is mentioned but not prioritized. It is unclear whether this should be implemented now or in the future. Additionally, there is no specification on the limit to the size of the numbers or the number of operations that can be performed in sequence. These aspects will need clarification for complete implementation." +} +""" + +INCREMENTAL_CHANGE_EXAMPLE = """ +{ + "Incremental Change": "- operations.py: Implement the Operations class with a method to perform the requested arithmetic operation. This class will be used by the Calculator class to execute the operations.\n```python\n## operations.py\nclass Operations:\n @staticmethod\n def perform_operation(operation: str, number1: float, number2: float) -> float:\n if operation == 'add':\n return number1 + number2\n elif operation == 'subtract':\n return number1 - number2\n elif operation == 'multiply':\n return number1 * number2\n elif operation == 'divide':\n if number2 == 0:\n raise ValueError('Cannot divide by zero')\n return number1 / number2\n else:\n raise ValueError('Invalid operation')\n```\n\n- calculator.py: Extend the Calculator class to include methods for subtraction, multiplication, and division. These methods will utilize the Operations class to perform the actual calculations.\n```python\n## calculator.py\nfrom operations import Operations\nclass Calculator:\n ...\n def subtract(self, number1: float, number2: float) -> float:\n return Operations.perform_operation('subtract', number1, number2)\n\n def multiply(self, number1: float, number2: float) -> float:\n return Operations.perform_operation('multiply', number1, number2)\n\n def divide(self, number1: float, number2: float) -> float:\n return Operations.perform_operation('divide', number1, number2)\n```\n\n- interface.py: Update the Interface class to include buttons for subtraction, multiplication, and division, and link them to the corresponding methods in the Calculator class. Also, handle the display of errors such as division by zero.\n```python\n## interface.py\nimport tkinter as tk\nfrom tkinter import messagebox\nfrom calculator import Calculator\n...\nclass Interface:\n ...\n def create_widgets(self):\n ...\n self.subtract_button = tk.Button(self.root, text='-', command=self.subtract, font=('Arial', 18))\n self.subtract_button.grid(row=3, column=0, sticky='nsew')\n\n self.multiply_button = tk.Button(self.root, text='*', command=self.multiply, font=('Arial', 18))\n self.multiply_button.grid(row=3, column=1, sticky='nsew')\n\n self.divide_button = tk.Button(self.root, text='/', command=self.divide, font=('Arial', 18))\n self.divide_button.grid(row=3, column=2, sticky='nsew')\n ...\n\n def subtract(self):\n number1, number2 = self.get_input()\n if number1 is not None and number2 is not None:\n result = self.calculator.subtract(number1, number2)\n self.display_result(result)\n\n def multiply(self):\n number1, number2 = self.get_input()\n if number1 is not None and number2 is not None:\n result = self.calculator.multiply(number1, number2)\n self.display_result(result)\n\n def divide(self):\n number1, number2 = self.get_input()\n if number1 is not None and number2 is not None:\n try:\n result = self.calculator.divide(number1, number2)\n except ValueError as e:\n self.show_error(str(e))\n return\n self.display_result(result)\n```\n\n- main.py: No changes needed in main.py as it serves as the entry point and will run the updated Interface class.\n```python\n## main.py\nfrom interface import Interface\n...\n```\n\nNote: Ensure that the new operations buttons in the Interface class are properly arranged and that the grid layout is adjusted accordingly. Also, make sure to import the messagebox module from tkinter for error handling." +} +""" + +llm = OpenAIGPTAPI() + + +@pytest.mark.asyncio +async def test_write_code_guideline_an(): + write_code_guideline = WriteCodeGuideline() + context = CODE_GUIDELINE_CONTEXT.format( + requirement=REQUIREMENT_EXAMPLE, design=DESIGN_EXAMPLE, tasks=TASKS_EXAMPLE, code=CODE_GUIDELINE_SCRIPT_EXAMPLE + ) + node = await write_code_guideline.run(context=context) + assert node.instruct_content + assert "Incremental Change" in node.instruct_content.json(ensure_ascii=False) + + +@pytest.mark.asyncio +async def test_refine_code(): + prompt = REFINED_CODE_TEMPLATE.format( + requirement=REQUIREMENT_EXAMPLE, + guideline=INCREMENTAL_CHANGE_EXAMPLE, + design=DESIGN_EXAMPLE, + tasks=TASKS_EXAMPLE, + code=REFINE_CODE_SCRIPT_EXAMPLE, + logs="", + feedback="", + filename="interface.py", + summary_log="", + ) + code = await WriteCode().write_code(prompt=prompt) + assert code + assert "def create_widgets" in code diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py new file mode 100644 index 000000000..a7424bf49 --- /dev/null +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -0,0 +1,85 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/03 +@Author : mannaandpoem +@File : test_write_prd_an.py.py +""" +import pytest + +from metagpt.actions.write_prd_an import REFINE_PRD_NODE +from metagpt.provider import OpenAIGPTAPI + +CONTEXT = """ +### New Project Name +py2048_game + +### New Requirements +Changed score target for 2048 game from 2048 to 4096. +Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8. + +### Legacy Content +{ + "Language": "en_us", + "Programming Language": "Python", + "Original Requirements": "make a simple 2048 game based on pygame", + "Project Name": "pygame_2048", + "Product Goals": [ + "Develop a user-friendly and intuitive 2048 game", + "Ensure the game is visually appealing and maintains a consistent theme", + "Implement smooth and responsive game controls" + ], + "User Stories": [ + "As a player, I want to experience a clear and simple interface so that I can focus on the gameplay", + "As a player, I want to see my current and high scores to track my progress", + "As a player, I want the option to undo my last move to improve my strategy" + ], + "Competitive Analysis": [ + "2048 Original: Classic gameplay with minimalistic design, but lacks modern features", + "2048 by Gabriele Cirulli: Open-source version with clean UI, but no additional features", + "2048 Hex: Unique hexagon board, providing a different challenge", + "2048 Multiplayer: Allows playing against others, but the interface is cluttered", + "2048 with AI: Includes AI challenge mode, but the AI is often too difficult for casual players", + "2048.io: Combines 2048 gameplay with .io style, though it can be overwhelming for new players", + "2048 Animated: Features animations, but has performance issues on some devices" + ], + "Competitive Quadrant Chart": "quadrantChart\n title \"2048 Game Market Positioning\"\n x-axis \"Basic Features\" --> \"Advanced Features\"\n y-axis \"Low User Engagement\" --> \"High User Engagement\"\n quadrant-1 \"Niche Innovators\"\n quadrant-2 \"Market Leaders\"\n quadrant-3 \"Emerging Contenders\"\n quadrant-4 \"Falling Behind\"\n \"2048 Original\": [0.2, 0.7]\n \"2048 by Gabriele Cirulli\": [0.3, 0.8]\n \"2048 Hex\": [0.5, 0.4]\n \"2048 Multiplayer\": [0.6, 0.6]\n \"2048 with AI\": [0.7, 0.5]\n \"2048.io\": [0.4, 0.3]\n \"2048 Animated\": [0.3, 0.2]\n \"Our Target Product\": [0.8, 0.9]", + "Requirement Analysis": "The game should be simple yet engaging, with a focus on smooth performance and an intuitive user interface. High scores and undo functionality are important to users for a competitive and strategic gameplay experience. Aesthetic appeal and a consistent theme will also contribute to the game's success.", + "Requirement Pool": [ + [ + "P0", + "Develop core 2048 game mechanics using pygame" + ], + [ + "P0", + "Design a clean and intuitive user interface" + ], + [ + "P1", + "Implement score tracking with high score memory" + ], + [ + "P1", + "Add undo move feature for enhanced gameplay strategy" + ], + [ + "P2", + "Create visually appealing graphics and animations" + ] + ], + "UI Design draft": "The UI will feature a minimalist design with a focus on ease of use. The main game screen will display the game grid, current score, high score, and an undo button. The color scheme will be consistent and pleasant to the eye, with smooth transitions for tile movements.", + "Anything UNCLEAR": "The specifics of the undo feature need to be clarified, such as how many moves can be undone and whether it affects the scoring." +} + +### Search Information +- +""" + +llm = OpenAIGPTAPI() + + +@pytest.mark.asyncio +async def test_write_prd_an(): + node = await REFINE_PRD_NODE.fill(CONTEXT, llm) + assert node.instruct_content + assert "Refined Requirement Pool" in node.instruct_content.json(ensure_ascii=False) From 0ed2c56035b4e710d971a766c44c374165e375da Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 3 Jan 2024 14:41:49 +0800 Subject: [PATCH 253/637] Modify to make LLM a fixture in test cases --- tests/metagpt/actions/test_design_api_an.py | 4 +++- tests/metagpt/actions/test_project_management_an.py | 7 +++++-- tests/metagpt/actions/test_write_code_guideline_an.py | 5 +---- tests/metagpt/actions/test_write_prd_an.py | 7 +++++-- 4 files changed, 14 insertions(+), 9 deletions(-) diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index e50288640..32d6c11b0 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -92,7 +92,9 @@ CONTEXT = """ """ -llm = OpenAIGPTAPI() +@pytest.fixture() +def llm(): + return OpenAIGPTAPI() @pytest.mark.asyncio diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index 86c5e3685..116dffe83 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -131,11 +131,14 @@ CONTEXT = """ } """ -llm = OpenAIGPTAPI() + +@pytest.fixture() +def llm(): + return OpenAIGPTAPI() @pytest.mark.asyncio -async def test_project_management_an(): +async def test_project_management_an(llm): node = await REFINED_PM_NODES.fill(CONTEXT, llm) assert node.instruct_content assert "Refined Logic Analysis" in node.instruct_content.json(ensure_ascii=False) diff --git a/tests/metagpt/actions/test_write_code_guideline_an.py b/tests/metagpt/actions/test_write_code_guideline_an.py index c9fb78e0e..602145ce8 100644 --- a/tests/metagpt/actions/test_write_code_guideline_an.py +++ b/tests/metagpt/actions/test_write_code_guideline_an.py @@ -7,7 +7,7 @@ """ import pytest -from metagpt.actions import WriteCode +from metagpt.actions.write_code import WriteCode from metagpt.actions.write_code_guideline_an import ( CODE_GUIDELINE_CONTEXT, CODE_GUIDELINE_SCRIPT_EXAMPLE, @@ -15,7 +15,6 @@ from metagpt.actions.write_code_guideline_an import ( REFINED_CODE_TEMPLATE, WriteCodeGuideline, ) -from metagpt.provider import OpenAIGPTAPI REQUIREMENT_EXAMPLE = """Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator @@ -80,8 +79,6 @@ INCREMENTAL_CHANGE_EXAMPLE = """ } """ -llm = OpenAIGPTAPI() - @pytest.mark.asyncio async def test_write_code_guideline_an(): diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index a7424bf49..ef0dd2eff 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -75,11 +75,14 @@ Please change the game's score target from 2048 to 4096, and change the interfac - """ -llm = OpenAIGPTAPI() + +@pytest.fixture() +def llm(): + return OpenAIGPTAPI() @pytest.mark.asyncio -async def test_write_prd_an(): +async def test_write_prd_an(llm): node = await REFINE_PRD_NODE.fill(CONTEXT, llm) assert node.instruct_content assert "Refined Requirement Pool" in node.instruct_content.json(ensure_ascii=False) From 1c68d7f714199851f38599613efe6a5f4bcf9539 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 3 Jan 2024 15:02:54 +0800 Subject: [PATCH 254/637] Modify filename in comment --- metagpt/actions/write_code_guideline_an.py | 2 +- tests/metagpt/actions/test_design_api_an.py | 2 +- tests/metagpt/actions/test_project_management_an.py | 2 +- tests/metagpt/actions/test_write_code_guideline_an.py | 2 +- tests/metagpt/actions/test_write_prd_an.py | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index ee6dc81b7..55dc14d25 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -3,7 +3,7 @@ """ @Time : 2023/12/26 @Author : mannaandpoem -@File : write_code_guide_an.py +@File : write_code_guideline_an.py """ import asyncio diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index 32d6c11b0..305c708c0 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_design_api_an.py.py +@File : test_design_api_an.py """ import pytest diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index 116dffe83..9ad4360cf 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_project_management_an.py.py +@File : test_project_management_an.py """ import pytest diff --git a/tests/metagpt/actions/test_write_code_guideline_an.py b/tests/metagpt/actions/test_write_code_guideline_an.py index 602145ce8..d68b85c6f 100644 --- a/tests/metagpt/actions/test_write_code_guideline_an.py +++ b/tests/metagpt/actions/test_write_code_guideline_an.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_write_code_guideline_an.py.py +@File : test_write_code_guideline_an.py """ import pytest diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index ef0dd2eff..d520eb863 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_write_prd_an.py.py +@File : test_write_prd_an.py """ import pytest From 71d9fe31ab3364b3abbe145f26cf374e96b735e6 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 3 Jan 2024 15:02:54 +0800 Subject: [PATCH 255/637] Modify filename in comment and prompt in write_code_guideline_an.py --- metagpt/actions/write_code_guideline_an.py | 41 +------ tests/metagpt/actions/test_design_api_an.py | 2 +- .../actions/test_project_management_an.py | 2 +- .../actions/test_write_code_guideline_an.py | 116 +++++++++++++++++- tests/metagpt/actions/test_write_prd_an.py | 2 +- 5 files changed, 123 insertions(+), 40 deletions(-) diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index ee6dc81b7..0677f6edc 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -3,7 +3,7 @@ """ @Time : 2023/12/26 @Author : mannaandpoem -@File : write_code_guide_an.py +@File : write_code_guideline_an.py """ import asyncio @@ -78,19 +78,7 @@ if __name__ == '__main__': ) CODE_GUIDELINE_CONTEXT = """ -NOTICE -Role: You are a professional software engineer, and your main task is to craft comprehensive incremental development plans and provide detailed code guidance with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". -1. Determine the scope of responsibilities of each file and what classes and methods need to be implemented. -2. Import all referenced classes. -3. Implement all methods. -4. Add necessary explanation to all methods. -5. Ensure there are no potential bugs. -6. Confirm that the entire project conforms to the tasks proposed by the user. -7. Examine the code closely to find and fix errors, and confirm that the logic is sound to ensure smooth user interaction while meeting all specified requirements. -8. Attention: Code files in the task list may have a different number of files compared to legacy code files. This requires integrating legacy code files that do not appear in the task list into the code files of the task list. Therefore, when writing code guidance and incremental changes for the code files in the task list, also include how to seamlessly merge and adjust legacy code files. - -# Context -## Requirement +## New Requirements {requirement} ## Design @@ -104,19 +92,7 @@ Role: You are a professional software engineer, and your main task is to craft c """ CODE_GUIDELINE_CONTEXT_EXAMPLE = """ -NOTICE -Role: You are a professional software engineer, and your main task is to craft comprehensive incremental development plans and provide detailed code guidance with triple quote, based on the following attentions and context. Output format carefully referenced "Format example". -1. Determine the scope of responsibilities of each file and what classes and methods need to be implemented. -2. Import all referenced classes. -3. Implement all methods. -4. Add necessary explanation to all methods. -5. Ensure there are no potential bugs. -6. Confirm that the entire project conforms to the tasks proposed by the user. -7. Examine the code closely to find and fix errors, and confirm that the logic is sound to ensure smooth user interaction while meeting all specified requirements. -8. Attention: Code files in the task list may have a different number of files compared to legacy code files. This requires integrating legacy code files that do not appear in the task list into the code files of the task list. Therefore, when writing code guidance and incremental changes for the code files in the task list, also include how to seamlessly merge and adjust legacy code files. - -# Context -## Requirement +## New Requirements Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator @@ -172,10 +148,6 @@ The current calculator can only perform basic addition operations, and it is nec } ## Legacy Code -{code} -""" - -CODE_GUIDELINE_SCRIPT_EXAMPLE = """ ----- calculator.py ```## calculator.py @@ -381,9 +353,6 @@ class Interface: def show_error(self, message: str): tk.messagebox.showerror("Error", message) - -# This code is meant to be used as a module and not as a standalone script. -# The Interface class will be instantiated and started by the main.py file. ``` """ @@ -448,12 +417,14 @@ WRITE_CODE_GUIDELINE_NODE = ActionNode.from_children("WriteCodeGuideline", GUIDE class WriteCodeGuideline(Action): async def run(self, context): + self.llm.system_prompt = "You are a professional software engineer, your primary responsibility is to " + "meticulously craft comprehensive incremental development plans and deliver detailed Incremental Change" return await WRITE_CODE_GUIDELINE_NODE.fill(context=context, llm=self.llm, schema="json") async def main(): write_code_guideline = WriteCodeGuideline() - node = await write_code_guideline.run(CODE_GUIDELINE_CONTEXT_EXAMPLE.format(code=CODE_GUIDELINE_SCRIPT_EXAMPLE)) + node = await write_code_guideline.run(CODE_GUIDELINE_CONTEXT_EXAMPLE) guideline = node.instruct_content.json(ensure_ascii=False) print(guideline) diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index 32d6c11b0..305c708c0 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_design_api_an.py.py +@File : test_design_api_an.py """ import pytest diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index 116dffe83..9ad4360cf 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_project_management_an.py.py +@File : test_project_management_an.py """ import pytest diff --git a/tests/metagpt/actions/test_write_code_guideline_an.py b/tests/metagpt/actions/test_write_code_guideline_an.py index 602145ce8..d740a6bd8 100644 --- a/tests/metagpt/actions/test_write_code_guideline_an.py +++ b/tests/metagpt/actions/test_write_code_guideline_an.py @@ -3,14 +3,13 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_write_code_guideline_an.py.py +@File : test_write_code_guideline_an.py """ import pytest from metagpt.actions.write_code import WriteCode from metagpt.actions.write_code_guideline_an import ( CODE_GUIDELINE_CONTEXT, - CODE_GUIDELINE_SCRIPT_EXAMPLE, REFINE_CODE_SCRIPT_EXAMPLE, REFINED_CODE_TEMPLATE, WriteCodeGuideline, @@ -73,6 +72,119 @@ TASKS_EXAMPLE = """ } """ +CODE_GUIDELINE_SCRIPT_EXAMPLE = """ +----- calculator.py +```## calculator.py + +class Calculator: + def __init__(self): + self.result = 0.0 # Default value for the result + + def add(self, number1: float, number2: float) -> float: + ''' + Adds two numbers and returns the result. + + Args: + number1 (float): The first number to add. + number2 (float): The second number to add. + + Returns: + float: The sum of number1 and number2. + ''' + self.result = number1 + number2 + return self.result + + def clear(self) -> None: + ''' + Clears the result to its default value. + ''' + self.result = 0.0 +``` + +---- interface.py +```## interface.py +import tkinter as tk +from calculator import Calculator + +class Interface: + def __init__(self): + self.calculator = Calculator() + self.root = tk.Tk() + self.root.title("Calculator") + self.create_widgets() + + def create_widgets(self): + self.result_var = tk.StringVar() + self.result_display = tk.Entry(self.root, textvariable=self.result_var, state='readonly', justify='right', font=('Arial', 24)) + self.result_display.grid(row=0, column=0, columnspan=4, sticky='nsew') + + self.entry_number1 = tk.Entry(self.root, justify='right', font=('Arial', 18)) + self.entry_number1.grid(row=1, column=0, columnspan=2, sticky='nsew') + + self.entry_number2 = tk.Entry(self.root, justify='right', font=('Arial', 18)) + self.entry_number2.grid(row=1, column=2, columnspan=2, sticky='nsew') + + self.add_button = tk.Button(self.root, text='+', command=self.add, font=('Arial', 18)) + self.add_button.grid(row=2, column=0, sticky='nsew') + + self.clear_button = tk.Button(self.root, text='C', command=self.clear, font=('Arial', 18)) + self.clear_button.grid(row=2, column=1, sticky='nsew') + + self.quit_button = tk.Button(self.root, text='Quit', command=self.root.quit, font=('Arial', 18)) + self.quit_button.grid(row=2, column=2, columnspan=2, sticky='nsew') + + self.root.grid_rowconfigure(1, weight=1) + self.root.grid_columnconfigure(0, weight=1) + + def start(self): + self.root.mainloop() + + def display_result(self, result: float): + self.result_var.set(str(result)) + + def get_input(self): + try: + number1 = float(self.entry_number1.get()) + number2 = float(self.entry_number2.get()) + return number1, number2 + except ValueError: + self.show_error("Invalid input! Please enter valid numbers.") + return None, None + + def add(self): + number1, number2 = self.get_input() + if number1 is not None and number2 is not None: + result = self.calculator.add(number1, number2) + self.display_result(result) + + def clear(self): + self.entry_number1.delete(0, tk.END) + self.entry_number2.delete(0, tk.END) + self.result_var.set("") + + def show_error(self, message: str): + tk.messagebox.showerror("Error", message) + +# This code is meant to be used as a module and not as a standalone script. +# The Interface class will be instantiated and started by the main.py file. +``` + +---- main.py +```## main.py +from interface import Interface + + +class CalculatorApp: + @staticmethod + def main(): + interface = Interface() + interface.start() + + +if __name__ == "__main__": + CalculatorApp.main() +```""" + INCREMENTAL_CHANGE_EXAMPLE = """ { "Incremental Change": "- operations.py: Implement the Operations class with a method to perform the requested arithmetic operation. This class will be used by the Calculator class to execute the operations.\n```python\n## operations.py\nclass Operations:\n @staticmethod\n def perform_operation(operation: str, number1: float, number2: float) -> float:\n if operation == 'add':\n return number1 + number2\n elif operation == 'subtract':\n return number1 - number2\n elif operation == 'multiply':\n return number1 * number2\n elif operation == 'divide':\n if number2 == 0:\n raise ValueError('Cannot divide by zero')\n return number1 / number2\n else:\n raise ValueError('Invalid operation')\n```\n\n- calculator.py: Extend the Calculator class to include methods for subtraction, multiplication, and division. These methods will utilize the Operations class to perform the actual calculations.\n```python\n## calculator.py\nfrom operations import Operations\nclass Calculator:\n ...\n def subtract(self, number1: float, number2: float) -> float:\n return Operations.perform_operation('subtract', number1, number2)\n\n def multiply(self, number1: float, number2: float) -> float:\n return Operations.perform_operation('multiply', number1, number2)\n\n def divide(self, number1: float, number2: float) -> float:\n return Operations.perform_operation('divide', number1, number2)\n```\n\n- interface.py: Update the Interface class to include buttons for subtraction, multiplication, and division, and link them to the corresponding methods in the Calculator class. Also, handle the display of errors such as division by zero.\n```python\n## interface.py\nimport tkinter as tk\nfrom tkinter import messagebox\nfrom calculator import Calculator\n...\nclass Interface:\n ...\n def create_widgets(self):\n ...\n self.subtract_button = tk.Button(self.root, text='-', command=self.subtract, font=('Arial', 18))\n self.subtract_button.grid(row=3, column=0, sticky='nsew')\n\n self.multiply_button = tk.Button(self.root, text='*', command=self.multiply, font=('Arial', 18))\n self.multiply_button.grid(row=3, column=1, sticky='nsew')\n\n self.divide_button = tk.Button(self.root, text='/', command=self.divide, font=('Arial', 18))\n self.divide_button.grid(row=3, column=2, sticky='nsew')\n ...\n\n def subtract(self):\n number1, number2 = self.get_input()\n if number1 is not None and number2 is not None:\n result = self.calculator.subtract(number1, number2)\n self.display_result(result)\n\n def multiply(self):\n number1, number2 = self.get_input()\n if number1 is not None and number2 is not None:\n result = self.calculator.multiply(number1, number2)\n self.display_result(result)\n\n def divide(self):\n number1, number2 = self.get_input()\n if number1 is not None and number2 is not None:\n try:\n result = self.calculator.divide(number1, number2)\n except ValueError as e:\n self.show_error(str(e))\n return\n self.display_result(result)\n```\n\n- main.py: No changes needed in main.py as it serves as the entry point and will run the updated Interface class.\n```python\n## main.py\nfrom interface import Interface\n...\n```\n\nNote: Ensure that the new operations buttons in the Interface class are properly arranged and that the grid layout is adjusted accordingly. Also, make sure to import the messagebox module from tkinter for error handling." diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index ef0dd2eff..d520eb863 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_write_prd_an.py.py +@File : test_write_prd_an.py """ import pytest From 994cfe814b3f749934e8187edf28b6567244ac2e Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 3 Jan 2024 18:10:53 +0800 Subject: [PATCH 256/637] 1. Added test_increment.py 2. Modify prompt in design_api_an.py 3. Modify rename_root function using method of copy in git_repository.py --- metagpt/actions/design_api_an.py | 2 +- metagpt/utils/git_repository.py | 13 ++- tests/metagpt/test_increment.py | 158 +++++++++++++++++++++++++++++++ 3 files changed, 168 insertions(+), 5 deletions(-) create mode 100644 tests/metagpt/test_increment.py diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index bcfaf0bfb..3e8265e95 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -138,7 +138,7 @@ Based on new requirements, review and refine the system design. Integrate existi NODES = [ IMPLEMENTATION_APPROACH, # PROJECT_NAME, - REFINED_FILE_LIST, + FILE_LIST, DATA_STRUCTURES_AND_INTERFACES, PROGRAM_CALL_FLOW, ANYTHING_UNCLEAR, diff --git a/metagpt/utils/git_repository.py b/metagpt/utils/git_repository.py index d2bdf5d85..14df607e3 100644 --- a/metagpt/utils/git_repository.py +++ b/metagpt/utils/git_repository.py @@ -190,7 +190,7 @@ class GitRepository: return self._dependency def rename_root(self, new_dir_name): - """Rename the root directory of the Git repository. + """Rename/Copy the root directory of the Git repository. :param new_dir_name: The new name for the root directory. """ @@ -201,10 +201,15 @@ class GitRepository: logger.info(f"Delete directory {str(new_path)}") shutil.rmtree(new_path) try: - shutil.move(src=str(self.workdir), dst=str(new_path)) + shutil.copytree(src=str(self.workdir), dst=str(new_path)) except Exception as e: - logger.warning(f"Move {str(self.workdir)} to {str(new_path)} error: {e}") - logger.info(f"Rename directory {str(self.workdir)} to {str(new_path)}") + logger.warning(f"Copy {str(self.workdir)} to {str(new_path)} error: {e}") + logger.info(f"Copy directory {str(self.workdir)} to {str(new_path)}") + # try: + # shutil.move(src=str(self.workdir), dst=str(new_path)) + # except Exception as e: + # logger.warning(f"Move {str(self.workdir)} to {str(new_path)} error: {e}") + # logger.info(f"Rename directory {str(self.workdir)} to {str(new_path)}") self._repository = Repo(new_path) self._gitignore_rules = parse_gitignore(full_path=str(new_path / ".gitignore")) diff --git a/tests/metagpt/test_increment.py b/tests/metagpt/test_increment.py new file mode 100644 index 000000000..68435e930 --- /dev/null +++ b/tests/metagpt/test_increment.py @@ -0,0 +1,158 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/03 +@Author : mannaandpoem +@File : test_increment.py +""" +import pytest +from typer.testing import CliRunner + +from metagpt.logs import logger +from metagpt.startup import app + +runner = CliRunner() + + +def test_refine_calculator(): + args = [ + "Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator", + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\simple_add_calculator", + "--project-name", + "calculator", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refine_number_guessing_game(): + args = [ + "Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal" + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\number_guessing_game", + "--project-name", + "number_guessing_game", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refine_dice_simulator_1(): + args = [ + "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores" + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\dice_simulator_new", + "--project-name", + "dice_simulator_b_1", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refine_dice_simulator_2(): + args = [ + "Add functionality to view the history of scores and perform statistical analysis on them. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores and display the statistical analysis results of the current score", + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\dice_simulator_new", + "--project-name", + "dice_simulator_2", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refine_dice_simulator_3(): + args = [ + "Add functionality to set the number of sides on a die; Add functionality to view the history of scores; Add functionality to perform statistical analysis on all scores. The original dice rolling game could roll the dice multiple times and only display the current game result. But the new requirement add function that players to customize the number of sides of the dice and to view the history of scores and display the statistical analysis" + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\dice_simulator_new", + "--project-name", + "dice_simulator_3", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refine_pygame_2048_1(): + args = [ + "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8" + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\pygame_2048", + "--project-name", + "pygame_2048_1", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refine_pygame_2048_2(): + args = [ + "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores" + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\pygame_2048", + "--project-name", + "pygame_2048_2", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refine_pygame_2048_3(): + args = [ + "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game." + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\pygame_2048", + "--project-name", + "pygame_2048_3", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refine_word_cloud_1(): + args = [ + "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove." + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\word_cloud", + "--project-name", + "word_cloud_1", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refine_word_cloud_2(): + args = [ + "Add a feature to customize the resolution of the word cloud.The new version allows users to customize the size and resolution of the generated word cloud after uploading a text file, and then generate the word cloud." + "--inc", + "--project-path", + r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\word_cloud", + "--project-name", + "word_cloud_2", + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +if __name__ == "__main__": + pytest.main([__file__, "-s"]) From dcefed8a98ad6b5949be193ef27c5dce211c61c5 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 3 Jan 2024 19:11:23 +0800 Subject: [PATCH 257/637] Update test_increment.py --- tests/metagpt/test_increment.py | 27 ++++++++++++++------------- 1 file changed, 14 insertions(+), 13 deletions(-) diff --git a/tests/metagpt/test_increment.py b/tests/metagpt/test_increment.py index 68435e930..26b9951b7 100644 --- a/tests/metagpt/test_increment.py +++ b/tests/metagpt/test_increment.py @@ -8,20 +8,21 @@ import pytest from typer.testing import CliRunner +from metagpt.const import DEFAULT_WORKSPACE_ROOT from metagpt.logs import logger from metagpt.startup import app runner = CliRunner() -def test_refine_calculator(): +def test_refine_simple_calculator(): args = [ "Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator", "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\simple_add_calculator", + f"{DEFAULT_WORKSPACE_ROOT}/simple_add_calculator", "--project-name", - "calculator", + "simple_calculator", ] result = runner.invoke(app, args) logger.info(result) @@ -33,7 +34,7 @@ def test_refine_number_guessing_game(): "Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal" "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\number_guessing_game", + f"{DEFAULT_WORKSPACE_ROOT}/number_guessing_game", "--project-name", "number_guessing_game", ] @@ -47,9 +48,9 @@ def test_refine_dice_simulator_1(): "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores" "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\dice_simulator_new", + f"{DEFAULT_WORKSPACE_ROOT}/dice_simulator_new", "--project-name", - "dice_simulator_b_1", + "dice_simulator_1", ] result = runner.invoke(app, args) logger.info(result) @@ -61,7 +62,7 @@ def test_refine_dice_simulator_2(): "Add functionality to view the history of scores and perform statistical analysis on them. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores and display the statistical analysis results of the current score", "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\dice_simulator_new", + f"{DEFAULT_WORKSPACE_ROOT}/dice_simulator_new", "--project-name", "dice_simulator_2", ] @@ -75,7 +76,7 @@ def test_refine_dice_simulator_3(): "Add functionality to set the number of sides on a die; Add functionality to view the history of scores; Add functionality to perform statistical analysis on all scores. The original dice rolling game could roll the dice multiple times and only display the current game result. But the new requirement add function that players to customize the number of sides of the dice and to view the history of scores and display the statistical analysis" "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\dice_simulator_new", + f"{DEFAULT_WORKSPACE_ROOT}/dice_simulator_new", "--project-name", "dice_simulator_3", ] @@ -89,7 +90,7 @@ def test_refine_pygame_2048_1(): "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8" "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\pygame_2048", + f"{DEFAULT_WORKSPACE_ROOT}/pygame_2048", "--project-name", "pygame_2048_1", ] @@ -103,7 +104,7 @@ def test_refine_pygame_2048_2(): "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores" "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\pygame_2048", + f"{DEFAULT_WORKSPACE_ROOT}/pygame_2048", "--project-name", "pygame_2048_2", ] @@ -117,7 +118,7 @@ def test_refine_pygame_2048_3(): "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game." "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\pygame_2048", + f"{DEFAULT_WORKSPACE_ROOT}/pygame_2048", "--project-name", "pygame_2048_3", ] @@ -131,7 +132,7 @@ def test_refine_word_cloud_1(): "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove." "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\word_cloud", + f"{DEFAULT_WORKSPACE_ROOT}/word_cloud", "--project-name", "word_cloud_1", ] @@ -145,7 +146,7 @@ def test_refine_word_cloud_2(): "Add a feature to customize the resolution of the word cloud.The new version allows users to customize the size and resolution of the generated word cloud after uploading a text file, and then generate the word cloud." "--inc", "--project-path", - r"C:\Users\ASUS\PycharmProjects\MetaGPT\workspace\word_cloud", + f"{DEFAULT_WORKSPACE_ROOT}/word_cloud", "--project-name", "word_cloud_2", ] From 9435352031687a8abd5906121086dd83252409c9 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 3 Jan 2024 20:04:16 +0800 Subject: [PATCH 258/637] Update design_api_an.py and test_increment.py --- metagpt/actions/design_api_an.py | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index 0833297f1..3e8265e95 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -8,6 +8,7 @@ from typing import List from metagpt.actions.action_node import ActionNode +from metagpt.logs import logger from metagpt.utils.mermaid import MMC1, MMC1_REFINE, MMC2, MMC2_REFINE IMPLEMENTATION_APPROACH = ActionNode( @@ -156,3 +157,14 @@ REFINE_NODES = [ DESIGN_API_NODE = ActionNode.from_children("DesignAPI", NODES) INCREMENTAL_DESIGN_NODES = ActionNode.from_children("Incremental_Design_API", INC_NODES) REFINED_DESIGN_NODES = ActionNode.from_children("Refined_Design_API", REFINE_NODES) + + +def main(): + prompt = DESIGN_API_NODE.compile(context="") + logger.info(prompt) + prompt = REFINED_DESIGN_NODES.compile(context="") + logger.info(prompt) + + +if __name__ == "__main__": + main() From 21d8b48e8e5b000c63e8c87c4e5aaf6d4d6d839d Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 3 Jan 2024 21:20:36 +0800 Subject: [PATCH 259/637] Update test_increment.py for "--project-path" --- tests/metagpt/test_increment.py | 21 ++++++++++----------- 1 file changed, 10 insertions(+), 11 deletions(-) diff --git a/tests/metagpt/test_increment.py b/tests/metagpt/test_increment.py index 26b9951b7..a6d5e0821 100644 --- a/tests/metagpt/test_increment.py +++ b/tests/metagpt/test_increment.py @@ -8,7 +8,6 @@ import pytest from typer.testing import CliRunner -from metagpt.const import DEFAULT_WORKSPACE_ROOT from metagpt.logs import logger from metagpt.startup import app @@ -20,7 +19,7 @@ def test_refine_simple_calculator(): "Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator", "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/simple_add_calculator", + "data/simple_add_calculator", "--project-name", "simple_calculator", ] @@ -34,7 +33,7 @@ def test_refine_number_guessing_game(): "Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal" "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/number_guessing_game", + "data/number_guessing_game", "--project-name", "number_guessing_game", ] @@ -48,7 +47,7 @@ def test_refine_dice_simulator_1(): "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores" "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/dice_simulator_new", + "data/dice_simulator_new", "--project-name", "dice_simulator_1", ] @@ -62,7 +61,7 @@ def test_refine_dice_simulator_2(): "Add functionality to view the history of scores and perform statistical analysis on them. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores and display the statistical analysis results of the current score", "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/dice_simulator_new", + "data/dice_simulator_new", "--project-name", "dice_simulator_2", ] @@ -76,7 +75,7 @@ def test_refine_dice_simulator_3(): "Add functionality to set the number of sides on a die; Add functionality to view the history of scores; Add functionality to perform statistical analysis on all scores. The original dice rolling game could roll the dice multiple times and only display the current game result. But the new requirement add function that players to customize the number of sides of the dice and to view the history of scores and display the statistical analysis" "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/dice_simulator_new", + "data/dice_simulator_new", "--project-name", "dice_simulator_3", ] @@ -90,7 +89,7 @@ def test_refine_pygame_2048_1(): "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8" "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/pygame_2048", + "data/pygame_2048", "--project-name", "pygame_2048_1", ] @@ -104,7 +103,7 @@ def test_refine_pygame_2048_2(): "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores" "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/pygame_2048", + "data/pygame_2048", "--project-name", "pygame_2048_2", ] @@ -118,7 +117,7 @@ def test_refine_pygame_2048_3(): "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game." "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/pygame_2048", + "data/pygame_2048", "--project-name", "pygame_2048_3", ] @@ -132,7 +131,7 @@ def test_refine_word_cloud_1(): "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove." "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/word_cloud", + "data/word_cloud", "--project-name", "word_cloud_1", ] @@ -146,7 +145,7 @@ def test_refine_word_cloud_2(): "Add a feature to customize the resolution of the word cloud.The new version allows users to customize the size and resolution of the generated word cloud after uploading a text file, and then generate the word cloud." "--inc", "--project-path", - f"{DEFAULT_WORKSPACE_ROOT}/word_cloud", + "data/word_cloud", "--project-name", "word_cloud_2", ] From 0168c993235ed7bbf984679903ee3c29fde82f1d Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 3 Jan 2024 23:15:43 +0800 Subject: [PATCH 260/637] Replace self._rc to self.rc in engineer.py --- metagpt/roles/engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index eeade5856..d7a0312be 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -332,7 +332,7 @@ class Engineer(Role): async def _write_code_guideline(self): logger.info("Writing code guideline..") - requirement = str(self._rc.memory.get_by_role("Human")[0]) + requirement = str(self.rc.memory.get_by_role("Human")[0]) # prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) design_file_repo = CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO) task_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) From e3cd1a2cc18b1db74ca31a6a094960effd0f164c Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 4 Jan 2024 10:41:36 +0800 Subject: [PATCH 261/637] Update prompt and fix up some bug --- metagpt/actions/write_code_guideline_an.py | 2 +- metagpt/actions/write_prd_an.py | 2 +- metagpt/roles/engineer.py | 2 +- metagpt/utils/git_repository.py | 13 ++--- tests/metagpt/actions/test_design_api_an.py | 6 +- .../actions/test_project_management_an.py | 6 +- .../actions/test_write_code_guideline_an.py | 2 +- tests/metagpt/actions/test_write_prd_an.py | 6 +- tests/metagpt/test_increment.py | 56 ++++++------------- 9 files changed, 35 insertions(+), 60 deletions(-) diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index 0677f6edc..43645e80c 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -425,7 +425,7 @@ class WriteCodeGuideline(Action): async def main(): write_code_guideline = WriteCodeGuideline() node = await write_code_guideline.run(CODE_GUIDELINE_CONTEXT_EXAMPLE) - guideline = node.instruct_content.json(ensure_ascii=False) + guideline = node.instruct_content.model_dump_json() print(guideline) diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index e2fbb2599..91c1b2837 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -214,7 +214,7 @@ Based on New Requirements, output a New PRD that seamlessly integrates both the """ REFINE_PRD_TEMPLATE = """ -### New Project Name +### Project Name {project_name} ### New Requirements diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index d7a0312be..d4947a8f8 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -346,7 +346,7 @@ class Engineer(Role): context = CODE_GUIDELINE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) node = await WriteCodeGuideline().run(context=context) - guideline = node.instruct_content.json(ensure_ascii=False) + guideline = node.instruct_content.model_dump_json() return guideline @staticmethod diff --git a/metagpt/utils/git_repository.py b/metagpt/utils/git_repository.py index b42111620..e9855df05 100644 --- a/metagpt/utils/git_repository.py +++ b/metagpt/utils/git_repository.py @@ -189,7 +189,7 @@ class GitRepository: return self._dependency def rename_root(self, new_dir_name): - """Rename/Copy the root directory of the Git repository. + """Rename the root directory of the Git repository. :param new_dir_name: The new name for the root directory. """ @@ -200,15 +200,10 @@ class GitRepository: logger.info(f"Delete directory {str(new_path)}") shutil.rmtree(new_path) try: - shutil.copytree(src=str(self.workdir), dst=str(new_path)) + shutil.move(src=str(self.workdir), dst=str(new_path)) except Exception as e: - logger.warning(f"Copy {str(self.workdir)} to {str(new_path)} error: {e}") - logger.info(f"Copy directory {str(self.workdir)} to {str(new_path)}") - # try: - # shutil.move(src=str(self.workdir), dst=str(new_path)) - # except Exception as e: - # logger.warning(f"Move {str(self.workdir)} to {str(new_path)} error: {e}") - # logger.info(f"Rename directory {str(self.workdir)} to {str(new_path)}") + logger.warning(f"Move {str(self.workdir)} to {str(new_path)} error: {e}") + logger.info(f"Rename directory {str(self.workdir)} to {str(new_path)}") self._repository = Repo(new_path) self._gitignore_rules = parse_gitignore(full_path=str(new_path / ".gitignore")) diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index 305c708c0..5b016ecce 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -8,7 +8,7 @@ import pytest from metagpt.actions.design_api_an import REFINED_DESIGN_NODES -from metagpt.provider import OpenAIGPTAPI +from metagpt.llm import LLM CONTEXT = """ ### Legacy Content @@ -94,11 +94,11 @@ CONTEXT = """ @pytest.fixture() def llm(): - return OpenAIGPTAPI() + return LLM() @pytest.mark.asyncio async def test_write_design_an(): node = await REFINED_DESIGN_NODES.fill(CONTEXT, llm) assert node.instruct_content - assert "Refined Data Structures and Interfaces" in node.instruct_content.json(ensure_ascii=False) + assert "Refined Data Structures and Interfaces" in node.instruct_content.model_dump_json() diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index 9ad4360cf..e0c1381ec 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -8,7 +8,7 @@ import pytest from metagpt.actions.project_management_an import REFINED_PM_NODES -from metagpt.provider import OpenAIGPTAPI +from metagpt.llm import LLM CONTEXT = """ ### Legacy Content @@ -134,11 +134,11 @@ CONTEXT = """ @pytest.fixture() def llm(): - return OpenAIGPTAPI() + return LLM() @pytest.mark.asyncio async def test_project_management_an(llm): node = await REFINED_PM_NODES.fill(CONTEXT, llm) assert node.instruct_content - assert "Refined Logic Analysis" in node.instruct_content.json(ensure_ascii=False) + assert "Refined Logic Analysis" in node.instruct_content.model_dump_json() diff --git a/tests/metagpt/actions/test_write_code_guideline_an.py b/tests/metagpt/actions/test_write_code_guideline_an.py index d740a6bd8..f8a1ae626 100644 --- a/tests/metagpt/actions/test_write_code_guideline_an.py +++ b/tests/metagpt/actions/test_write_code_guideline_an.py @@ -200,7 +200,7 @@ async def test_write_code_guideline_an(): ) node = await write_code_guideline.run(context=context) assert node.instruct_content - assert "Incremental Change" in node.instruct_content.json(ensure_ascii=False) + assert "Incremental Change" in node.instruct_content.model_dump_json() @pytest.mark.asyncio diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index d520eb863..79cd913cd 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -8,7 +8,7 @@ import pytest from metagpt.actions.write_prd_an import REFINE_PRD_NODE -from metagpt.provider import OpenAIGPTAPI +from metagpt.llm import LLM CONTEXT = """ ### New Project Name @@ -78,11 +78,11 @@ Please change the game's score target from 2048 to 4096, and change the interfac @pytest.fixture() def llm(): - return OpenAIGPTAPI() + return LLM() @pytest.mark.asyncio async def test_write_prd_an(llm): node = await REFINE_PRD_NODE.fill(CONTEXT, llm) assert node.instruct_content - assert "Refined Requirement Pool" in node.instruct_content.json(ensure_ascii=False) + assert "Refined Requirement Pool" in node.instruct_content.model_dump_json() diff --git a/tests/metagpt/test_increment.py b/tests/metagpt/test_increment.py index a6d5e0821..25769ff6a 100644 --- a/tests/metagpt/test_increment.py +++ b/tests/metagpt/test_increment.py @@ -14,140 +14,120 @@ from metagpt.startup import app runner = CliRunner() -def test_refine_simple_calculator(): +def test_refined_simple_calculator(): args = [ "Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator", "--inc", "--project-path", "data/simple_add_calculator", - "--project-name", - "simple_calculator", ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) -def test_refine_number_guessing_game(): +def test_refined_number_guessing_game(): args = [ - "Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal" + "Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal", "--inc", "--project-path", "data/number_guessing_game", - "--project-name", - "number_guessing_game", ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) -def test_refine_dice_simulator_1(): +def test_refined_dice_simulator_1(): args = [ - "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores" + "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores", "--inc", "--project-path", "data/dice_simulator_new", - "--project-name", - "dice_simulator_1", ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) -def test_refine_dice_simulator_2(): +def test_refined_dice_simulator_2(): args = [ "Add functionality to view the history of scores and perform statistical analysis on them. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores and display the statistical analysis results of the current score", "--inc", "--project-path", "data/dice_simulator_new", - "--project-name", - "dice_simulator_2", ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) -def test_refine_dice_simulator_3(): +def test_refined_dice_simulator_3(): args = [ - "Add functionality to set the number of sides on a die; Add functionality to view the history of scores; Add functionality to perform statistical analysis on all scores. The original dice rolling game could roll the dice multiple times and only display the current game result. But the new requirement add function that players to customize the number of sides of the dice and to view the history of scores and display the statistical analysis" + "Add functionality to set the number of sides on a die; Add functionality to view the history of scores; Add functionality to perform statistical analysis on all scores. The original dice rolling game could roll the dice multiple times and only display the current game result. But the new requirement add function that players to customize the number of sides of the dice and to view the history of scores and display the statistical analysis", "--inc", "--project-path", "data/dice_simulator_new", - "--project-name", - "dice_simulator_3", ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) -def test_refine_pygame_2048_1(): +def test_refined_pygame_2048_1(): args = [ - "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8" + "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8", "--inc", "--project-path", "data/pygame_2048", - "--project-name", - "pygame_2048_1", ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) -def test_refine_pygame_2048_2(): +def test_refined_pygame_2048_2(): args = [ - "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores" + "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores", "--inc", "--project-path", "data/pygame_2048", - "--project-name", - "pygame_2048_2", ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) -def test_refine_pygame_2048_3(): +def test_refined_pygame_2048_3(): args = [ - "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game." + "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game.", "--inc", "--project-path", "data/pygame_2048", - "--project-name", - "pygame_2048_3", ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) -def test_refine_word_cloud_1(): +def test_refined_word_cloud_1(): args = [ - "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove." + "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove.", "--inc", "--project-path", "data/word_cloud", - "--project-name", - "word_cloud_1", ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) -def test_refine_word_cloud_2(): +def test_refined_word_cloud_2(): args = [ - "Add a feature to customize the resolution of the word cloud.The new version allows users to customize the size and resolution of the generated word cloud after uploading a text file, and then generate the word cloud." + "Add a feature to customize the resolution of the word cloud.The new version allows users to customize the size and resolution of the generated word cloud after uploading a text file, and then generate the word cloud.", "--inc", "--project-path", "data/word_cloud", - "--project-name", - "word_cloud_2", ] result = runner.invoke(app, args) logger.info(result) From 02d6db6506a5fd7a36df38f4c3f8bbcdb9e20e3d Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Thu, 4 Jan 2024 16:56:17 +0800 Subject: [PATCH 262/637] release v0.6.1 --- setup.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/setup.py b/setup.py index ae0b0d8aa..17fe8815e 100644 --- a/setup.py +++ b/setup.py @@ -56,7 +56,7 @@ extras_require["dev"] = (["pylint~=3.0.3", "black~=23.3.0", "isort~=5.12.0", "pr setup( name="metagpt", - version="0.6.0", + version="0.6.1", description="The Multi-Agent Framework", long_description=long_description, long_description_content_type="text/markdown", From 97ee2a0a61c3ccf93c355c77f23cad145717e1c8 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 4 Jan 2024 17:58:55 +0800 Subject: [PATCH 263/637] 1. Added 4 compressed package of code examples 2. Update test_incremental_dev.py and engineer.py --- data/dice_simulator_new.zip | Bin 0 -> 27923 bytes data/number_guessing_game.zip | Bin 0 -> 65020 bytes data/pygame_2048.zip | Bin 0 -> 19338 bytes data/simple_add_calculator.zip | Bin 0 -> 53760 bytes data/word_cloud.zip | Bin 0 -> 54076 bytes metagpt/roles/engineer.py | 19 ++- ...t_increment.py => test_incremental_dev.py} | 112 ++++++++++++++++-- 7 files changed, 119 insertions(+), 12 deletions(-) create mode 100644 data/dice_simulator_new.zip create mode 100644 data/number_guessing_game.zip create mode 100644 data/pygame_2048.zip create mode 100644 data/simple_add_calculator.zip create mode 100644 data/word_cloud.zip rename tests/metagpt/{test_increment.py => test_incremental_dev.py} (59%) diff --git a/data/dice_simulator_new.zip b/data/dice_simulator_new.zip new file mode 100644 index 0000000000000000000000000000000000000000..307e41541899d742a97d1209813169f672fe13cd GIT binary patch literal 27923 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 t`cn=62#v`Dtr9SA^3b literal 0 HcmV?d00001 diff --git a/data/pygame_2048.zip b/data/pygame_2048.zip new file mode 100644 index 0000000000000000000000000000000000000000..476d0b438b207f594418d2e8fb78a29371590f66 GIT binary patch literal 19338 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 z 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

Ei(XwNOEtpM1u_&thFjw8IQWXi0_ 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 UqTY8I_-Dh4fe{3}4Flu<0l%(&;s5{u literal 0 HcmV?d00001 diff --git a/data/word_cloud.zip b/data/word_cloud.zip new file mode 100644 index 0000000000000000000000000000000000000000..fd395c5bd2c2770fb8a1f2482d16a2a3cd9f0686 GIT binary patch literal 54076 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 An*aa+ literal 0 HcmV?d00001 diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index d4947a8f8..a1e93be40 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -117,9 +117,22 @@ class Engineer(Role): action = WriteCodeReview(context=coding_context, llm=self.llm) self._init_action_system_message(action) coding_context = await action.run() + + # Get dependencies + if guideline: + dependencies = { + coding_context.design_doc.root_relative_path, + coding_context.task_doc.root_relative_path, + "code_guideline.json", + } + else: + dependencies = { + coding_context.design_doc.root_relative_path, + coding_context.task_doc.root_relative_path, + } await src_file_repo.save( coding_context.filename, - dependencies={coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path}, + dependencies=dependencies, content=coding_context.code_doc.content, ) msg = Message( @@ -347,6 +360,10 @@ class Engineer(Role): context = CODE_GUIDELINE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) node = await WriteCodeGuideline().run(context=context) guideline = node.instruct_content.model_dump_json() + + await CONFIG.git_repo.new_file_repository(CONFIG.git_repo.workdir).save( + filename="code_guideline.json", content=guideline + ) return guideline @staticmethod diff --git a/tests/metagpt/test_increment.py b/tests/metagpt/test_incremental_dev.py similarity index 59% rename from tests/metagpt/test_increment.py rename to tests/metagpt/test_incremental_dev.py index 25769ff6a..dfb8fe039 100644 --- a/tests/metagpt/test_increment.py +++ b/tests/metagpt/test_incremental_dev.py @@ -3,11 +3,14 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_increment.py +@File : test_incremental_dev.py """ +import os + import pytest from typer.testing import CliRunner +from metagpt.const import DATA_PATH from metagpt.logs import logger from metagpt.startup import app @@ -15,11 +18,29 @@ runner = CliRunner() def test_refined_simple_calculator(): + project_path = f"{DATA_PATH}/simple_add_calculator" + check_or_create_base_tag(project_path) + args = [ "Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator", "--inc", "--project-path", - "data/simple_add_calculator", + project_path, + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + + +def test_refined_simple_calculator_2(): + project_path = f"{DATA_PATH}/simple_add_calculator" + check_or_create_base_tag(project_path) + + args = [ + "Add exponentiation operation to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce exponentiation operation into the calculator", + "--inc", + "--project-path", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -27,11 +48,14 @@ def test_refined_simple_calculator(): def test_refined_number_guessing_game(): + project_path = f"{DATA_PATH}/number_guessing_game" + check_or_create_base_tag(project_path) + args = [ "Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal", "--inc", "--project-path", - "data/number_guessing_game", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -39,11 +63,14 @@ def test_refined_number_guessing_game(): def test_refined_dice_simulator_1(): + project_path = f"{DATA_PATH}/dice_simulator_new" + check_or_create_base_tag(project_path) + args = [ "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores", "--inc", "--project-path", - "data/dice_simulator_new", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -51,11 +78,14 @@ def test_refined_dice_simulator_1(): def test_refined_dice_simulator_2(): + project_path = f"{DATA_PATH}/dice_simulator_new" + check_or_create_base_tag(project_path) + args = [ "Add functionality to view the history of scores and perform statistical analysis on them. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores and display the statistical analysis results of the current score", "--inc", "--project-path", - "data/dice_simulator_new", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -63,11 +93,14 @@ def test_refined_dice_simulator_2(): def test_refined_dice_simulator_3(): + project_path = f"{DATA_PATH}/dice_simulator_new" + check_or_create_base_tag(project_path) + args = [ "Add functionality to set the number of sides on a die; Add functionality to view the history of scores; Add functionality to perform statistical analysis on all scores. The original dice rolling game could roll the dice multiple times and only display the current game result. But the new requirement add function that players to customize the number of sides of the dice and to view the history of scores and display the statistical analysis", "--inc", "--project-path", - "data/dice_simulator_new", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -75,11 +108,14 @@ def test_refined_dice_simulator_3(): def test_refined_pygame_2048_1(): + project_path = f"{DATA_PATH}/pygame_2048" + check_or_create_base_tag(project_path) + args = [ "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8", "--inc", "--project-path", - "data/pygame_2048", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -87,11 +123,14 @@ def test_refined_pygame_2048_1(): def test_refined_pygame_2048_2(): + project_path = f"{DATA_PATH}/pygame_2048" + check_or_create_base_tag(project_path) + args = [ "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores", "--inc", "--project-path", - "data/pygame_2048", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -99,11 +138,14 @@ def test_refined_pygame_2048_2(): def test_refined_pygame_2048_3(): + project_path = f"{DATA_PATH}/pygame_2048" + check_or_create_base_tag(project_path) + args = [ "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game.", "--inc", "--project-path", - "data/pygame_2048", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -111,11 +153,14 @@ def test_refined_pygame_2048_3(): def test_refined_word_cloud_1(): + project_path = f"{DATA_PATH}/word_cloud" + check_or_create_base_tag(project_path) + args = [ "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove.", "--inc", "--project-path", - "data/word_cloud", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -123,16 +168,61 @@ def test_refined_word_cloud_1(): def test_refined_word_cloud_2(): + project_path = f"{DATA_PATH}/word_cloud" + check_or_create_base_tag(project_path) + args = [ "Add a feature to customize the resolution of the word cloud.The new version allows users to customize the size and resolution of the generated word cloud after uploading a text file, and then generate the word cloud.", "--inc", "--project-path", - "data/word_cloud", + project_path, ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) +def check_or_create_base_tag(project_path): + # Change the current working directory to the specified project path + os.chdir(project_path) + + # Initialize a Git repository + os.system("git init") + + # Check if the 'base' tag exists + check_base_tag_cmd = "git show-ref --verify --quiet refs/tags/base" + has_base_tag = os.system(check_base_tag_cmd) == 0 + + if has_base_tag: + logger.info("base tag exists") + # Switch to the 'base' branch if it exists + switch_to_base_branch_cmd = "git checkout base" + if os.system(switch_to_base_branch_cmd) == 0: + logger.info("switched to base branch") + else: + logger.debug("Failed to switch to base branch.") + else: + logger.info("Base tag doesn't exist.") + # Add and commit the current code if 'base' tag doesn't exist + add_cmd = "git add ." + commit_cmd = 'git commit -m "Initial commit"' + add_and_commit_success = os.system(add_cmd) == 0 & os.system(commit_cmd) == 0 + + if add_and_commit_success: + logger.info("Added and committed all files with the message 'Initial commit'.") + else: + logger.debug("Failed to add and commit all files.") + + # Add 'base' tag + add_base_tag_cmd = "git tag base" + + # Check if the 'git tag' command was successful + tag_cmd_success = os.system(add_base_tag_cmd) == 0 + if tag_cmd_success: + logger.info("Successfully added 'base' tag.") + else: + logger.debug("Failed to add 'base' tag.") + + if __name__ == "__main__": pytest.main([__file__, "-s"]) From 40db41cda5c489a7cfa7b99035d76808caa62e1d Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 4 Jan 2024 17:58:55 +0800 Subject: [PATCH 264/637] 1. Added 4 compressed package of code examples 2. Update test_incremental_dev.py and engineer.py --- data/dice_simulator_new.zip | Bin 0 -> 27923 bytes data/number_guessing_game.zip | Bin 0 -> 65020 bytes data/pygame_2048.zip | Bin 0 -> 19338 bytes data/simple_add_calculator.zip | Bin 0 -> 53760 bytes data/word_cloud.zip | Bin 0 -> 54076 bytes metagpt/roles/engineer.py | 19 +++- ...t_increment.py => test_incremental_dev.py} | 97 ++++++++++++++++-- 7 files changed, 104 insertions(+), 12 deletions(-) create mode 100644 data/dice_simulator_new.zip create mode 100644 data/number_guessing_game.zip create mode 100644 data/pygame_2048.zip create mode 100644 data/simple_add_calculator.zip create mode 100644 data/word_cloud.zip rename tests/metagpt/{test_increment.py => test_incremental_dev.py} (63%) diff --git a/data/dice_simulator_new.zip b/data/dice_simulator_new.zip new file mode 100644 index 0000000000000000000000000000000000000000..307e41541899d742a97d1209813169f672fe13cd GIT binary patch literal 27923 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 t`cn=62#v`Dtr9SA^3b literal 0 HcmV?d00001 diff --git a/data/pygame_2048.zip b/data/pygame_2048.zip new file mode 100644 index 0000000000000000000000000000000000000000..476d0b438b207f594418d2e8fb78a29371590f66 GIT binary patch literal 19338 zcmb7s1z43!*Y>8QL!?2v8>G9D?rsU`?v!p2q`RdXq>=9KPH6gwX_q=*Iob!M0 za6PiUu08j&*36n&Gxu5xSqU(36aWAK1?Ws8RILgS!uJ6FPZ|^ezypPM(`uF)5C!u#a=^7a8Gdg5ZVZh zA|`a25J@V$5j}Q~ZU+fYR!N$3fCO_HEJn&BA+=ov`%oPol{lMh%cbAY7ZP_ple&=S z>kErD@ij#Lc!MoXcf=dbUno*CYbs4?=9D#w5#A5U~rwvJ<{UO=3$viTixo-&>m76@uk0uOEN z{?xSsYxBZP<7P}{I9G0SG1{vFCSnWe3E}#>c=(j`r64~2;5HpgEoTWG8RnY`S)>?f zt{!`lALiS^O6HUagLyh7B8hV2oi+0qB zzT`<-x@s#gKI9MYbvWv&bl43bJvQfOiPIHn(HY6JCULFteao9txagv?JBN*CS7eJJ z$9Q%SIf=Oh)mKHqn9B(A0gl+%r~)*ANOxq*xcka`?Mp;R_zI5UUiM>M{8i?1S*9CJXFDx9PQ}fg_1-$Qe`FwKZL3@z^sE=vl@v zV>t*isr2G|_ykj^2Sd*^b?yZ_mSc@y*LZ^3A|o$|GwuYk=Sn9X))sZ$*DV?#I0$>E z%)8BTvki0cTKh-Ss-}BCjvV+orHj~E%(`W z9UXyCn6*vKDaqH$vdT5L&+*oJA`>kVLt;bf>oPON(DHO;mi9@nZ7}ZPf9TGHWM|nm7g}gR8jzU zLUActesX3xP#7xmS>o+&w9W0@tyuC+lC8XXN`DgcMgugE*e!IWYf>~606b*?IRxdf zKNNY+XT@2{0&wsvw|b7i&PL~F1>_VlB5?IDRu-0X96_R+!}lI(UhoJwck0OuM>)GSk0204=Q}!gAhQs#38luBl006>x@sSv&mvs zz73;J=~df^k@5`&fg?aHOR?aiWskOG2y{P+#)cbFdHZ!v4Keb>q{#i8>Wr?_`u?%4892uMjp6fFd4$!@psO)D5tX?vZ3$fMRu>#^kDZUr+GDI%2~C_ONYvO290973Dn0>U09Fh z*2WBAd&ek^KKRV~OX0mdUFfuf;wi+#tEUKfz91GYfy!bt2wAZ)Rz}Zh|Ft~T!WwDT z66bl)-VU1%v)Ek_&fWDk3+d6>%Q-$!9mAkz&tN}d7m3#U6Uv)wl4zn?yUjNoG1wU@ zJ3*E;`KvQQhn&{El+qnDP}5td7PCz0SAdAGp=AD?$)tPV>!?kg)4t%H((ND|0iZvh zX4CArLJ-&GXCBQcA2LEY5&90~7=P(2g?df`?1hW~f66!-Q| zq|$5o>hG?|c)zU&CT>KtEipjN)6P14pX`|@Yld9W&>#2J|GF-Pz@q%Z}Z4iM+hxQLsdzRfI5*2CjkU}S8!>VcAgHFc7OLGSNYS37(306ynA zeF|!}c-KX?oW#Fq#ef)-lBb9=h7pn=;FZdG01EGZdw1&{kmO9fral&T2EDL`G)GdM zG?&-8MGcE$d z(!BFC^8&6s!`&W;XJwe_mpagQzQ~62uoex9#MskEiw*-q3DO0UwZ^48C6lSeWJYN)296)7D3OU{O1Z8g2nME5!qraiSpAVhH3KJ+4sY`P~g=dr)ZhR zml7L!N>1@yd>cSSWlbR`G4>^eEGjYtq@t!=?;JlI&q}LPqqaqxs~Qp%`PgQtdDAq# zY(R>krJ~YMJ~vc>Jw2;y({v-Hx+!Aqe3u#cl^=^K$2wF#qz$Fu0L$>p zd~c+&_%}68;@?xEuzE^Z8h4IP?{J!IImIXJghfJztZEh1VP}OG5=xXP=T+L;IseMKfGz6M4 zOOZrQ&W*KD4#IgOZ0L2rh(BzdTGx}ZByemGTIPG&HbS5ZeyZ@jrgqGvk63#l@5Kqn z1=@NEb?stRUSb;a`x8FcaCqiATK#!8&SHj)1`4ZYDsCn)v1D$J!4mQi(t7oq&CjIZ zt~A=gGfiI>jog=_4zxOumlyIN;@`XQ^)w+F#1`ST38Jalfp@Fg$qKf0Uq_P44+&wh zwzZno?B`YH;`N8q5>KFP$tSkp-B$N{mSv-}oo)J~e(E`>cQ%$-6|4vM{t;eAEJ0zH zH|meF)~|;gBbOvG7md_xG)D(gGi}lz8!uTJBf};nEL}P!w~LF(1~w1I@k|eY#K1)T z`Is|3PWop($`~x;ENmA_Of3*}aCMG62N)%L#3AMsF#UqUdK1B@;Q4R^zVU<0bpF8v zMN8NtR4o2siYdDRB%wmvqHTY0mn_fp_+}b<+G_|R&AWfO7e0UF@eQo>f87E9evQO` z>g)Pe28KFz4i*-Awm{zCzcxklwTOYF02((W{#@g5frBu#By!`~)ViknIYS@Aj-|h35($^qYQQk}iC0tY`XC&-n^Tn$UfWgV zQ=YC0F!;E(r5-@2qqKF;)-ek1>mpSao_`m+%3OLs{e~Hx3Lg4hAf6x`NMjp*nr@#t z0~L7*v|JHfR31*@2Vz`Ioh5rVeB^H^4LG7_V;qn zWVY5Yep>uKAg?|maj0>P%u~$V1FFU}!4Ytl=F}-_a?}@!y&+JAA9+sP&y&_G%GZ$A z6uUU%F}okk#(o8nj(!!!aOy~?oqX?xWO?Ao6F^kBTm7zqV`wu2pV5d$pka)&`amF%JTwqq%~o z0%5^g&oamRsWp~Id*}Pa}i&ZcOn16Ql&HU9a--XKMX1bpP%y% zCkt+-vwnYLU~OOFM-P^4rtFm~N_|Oq<>C#L|3rrK@ z;foDEvPq`QI_p6_rE!7x?)7RdT+3M+R)@~*ho0HvSeCP-6AyktzLWx$XPf**L<`PpaKq`*y@4EW$bwOQL5+Blfl8d?}y+SA%Q+Yd+$N-MoqBp)86 z=@nC=XQZ!?eJwL6rSw{yT3Sl3cSM0aAN&O5&v*RYi9Eie{^3OITB5g_7QUP%T2 zp}&fVud#-iUvY$yb+yoJenY@8f&1ch$Sn33s$)Z+` z_CG+dA*cq}GBX`1$E&vyNXbA+4e`f(f8!R3F8tl=WWF;Elxztb=RH z9+kYO6EDk_iB(tO6!8tOSP+vXWU_CdAH^82diBbmxB@sOB8Gk@fU%;H7^LbKD zxtrgbUCb1Q4`V8%TpT&K*;`)0cQ4}H`NL$3^zL!m1Z2OD&xV(Gof#un-g2#)REjAv&(!8Z2I3!I7=eGx*?VJ6s$%1INAqr0U$1%Y@{*@^l*>A%}pt-OW$QV<0Hl8aflj|ts z?1b3QRsi(Ec%Yi_&I$244|0Sjxl1FpUQ4ypg5DE;FAi?Gl0lRbkD1;-^B8FA;%u59Z3r^+^E}L4Sel#BX8Gbp zrXS3|W;nsWD}|r@b>Gt6MN>iWd5H0BH?+n!|PDQ?n znuC~yt`s(Nec4&7899WOxZ=*6O=;CvIl~$?+Sy%T{*4*yi&tgKrY~skbw9`i z&nd*ZKPTUR1UrV;KhvKxiAYvV( znbO$mNQbzc@wnwO;cR^C?i=bqZ!AFm>o4J;KQ$y`~OJogW^r-fML_ z0B31brzNOJUukmj>{z)x0z{+U7v>6sIEB-N2ZWq}QH_%5tiA;x<(R@ab|xo}_(uB5 zh^e|u>ZEC1Bh!%Qg(eOA(B-1yQ5t!iqOG(y@j(^K!cJ;tfi1{g3=B#Uu){Gcgn!`9 z`WlF|XuO?xVJXWj=BkE^d6+nfkRY#!yygQ&f(SLC#In5nmY0;myt2!=e$KM;#MSXC zNvd+!JhgphZO5@eork)6Va&EgGq-$r>%+wzkI4XcUYRh&MCnfAIGS}=itmRr5}c?e zbcBm;0k&t9wFjo!pmx?wG6@1D{$k$oy@(5d`SC(2ZxjAf_A{H4!X$O=v2X7k#Dj7O zy*8!wPzb3x#k7PrV5o#)1RBIENU1q2q0)V*q`-xzMtppfXQ!62J(R|x`gO+6Qq$gg ziHniGrfD`1zt0R3;w}-(3z-TdQAznU=(Hg3RvqRhcJ~?K4qEP09_06WeHg=!CFr%x zc97zMr_G;RP|Id!ZpK$7?6|kAm~*Y2_%Vad7u9Ws!Q%-!-3lTnP>cQ0Qc8zQl_>-#m_(sH*!&vsz zp(3j1DY*96fl2r5px^Zvl4JskGI&uCEc%-_5e9jX{RcY_n1(kPD&<_)l_^3ihfrSc z4JFTNNY;KrPT!Fr;^xYwl#U=VLIJ~oA?RS9@MAT}XZwTh{PSQmVIpSySp}4b5Q&`x(~^TYRdG@` zE8BMXywh84uiY;_R#H8jnsGZB>6nHbPOhx?92-(`{cd~}v24+*vNHC--4byisNQ|d zm>nTfp>C_o1;5Bk>@NUVvJ3kQQYz7C?G3B;+dV^h!ygJ-?I#%mo{yCAN?BBc_P#3H zZTc`~%n99o{P5isB|9ATcdKWNG-4rSu|5)9Kd60kx;RoS(%O$sp^r5?(2r$%-A<`t ztK7-%`cH*glG26V@|}V+b~`0neu09fL-0uxzm}*=DbZ%3Tcd6$bB8kVzC5+I!P6Qg zAlTvXng|iF$zZ7ZCQPlAYtcM}YCqL-(E^<$R8rP`+bSc4+YA*Om^cj;4?>v>a( z#aCs-qI2-G6a0j}%?}%qH>+WgTNqJh_NsK2Bd)FaY?X*5G}03r9Zl69NeMd`bgMGZYoesbeI$Y#c)G|u^UI%-#;O+D|Bl<}&PoOB(+1ImY#tvqiTN=<|ng9XS(gMj(MYY6q6 z7llD3acIAA49vz7S+jZZlEX4Qn&*3oDbBj`6o)svh;LkVq8}E^{Bl0@rEU_jrd@Ey zqVw<|6Q@G_C&?jsU1(`u+xwP%h@^BrQ{u^v;#Y-wR#?shdxz!RQ>nq>cQ|q4Iw}Lk zRVh44AT-sCsW!r>AJ_K|_#;8d=y)U3=Xegss?N6D8#iA91LKjqu=-8G?K_#ZJ!Vy3 zpU-41GyIT=vDnMAv4j3RB-QL9yT991N*obcKNrxwVc!fDKlSGGj%cByQAn+1D`Q-| z%-G@We>o8-z%f(n;aU7aUpYNH6NNx#SK_r*);VWmS5ZD&(G8XRc-jsyF@EKl$|is7L1LRwM;YjOI3 zmxI3&WZXBVSk@*MW-igu%Fg1-h|{3p(k{y2+1RNCKU`SINdJvCtP#E8NMc;ltvs#g#CU3xVRv?iv z-wY>k?%&B?*g0P(uNP1fcXcei)1!vetIeEiQ5;E9lyhxS8b zRT$C5kJ=sHx5PS7*oVYak+Bj*Ad!^K*90oV#@KL)A8l9pIL0O5dw#p5`FDQ+_T2Tr z>6%KNPSdkE6*z79thf*l3N4x|Q8G=jE~E?NMYn^- zzhpp2xg>7OplPYgzXkkU>e2&$rosZ1)!>^r%^>@spI1)rhsU;I&1f! zONtF^1sj7Y*JF;;un?f*i(hw_96Dt&+y6l<$-BO#mYG^sovzlOCng=Pu;D^<#8!N4 zbZfw1xymAe=c+Ihlfq4-5x^X7>`cREIyF_EZ-s09Z9ko?+oO^t%gVTE9R35Yl{6DT zr5M*-i`J&f4goLIm_0T18}(Ad0lccuc zZ##w^L2c6(=$AVd9rhIImaLY0za<7x1(3k$(W@DjeVhP-A!WAVV zfmtRkL??Ko+GshbA@dr!&GyMlB8Q|N;7Z(TD9^pA0#- zM7HTu#GdqXL`JT1%xXK7Xpt#=+fl2UQAs=<6~ONk4lU?w+x|Y+QP$D^<(Oydf_9OLo=zol zfk7*!evU4*==%ELW#Wj7Y2o?=5?ZCEp6hlvoLDXjbFV!isw5Z0R#z@N1wK3dO?)4^ ztS7yj0};+Rtiz=vvq_QL^%2aFP<#!-nv5AXlM934LR;$Fk6c*FAu!0k@<sy_6j_Lc(b!9>%t=R! z{CRhQ0x$!u=tb>db`p@+NCvXxkJC$jcUcZ5KpRR`wy;`YgS*$x_~9+cJGtzxjiuQN zJ`2)ZE8*(^J(@>u9msEOQ7^Zx8bE9XA8>b>eT&=|_u;}692jjt>A$t7oO(IO#5_>y*#po9_nVx+o-AyvD#Zk z`*xxz_lPsiVvvmzWuxAogZsu+EUxOydF~6%emg^fiPrXo@Y^sinwyc(tLVY!od^ZU z%ft6J^aEULo0*Cs+d`wWKCNv!jg|h9czSSNpLdPlE<1CqYA<{TaVE4pffl8vw0m#}TA(_UN*h*We5&kQU*?TX|Vfl{JKig)w)aMb(PfkeRYU@>hRhWp?vS2!*+TX^!a`m|sg!el<;@;(4ZQRxw zvCONghyBFQR*M~RCsUWiV7Gm+Fum4_`=*x${ctEu!rLOpbPl{v7NmG~qaWEmBs7>Vi#Bo^Q@ka^i+`?%?e5{?t&- z#(d(U72IbOol-V!9vo}GAEScVpY%SFxfSrHWnodv9%@DI-`le|57pV=u)W&0*xjGF zQ{+tZ)wZXVnGSJCC>*|swhzQAGwwpa7;=0G-eHEf5-=_~Qp{kQYFu-ojdf7x#OD}9 z9tu7t_KL}|Il6{U#H1T$B=KT6!)xku#ojA#$l+ISxf7)1!!Pem-VQt9%(q+{Nq=d? zUhZTnhjsfj;9|IMJGW+n`URR+qY=%fd;N&U3CeW4bt^X~XLY>kLNRNPBeM?u+Th30Ywwz7Qs@!|t1#VLeu&h*|L-{z05}7(slc1nf7|1H)8B|xy9Xxq&+Xvv9O>gg zo2{Xpm4mJRzXsk}o-=PomXQ99Y^y4hie}#EO zkysdLDf}cz%Bp|RkK@)E$SvI9lFPEF!%)Sc$``>jUA$Mt;H*i_-JNmj7E|pfu_fKO z+-i2IZ}dgIFRc-U7?Gk*^%w8;C8u9p2L1?6p{xfXBI*RB2c*~AJK5-CN}TEX(U>-X zR7`3wskHZO!5FUH-{hAI6pf09gSZDydyi;kl7to0ON}m_r-SX92eo(87bo#Q^R1|o zGsfn_5FCZtenx-&j$Y%-heL^b{iHTh9m%-lL(=qOyb-I&3HOV-;njWPTP6(}9rO=} z&g?p+jH=tGCT^aXx2An_-e${~Q+!y@K~wCh*MbbvV9K3v>+mGUpPQ=rKv~-#CGcCs z4)Dd1BWhUA%TK+?N&TcGxHd_N^%OD)Qr-;GopV^Nhdvz))&-n7VABp$(XhF z=6%HVE2vxvDU$}oH)+tzdl;#QiIJU-LwBiZ&CBv1$SoWu+jm;4+dCIR9F+td(i8)B zgb#Sku*YCAnTO!5Fl2&inWGlm3^AMoVti|$Am!T&wDlu_14&F#mEVkn> zXFu!kN-moi<@t5F$Wd5k`gTYDDd=2Gv(3ll#0Giujq+qKgI(S)TQgnGJTFV?20N)0 zj%wgmFQ9|a(?Li5f_nT!p1)2P*NU;A=lf8*6J3kavBTUlku2;g6e1E*PiCMfh(VF- z5_q=2tmf0sn-eBaMj7Hy|5?KzHxA@nt{QtgCao*#gRa&A8L8&z{0>{TuP>$ z59!%zV=(Z3`BtJl29Y}f073wAl0nv?XH0h3>8O&Y!E@idWFD@#sK2tkG$`FrM*n=$ zF)4F)0Xsj^F!@D(Id=FeCae#0Kbe{{Gl>REl0#t=7liXV@+7!Kgr2l9Ri->Y5|=j? z8RR4bN$ZNo?qZDc7Ss_5^q_qR1!|7Pgr;?pmT*gxRX6-s{$}fx=hQVUkkumnt;FC{ zk}ZUiJ6LCYLom_Ro=B46wK+c&p69b~)eFApS0#f#hMk9+@fe&Ua<}JPduh+vaY!;**r`nl6e`KwPl;=jnV1NU%4`38$G*f=a&aP!SD!$#$T7rj^R;fl`R3i+eCgi-2!22(3xj^~3@ zXu!nJk%`M9FMBlvj_>BMc?T;9`>cL6#aot9&?@N@61{T`z`|AbJQB0PP;GdWML%+xKYT%(P1w2V3 zHKXr%VXy^iniEM)M4sz`%p})78)sRtkQSC39ar5FA=HYt&GK4dB{rNBa|8n~@4a$A zH3}djv(J^BDhx^Ib-NE68kru5{P3c|vDUWGq`Dx&ptP~Pf)v)nreR!U>MYi)X5OZQ zR0u4*!BBqFp{gWtc!YLYJ{K!5_M2ujbM@h-+wnOax;M{uw*8gDHS*iLjLUSKJFN{pV$xZP(C{tJBHR0 zr5;O3i3WxBcH|W+wMq)`_AH|>SyynRp!%Pb>(0FoIZ3XMf2DR-F7x%m+ZP`oDXohKO@~X3xsPOVD_`m{#milB; z+-Bi59WdkkEoyP(YqnP9Aw|r2Tjc5uH&?M37BB5V zT|`bBYMprA;4)LK8RC~C5ab2r0Kn+@_OmR(cm#1s=ALpS?b0*>+qh}Lri1{QY$;lUB+7x9vJV>6^5%Yb zW7*+=!@Ayayd(&z3rYF1E@kW{j9ryVlw9c9n`{7zZh5I3slb{>gw`4jN9ATJ?JRFw zQ>%s&$J)0y*_ue+I#e-l3N@`)gQajCN+UY3DSq$WK;m;mv;Cmk5J)7Fz zbb=^6Y!XA%`o{m>nfN$bBp$v|ys+XkN{4}mz@CrDQ5)WK*B)$(erkFS)5Wo{mUO9+ z9(xv#(+%0xU^K}}I^m1MC~8;ArRM0y(CkKEBQeG@;jQy1Nh7ZYFQUsbgd_eTdj|p) zx=2O0>6e|3%dqYyFFh1Wkr8s>V{?{6Ch>R~M-bvR#{6!kj|d_i^CfnDxi&a43<~JW zUw&MjqF+7jXWy)sw6Y&Bzce62Akq`ZLlkL*>LF;hl#0qrK$v}}8g6Z+)r&1){(UyQ z5E#CPLZ!Wt|mW)?NgI4o|Q8R6CrY3{39zJcrA z7rNRS8XuKT?#`|c>?J?mW%D6(J3BIkiM_R7i~2Ci%uO%tg;Ui3wHJGa%UJ}cqh7_G zAOzCNdxxb|>I8Y8wrOm*fZ)yRNmXdU?}ZE!->Tn5jW^l<`zX+5;=n&IhD99L2!e!fA zQ|nFj^S_Ux=!IM4vY=ldtJgZ)x}hO|Cb>sPEjV{nNr<`(EqpxYpW!3>#<=xgw(!Sf z*e`c$XK15iWNzj3+dlr_t5I8v_K<+S`V4sLV))1X|G%9dZ{`*TxpIw`9rSREuxEH| z4SXLEd*2tl8*S4!@{Phwen(S}lA7VA(ZT*yb6n=Q*dHLKx44@hJUInYcc(ui4MX)jhp#chas4x2HIG76;RH`fSA z_BO${92E4z=V8mjo{ce0L#g57y~Lw=&Qe&I;Q*!csfGAl$YIJbxJ@ z&*rR9imFs&h%2jj;AtVOV2d`EUI-?Kz>-UtQ-iO2))NP6+I20?E@RJDGtDpo$dq6- zkYpGn-7K0p@Lzi#NH?ae|GK{#S4b}7>>$-909@4kDg>iI`R+euI(8zV!s>d|q-`TAM_9dNYm=H6GNuSB3j#&{_m+)}=vSxoHzaS;0hsrIn6y3o?e^<=;}@pZfLib-n%Q`hNY$WRD7=hBP#fN+t1 z=Ga-0Xz4(Mh}Cx;-jFIuQgpkKXOgpfx#!F7A6Heh0xvkI+)_rAWlnj^5XGHaQdUNA z=}1; z=2g!|XioBqaHtE4-G6uOM>_;rc%kiqK}{GuJgH+=zEfp1esOF5mZ+A6-<#f)O(toBU*SL(&lWg zXYr6&BR8Re<&sTDNct)BKS2L^S9x4FXK!d{uVdw4Z|z|JaKi=xMFISC?K*Hj0f79G zbolV;_q*QTE7^ZxJzY-z3+thr?=M*YE%5#w<8guYLoxZ!h6Mce1Ni&}<3FY4zw~*k zxcs62PkrcsKJD-Qf0dg5n00bx?|}gNjsMeoo~X=z{FcYHxWC?F2D}FT{Vl&$=Kc=#RI%$XD5YmlLH!2) zQ2P2i+*4Jgzu+W*RhhqarlbACi2hSk`a8^1Wt_iYo&$^A{|@t)qRwAnf9H}PDnx#^ z2Xg`g{_{K7f9gg4G@ZY9$Nx`f0N|lJ<5S)LRzLF3Xb;5p&-Q?3{S?}N>Ph~@c)FtG z0pq7J=lmnaqhrX!pTqbyPEddRO^V04&%a>hfRg@goPHzlfAaT#3ipwY1zk@xMMeqyE59k7ZSM2xC|H&!%9qg%K@-MJ{U}FF80Kb3!BeeVr>gf>H0~Fv7 zFP#pg+J5i;+j{s+|f4e3{a>z^?mLR>%FgDJpMKbGcCj3)zLf5Jlj z5vO|`@_K;%X&Z|F3+y9XwlMg8FdrW=9*D}H?LmA*x_=V?AEFh%6nH9L@c{f&0oDHv z_^}HAuEPT<|Fb>luuT6?Iy|P=e~k+i;3<;+0rRK$gg_qPuSWLgv_8iC-z0yamw&bg z$%lcJBu~HPC)xaKu%DuvA8>w3`|Q=f;XIb`U(MsME+YyUx_DZ^|FR*E$Mi9o@=MC$ z|3BDcQUBcx9>|2B?O`x!fI;Jbko7Obz+-_QlM4@sKZn)o|A_cl>VF;Jhr7zp_8_z` d&tHXpbaR0Ko&^B_a^Oz`Fh8dO_+mUL*HbG{NV_`ottu`!u=}vm=zH``Yi0%39bB3A)%neVyU!W z53Z+oMlaj~Ux?!0bA>qAn_F2Z#E78y1JK|mP8ody$u6k3D2_A( zt=`JuK6ZA7lZFwbI;niY3EEWNFL$SHlD`S$u{jK_j-#j}KaP})_=@#K{`T!uj5yjM zzh?P`d3lz?kF1c9GoxWWt1ltM4%bkBogG&cBAL+3tYlt@=HJiG)XW9qY~|=`q_M%tSXdI-#aTH>3r1 z2sK6Q)-HZFf!!avq5MCk`I9yAox+6!_1`zifnIM~tA*3%K}3bACibFpA? z0srAw8u12-uxx0->nlFhA-dh(F#LG)!GZ$^1frxAlpjd27S8}ZdvtJmvfOV* zI)n1|kTk<-bji%WAK_eY?!kR5sZCZFITPTM=1F6%BnW3#f^#>n+=!sbk?hH0w`Q5s z>z9KU5`Me*b@m8*-+9IzenDL66_)7Q_9q=D#wf0^v9CR0t(B<>tAlEbp1dmAxsf=} zN#=HR?>pDT32$V}aHF$d@rVtosqCYrbsn7=(gfav`ZPrW#XM%;val`R-*Qd{X-0oF zM$c_W75(B&ANS?g=f(FzLPuu`&!u=bsDCxU!ym(#cQ0!8e}N?6@2L78PWHd4`F(Fc;{*ky^2M>R<~F3lyj#KPUL zae)ES&x7cyW}jmFr|>_bTq`shh?O>Vg^k6W5NApns`+AQy>?#xG@Unvt6Q(B-ncW= zoNrayy~(!~sT5X>A2&TW8A&cb*Ic+l2}rU;ZbN%6vJHo>euLgryH7|fSX-q;G}q>U zvX&4Bt8bP3i@bmUoaNXVLPtU%ZR1_WfCN9~Cg~xGB1KDOimZe=tm81jyXt9HjJC9& zo{*+cK=zHSS+yr<;v<@13Pm3?7roO4romW4cAbBoF|WE$g;XCbrJWrv>)Y%Ojjmoc z_yE|t((~66&;dj`^bD#7Df9y?(zy6tmnWp0OBv`430Xn}-6A;g@W@6r(jTzKV8VRk zkI>wDl^+^08x$L=IN=m$UO_u@_w0?`WJC?kuHR~bFMe5jdw8}#IX)epS${v(iORyq|c&N>o#)XJ!C|==U*_zpo9ydo4K-PFsOL@RqKIr#$bIVOafy z^|3&0HIV=8+G;WJ$py@{xp#IoWK58GdlO-_as7uf^x@(g5v!h zGctFvb9jMm2WMuFKit+1Z2wQMtR)GBkt|) z*qO}?caUG5*qV$29-GH=_fb;^3e-0ZPVGD0JDyy8&s?=t5rq+{~b+BzsT)mAx0RcxfA(S9}2om7r%vO`E&m2e@Q{)g$c z+cY%$H(~1#S2StnV(P3Y=zQT<7f{S8P9fwoB10O?r(TPu)Div2svlJoi3HSK9w&xf z_fjMo9hj5_*?~v(VoxmDUHW1*ui4rB(ItGb`EzRb-*boL`1<}_{@mIqc(B6sF_lpk z(cDjWVxQtCZt<5d?RTtxoK;E@oc&IhWXR zn%av&Q{y1&Vu{s!i7XdZnBUCLAU~VulCzKE(WO!e!a)*nw*G`K?3}Ve#_~(j9p6;# zk(EMyaYz<&JW|S5{&aGV>=5kLZ^pD2ekO|Os!v82I1E%v7`D%&)+%V>3JG#>bU!5PjOiZoENVLGxRs`O%WB*-@5@z9?4@RmxoldD--u2$ zv+t$ifg>B_wu#ep8g^jIyrxWaq@$F{60wNiP!6P&%I?15Cq=*#K7DtPtJ_OUm1@N7 zbVg2j9c7QwBEwlZS~R{e*b?$MEI5L_&z-v;vQ<{XCKWp%7YM0i(ib%;(w&AU$4BlI zAK}l=pJ>xB0iE5jl^U!}rZyhrQSdb^ZG9@*3=uOZ#XJ(meD{8XauEZUCJIGRhEfiE zm}_{>oe9fL&^;qK$S8e96+cfRQqnJXhOg=)d|X{nQ!NNK3&xK-8{FPtXD5A&LG6Mv zkwaT$Wcb-MBd3>`)~ju<0JepP7ZX=k0bBBdK2Ml72eG3APjwAQs^BB*2&_}z`MmSp zB57@dOx%&R?CP}0iU#2xU}!d#^dN9`6e}nO7mLg;{grf020pEYUof91J^TZ(fsNnc zV8hnsxpwgZKBg+mXx+6O>nYS|?6%HEo_p+-Hym69Fa0Ft?Ptz9&*?BH%$Z|OB=zy3 zA3sazVz@Bo^y8%XP%-!$=LZe!g&zDwZU_1lZiSQ|4Y(ptsmY=2%!Morb)!7_Lt4T- zPz=9)osV0nX6FW%M{aowVO;?$wpW*wuM|86m6Gn@ee1?2|DvDY!uDV4P%3oFV? zfx7++Tcgt03G^0pMu_d=^L$70L>jVT!YaC^0$>v<;C2li%;|K5*-l+Xu8PU z7Pk33#~xgr?Uj*fAwGT|ue*Epn{l-HxWCRdF<4{y$Uw44SB~!!7JeF*KTouF?FUTC z)fcV{rPquJuO*tvZN75*2_MwVQ$3NI_nNd#s^i^3FNP=PE1}itg{lQ37l~Cp>+r)k zuCD^CMSATlkN317)O)Wk)i$Csv-Np;AN2=^3CV)>Lccl&lX`TjUoZ|0QBfz9ZwHu4hgPVJb-bznF;EIq^jb<0ghyX%g8*=jpqi2ZlATt^2NS7tXy zQ?RSqKlfVZF%|5!FD-?UUc@8IRYXV0d>)Pct z^w!;y;K1#c{c69{SP#=EjKkZ512qC9ihT@6EeMu$f423B5;q)cINJK&VP4q*yMR5dt!jv0zI7iHxLf(wN`}LqAuT^r zGNs79hjM%VvUhwx#aVN^Y&(kaSGzfH0HQ^|MEf=fzZvoWq1=B&{r^;~ojTZInG^HL z_9?RBHPtI)UKt^CH>fWowNN66mC(0&%m}*tI+@g})bxBTkx#ds1*RU*4+=PO9_O9j z9u^MO3-bm8ZQP9GmWSa+uFD+qO?P^cX0-SCtB0*SorT?jxowC!edb#UMcB)%(Cp&?9~6O!s7E9RSEeeG zsE=zJ?y$Fqlb0r^?-Mzqziahq!E7pVaw18m77;ySCvIZy!GCs?X}-fdr?SWOiJs-) z)evk99!KG5jL#Zc^jJ&J0YAgnbp++a!v!|0!>{)yexvGl^kqgsTDiAO#0cYg>I3L5 zyvw-KD3ZIMybtCFB+Kc9=91{cBO(|RoYA|t1?O0kT%e`Q3L_;s)!Nm^1lZON)23&C z#x|98cmu-KDY*_|Vfl(g=Ch?{b+k!uG)0)!%!C_s{=uLC?=8+X(zdG020>@jAy5{3 z6T{cYTiVVX4!-O=G@31>PcmEZ0U;~K)!T*Ouq}=v&Id({J#s$zTVmSsf+3B zOF@5O9@Mn@hS>bB@|P)l;?N1S$-O?fo(O~}Q!z%3qk2hy)oI3E1DEdR6?zcn7IR8r z*VO&g{dyGg)XdTO?#zq{bzySeh*IINr>pzDgszz`Dh%x}6ki6%~vnDI)c;l3uzB6jcs@Z&Av@qXhETWyc(UbR!=rN~@BdXW2{IX$* zS_M=idXAICZ0_3e<`&ydsfx=X>?j=sVJ^pjM9-38nXg=i9gMUA7O_0Qwe%5Y{XqZr zeDa5Nz%Ns%?{IjOV^lSB%Vahp_>R?|YM7T&n8zrMGhf|KD8JX}`3AF3WiaYn(`eU^ zEak zfZ2EcgrfxT#%}RuPfzV6JS*-NCd#wp4H|hkByqpAj+?pj_8Sf>{+X91rLZf{vTBgu;!*^;m1Q}jKu|jCbl3p%F z91+$jdh5z3B?%CbgS=F5#Hi!~3gJPhWEa4q%|5 zX5oJ`w|xOjNARDZ`M;G)Le!^JmjReQ_FY?!z~J~nmQ%en*HBeib~V|sLI;9ClN<=H zo-;K8%9nvVDjuucG|ogssoS&DQ~Q-wxfD^F;saMc1MUz0pWkVGL<&)DHdbY}M%5Il zs@o&%d4p6|&H#QXb--{PU+X2Xx86p)eg+5kKj}1(hXyyER6ssdIy5-3xBY@9+Tj-k9`Y8S_$h#5X3x+SWf# z#rIP0mD|M+Aj;29^KNHF*uBm)jU-iDYuT|(7k&G)T8kr87{6esQ04Gq`O&t89 z^r#btw(zZZnSX?iDK#?rYjhdE#cZ9bG}!?AW>*3IM{!6cot)|psyKz%U|3grFC$sF zV-b!{3Ps5eMOIc8nufl)pE*WZVxTJAp0$PgAhJ=aPa zz`VXpSciEl@T8lVF)1y(OsqcS_V`&ok-K=Ht5GW!t^GnRxD zM)$!GB|#r^EW#OAw|zj8xX?<8k}~0#864Y+EiTnAebJScnmy7zia2^9XTa0@gL_Jh zFaYV7V}q^DABzfn#KEW_ez%|UtU}S=i1r&K3yzB6P?=c<%rs;ZUE>=ONAfUK*9y)8 zLg5(e6Nr~!R!>cT4}g&DQJ3DL6Y@Um6FYN5x*<%d4EyVZ56tw_X5O+uE8oVEqiVfz zN{$?Xu4_J&bM*6Dq0!Z=q?X%2G1P_GO-G9k11}<1U)aC;4+aLxedmk+7{2%q;CDgE z(apv3pW#M``nt^u0L@qM8CUmgv1wA@ucN9lL+CIumRm^pJS{F3`8eq zw=I#>(7vYe zvouc&Y2JZg`zrQ+g-l*)R`0?$y`HjqS)w|<%=g_*Ki|g_CQiGJ?~6w3Q9wZBlqK0t zN2_5T&v8YTM~M!=r&8?;wu^|R!lN-4gIySzz4kh(jKaartvHqRm7^GH9dClBhUF=h z)uo&F(=2Qs3cbzaB}^p2#0UjL#ccR?Y?w}g+osryoYQKlBuGljI-KhkX;3WDZ#5#! zyqlV`><6_y1L`X+=w4_d?h)B>0TAuj1dgOM5aGv&9j6z!V3&}8o>i&A^w{Q81`$N?Gwh3AwMJ7-c zPAF2}5l@@H951lo1-g|O-$YS&P~Vq{2Zg#P^6#4j(9|{sErOo|>fkAo^fGg=rst37 zI(^dFCgm0*M?_qg>je1(ce3QLdRHe0J07nw#dev8wz7}s&- z(zsbO6O?!%&W#^!WS&$hU4rhb)7llIYFE~AO*5#oE^QPF6wB(Pn|Npws2;Uhs~^;~~< z#4DY-eu-o&;&;Xg2B>l{B5w#P4eTi>KYGE>ZSc_TSa>Vz?d7*5%)6rze_7L354{(Y zRz)Qs(lC?5Ok-$Ba!13kF0Fi{V?93!8&oyDY=skIL0idTa%p%e0Z>#3ayg*`ulN1lwRBCI4?9f z-a-gI6Er<}K$Qfqo5s@oK~W>6xn3H(>@-~@m_*)*@WD8O7a=oyJUcFb?b7M%^Dq67 zKs>^5h_#hCF@?Ls2f(DWYKp2Uhq4U+@G2zNUjg?qTV1pj5HCdS1&M#l>gB$OYNCiw%A0iLn46bhYyZG zEzRPF8x?MaWzTLEe+*`PMq|rmv!I^;5u9`_-K*~TrPBxoDj15x0$LmSHakbQdsqyz zp(Dib##=9%>Vv0#;oC#BKeh^5!zLKFg&%JUH;fgC}Oy_HakaB$iR~%2QFfb z<2~$rW&{hY14UaU1*j&4kVESV*5NE@{h5Wm8-^{p-pZA+r%UttXZsKh!`BgKc0M02 zV1z9UYG4}8+(x}AH?(*@NAi5B- z;x87Op__F{5aeMEGx2zbnk44OS%7?6kvn%ggZZ}c+EkNw$fbmB!KkU?7s;L2(ar07 z%MJKD!FNMJt=}DLPVR3kcXpv(e>vAD0)h?}pZn-DJH(SnVaf`2wdd|SRo^CGkL>mx z-<=!abV7HQYv+Gx!XVA3oqb=9OKcE^$qUp+w0l1GP&iuQoJ5U@t9TwDt6^EoTJP}n z#*$dgJ8?&Oi``ltY&idF>yqCP6_gVqj@$^DR#+Qq~jhWlJhH<(>`9A!}g?C z8T-<^rZ_Lar+m!maBvW zMO&`eKQYXh!REx`>H;1~Try=QA=n{_vPigFhZs~TpWUODl-}WRo~nP;gColbfMU1x zuX+G=yC`}(evrl{&tl)#kt@TO5U+_x>zmtWpLUH`B+r84EJCN_MPXPxC7N>lWuf8* zy}Fv6zT_CldSBSvwuo%h&6IYqPE6@fk^z5hzI-`;w%u}v!-+0!Ko(GG$Br0;C`eRI zb(Cw9*4n!VWA(=1oifU?=t(*ON>s&AU2`X-pEk= zTN{=1+sW`>tOl3S@qC%ly#XjS{OIp1ti%YOHa7H&kw3<5}uwQ0<)`@?MZWO;*;n`N!-1>o8mN62!(1mJ&chdSzB%aP@5HloSw1JZQr zD_hNq;U0n{6R}0zz#$d8s^GFr;1W7NWAe2^!*H{fGR-TU@cbVkp^3dcyefPr$%&ST z8?vSug{L9Gh6TlV!z&nb6(`hI6I2)52|3oH-!^OnEMW=}3j2nMoTgbt!{u1ZB<}AM zDr(s`G!&lFfvS94u>I=jQuBE=J=Yr;-B40F`DUCL5Dh6SBXi zH8TiE7kh9}Q0pkaTWfhqX_#Fdm?4&C5E}OJ38!eI~!kgd|4>!pO{Vx#lf{m_+n zE*hr1NGg|&3K!wt@XCIJLRL)7NuH>&rj0qes6o@ujzllbTPI6JHt#n^LJ%+Wn2Z0m zzPIQ>AfhvKq~Hp)w5&~593kw5o_TE(y5n<#xD7i*WAojfmC)Mk%yPq6vaKI3BFA(%zyfd%a1+|*rlIF2lJ;DqYeqO`@x^LwpMcXc<~rEB5z9>|askfKO5~ zXB5sLDp*h)Zq5~Ut~DQ;-@@(oGEv$bouWTja^Cp!KJnkv;{up6FGSk9E2Sz#%9~<; zc2F0jZ5vZmXqIgwR(w>U7rflHoEJR-6|T6oIU=sYexoGB%Nc{KCd!x{k8W^Zh>E`6 zyN(?A4y&HRpO0uuAI&jmw8Ml)YJ$zQY)J^~U8q1{yNGXK z+pSr*3YEaXZ@YwgYm?vAaIgN(%@`|oF2D8C5;l0M{@kUxlQ>+3y9bsy*`M)Qc7($2 zT{Y8ZxwWB${4hrj_f^;_ZC)y`3R9t*mdSVX+i>$2hNjElj={^w;SN=o9Cu9xuvAfN zn3z@+(9yG$W8&DDL}OkG(}647M#^7vFtCb$(mU_8H-_=)z2vw zXpRaABt)JpT6s)p`n(+Ocu5S?Te5IfmlFBY%=bxSSahHylut& z`j104o}3l@-{+)MG`t`!6@v5H3Rqcv>`WX|Ei)=*IvpDZjgM|SdxhQtZqkj|*|sTB zniCt})HEvChX+cXILL&+=udi6CaSy!6hP`r8~dMbHA>3z2~c6d)80C&OG+wLsde>h1HWhhAWys zzOrLgIQgjxf_&ceKs^#jUN5o8jCtKYJnt76Rys_pr*ZGk8Bs%<`TcEva;$tlec6v` z!QwS~vLBy8{KXrtCw|TIS0IPOUE;g>{w|qMtYmHz9?=TR0z<sRQHGSFI}oUj@mI_xQq2gFn; zD)oGnMJvyr|b|n+uI-}@V54sd1e`8HEeFZ== zubK%=p8xb38!)bFnw=2H{x&I^NHU8Tb3$u{r^`Sl)EPw`?@C%O_1XgpWh*+ua6NvN^h zC-07U5RL&B^Bx=2LGYm>AIrnkUC*19b5k&enjN&JGwd&!4#!UOVoD*@MdjpW%5d3s z2ZTNK%6{L!E;5(S(H(@(Y+UE`q_idWKRM!4$-5A9!ilNtVfrSekT*yUg}i-`q!4Fd ziI#l(@Fj|a?T1PN+bI#8hueCw#Q7HGMmjZ-Gq*?OIw7C2m4uMqH0_VCPPWf6Q*elM z&RPn+F{)(hwgE9t7s+_#ku)b3AJTapRSL(@NOrKRg%o#CR(FZg0ugc)^K*2Pd`Txa zOaa%sbw$5$=QSY(NRLyUBk^1$7y93CMUoZGDe)o(XV1_K1k|Bx7#;PATzgq=1u_pK zvXm327Jo+O-o*l>3g*fCipSN#XUF~X&&|3dBzA6(VK&J&tIjBguR1_|kzAQ499Q09 zYjaJ!#l9m)i^a(Mrv!nW2yQ+P8RSFSaQfK z+=g6D^h)s~qXswZ;TX)M99k?4%s*t$_Qr@X1N|8-9UHJ6H1lF&?@UiT5VzskxKqo~ z-n0|fkYZ#Y%?4TN#pKE^o_{j+*o=5wovlIB)Zs>ZZaugIR2_dENAp5nH$<)Rb8_3Z zLwU?x6#RUPdcs&t(zEbLDxbi3< z_3RyQ&AjSEnVj$(Vw!=S-^k3H*+O*cel}&-?S`=KO%XDF(eH+*82J3RD-6%c;fDGz zr|dT`MQ`$dk8oBmcm3bFT%tLf8yg$Lm}Jjd-Ef*KC0VPM5Dv?g!g&L_MYgfwGITC_ zGsP^<4KUO{&)r-0?n1(Rd2;q0%rK|rWSFJLlaA0h(s0_yTw z=`krY(K!9szre?PGgtcA_1FsreR}LOKyL4V;e32BNOwN{{(i)hRH2p^}+} zHu2MsW8jtRwnCe zm&9XnnP>4-r_AkWH1D&%pJJT^e2s60d!y;E;K4LQG}XSeR(PJW%Lsb<%b^IKp(36L z>U#{ki#=^`0%0Cumj#?0{^vD;5u?)PF+`>nk-GMaUt8PC2zf_v}}fa#;Xmg4O=vb2sJ-0=#=vqjiU8NsuhuE&~3{BGc5M7 zb?{8uZrL;gPYT0TQy!#J&1LV36?crBHgfZBDur2-JNm`fMlHIU&|y|O-E9BX@9^S` ze?H_d-~LU1ij}>&!{1sHJOsuls$NE2yb$egj5d1*v2`;wV}5BW=>_13-({QRMwKtR zv$kOCd%`+lE$471?0nn_Ta2;bNRc)MR0K8z6}PqmFL0aS?mwW*~)$IhemPP;JSesP&t9wb`rqN%KfH0 z-XAxZdE$?kd^^Zbp5X7MvuMc1h=n3(lJj_5tH_NX6Y{!V7y7Nl!}l+(g9EocN3Egf zXNdob)&6TQ2l>C3*w(?~@2wo?{D=;YFQbZHi0U^+|IuO+6CsBJ!-gij;1$V)D>AJ# zO-E@jAUQpnZ_LLIxRV02Z0HHBi9KQf?xIcE=IIMh_98RwlcN_alug*?6DVCt#N2)W zl}Jbf;MiTjURB3U<4?$W#)`yn$Sic>VMeERL25*HHV-`Oo1H`_`#=7?SNC7#rT&c> z|Ic2S|GUZR-{a@DeU)SRQ z(H`|L1_0on9{Ud-rTz{of-f)sgXDj25-&X1OgKRxpc#*u2^R+kH#-N=loMde#?EPO z&Iy8WaC4iObFg!8zk_C^msQXJuvq{=0029i3IGTIvb}ug1aZNOc2;gZ@`o}ko3`fE zF@&J5vsky+c=ZW=bCjio%FMwU{MTClf0vc#Z?b-+6Oj1t%F4sa$pHWXO*qWV*uY#I z5Fk6dDHzDfV-A6EngKZgJOCbcHZ~rv-;wp8Rh~tqI@W>2VGK25H_~`Hhg1_DDsmeW z`#+YI&HQh&hWCqr{-y@WkcDV(ki<_$9E$aeK06c z@F|g*<{3+~YKl2cI?_bc5ZBu4r>7q<7h$SIw}^rKJ%Gc)r3^28mJZ3)I2kUd;t5OJ zrl3-2qli%T3Y~T9Ohs()vK7!$YV@*2Q_gfhT#Ew)yG$yNLDidv?+-OKl-B3C&^Eooenn`02R7=oX#Tq`LdT7$^N1qM*`qd3pR)p(~Ya272&J zT%y|=Z{6k>55#@W5nuGng`}M(+36vGX?Q8R^7y=pq`DsNwoL)Lh?^hpP@;z=m&*B< zPv?g=JPF%)s^8`~I`3jf&YDY}QWR}g0(xkyE48e!P;Cub^D4MII+(+_)aQkij*q1T zm+Dc3Sd9JB@1q{sJy1=fL;}?cVTz=89nZd>Pgi@i3Yz=!&FZS*hIu3{rkR7tw~o*4$$GBo zQQw92l{Wo#d*S=3UEd8FkBxJC)>|ds`Ihgh@Kq~a+mSRbw!psk{p@|Wd|Oc4tdx&F zP8%s;GlRdf3>nes5|$&mc<~Zp#a4U zemKy$tR`2m4?c3IiO7_Vwd|@r=;~0Kid^FJ%iJaT=y@Q%fr;;P7wr#> zR*-~|lfHW0>veDQ)_mz9XFUOxot6F3%rsLk(J;iY0imax*o$NRNRz!jd3S`_4}AsA zS@t)ZO9yFKp4E{A93j;o?W?1bXOmUfHNvC2hR6&F(nfl9K5>kXFg?B%(Yz$Xwvug2 zlH@Kzl@(9_U7q+ot|wB7s(0|xc13Sl*hR=20=~z$;v}lPjFnuDAE`*Y zE|d$YO4O3+;09DmHKCBqlUR@LVrbUYP9|Rr6 zu*%sHD+FsGBNwdje?*USTS_Tjnw;f}oup)DQ%Q3kQAB#CnE!ovSS)i{&n_Wm9p?-7}e{6%ZQ3&IpSaf}r+wz+{D-yXlk+VwUa&sPEpap%4nm|EY|2A~`4;j@hDD4fffmH)FJN_J z^)~oB^iwfPauiw>JdK*wTqTkUPWm!w-jgH;E5fzJ*q%BJ-S8ET3tZ(YVe%@;^J>&A zz*@=$One?ecB$~r$>;(fnq=|KT^1iv*jwRyuT}PbEj}La^)go@uPq53I-~rtrhuR5 zH(sB&?83V)+f;QqlL>v7tgA4Rw-|n`LEEy|6U6CbOP(+k>K+ILQ{wCFAxN(z@C1)) zyoO)}xCJ&fy_Jr&*+84oh52ABKPhlN@bqhd<@8R8w0yF;P*AGubqenaHAkeX6q8$r z+!08Yq+xczK-EI!=(XJYMQXRM$Ya93=N~DyF ze6Jqz`*n|2Ogv54WGhc7!AsrOE!5B^i|yb#!1n1ziy-`s8`p{cPo^T7+&F0}X7{6n zK`7KceM1&--9X6r@~S#hW0P%#gw0RaE(xZd*s6CCV_gyo#~0X4I*z#?te%@t3~aKR zxIMwTcQcbYlvDPEqj9|%`chrCdIjL?!Z$nn_2 zXqYn~dl=(MeR0`HFe2<}tjS~R6Aa`l4&VrFQTkNIWGwBE#ddzfOzB9OSFZSRb&e`I z&;rvX)JbegLd08m$I#ZDqsb;5R7|-MtT9=TAho08&WMDpPH~wc9KoGR1;I zo*UivxN~#$`HfHg~2|_S2DU54TNtiT*-a~XCtS4raqM9=u4g#o?Eqw zEk$yY4VYV4ASb_D-HzVsFz3donAPB({ee@(l}MM8RHh#?lE7#&mqS738{2I5`TNve2)2=_xJv$8;*{4U%#ilJH7Z4uUxWD@U){);XEHw>FuI& zrC@h+AYExA7Ymlw?#|DxlcSHb*g+Bt*Uq@AHrht;J0N{Jk9`Yrg@8<&5;Td8h{=wH zDu?L6WHCN*cG^492IbSq7Bl?v7xx)fa@)p6C1K_5J}(r24mFPHHSN&zH|eE1oeFSb z%TsghwyVD;84jq&XBQ75=kEIs9QojC&z+X^oQo*C=1ejU_DF+hH+S&bB!DljoPVe%u~31j{%DDDKF zWRCi0g_xsxpU~}m?~yMoTYLh-TvEl6O_ZmNAm6jC3|G>+osze`MAgbE#1JX)^&BM9 z{e*9Ee9@NT{tKp3zhWBZ?~x zqubaT}W({8i*?{5)gXs*EucW{sRt-`AS@c=c{naa!=bAg?n$ zOZ^HoF$rJ94HICx9>wn$jABGe>R+wh#{zJW{o;*((C=F+*yJTa;$%1D<}!!yZ~;L; zFvpAOvT?G3z--)HY@9qMCMFyl|MEx>8w9`x1rrjOpJakCuKwnlV#dX0&cO*}2Xnk2Ki3QV|I3*EP0~wwL+}5X^4IWd;QPDeKFz!~dZ`wu1A2K7E0M%FuYN{~wDA2K{YWhtFiy zNnUQ#{XxGe>I)Ab1Oft>vw=BGK%5W|m>bB$!_LWWV#34CWeVbE19HFY=b)ErN{s4| zVh8}oZA>e(dk-s_oQ*{q#5U1{x}A-)#2`b)K?3^KNs=3qwYs{AcDEKQ8vCxrp{5+= zHC7_-SLr#Qr#F36;q%-jrbgWY!`z`@LvwaC+A+8q$l8L>B3*G6;GOt#qi_gbe}t0H zYcN5JIv@wFu0lE~c)Hs?sLxQc_{__ZBI4VYeT>bDV4rLX^9bq@32Vi-D4&~@)BZ4O$oM`U$-b@NAMqmCQ^d)zTmoxPkdACytZj!E#EX>%>V zuE0k*CV$D0+!MHC5ut7{zoz8J-_1F2gCN`HMDf#mS57_<3zzlS|C|7>npci=O0P_41nEa#s2eJr{V=a|elpmLliQAu^hOLLTN<{jb zO})JT&*wMs*2MI0JS6`IY2O%SNwaNRwr$(CZFSjZm(|r}+qP}n>auOyMpwQ1&KrmK zo^!wV?%pHD*!d$@u3V98MMUlybKWok`g~dWqy5R7`Lbcm!od19mozeAWM(loG&W#h zW9BqsHDqNoHsWAqX5%nqGhk=_O2N!X$*=!Y5CC|OP(9HL>K{l5b{EXhF-0ZI-Z?v>XOj%865vWD-|NWlhz{YtaM@ z_q_3kJrBR|tigv_W&i_o(SyN&)??rwV(vU6{{;2k3Jn?mRq@LZl{B~6Ou{LAa2)vw z`6%N`B;(uB?|tpcDw;~(7xJ<2v2y5eY3OijON+w#j(H6&8-5L~Z{Af+u55aI-=1F_ zSt0JW9a5TuEpFLP11;Woe)Tc)a6$(ul0($7Qxu`ikG!?AoCG`KvLkK-{;$KRNKTUmeIxMROiVAqN6La z3&H-zUXbl4Fnjqve-WhN!gxyUBvm$A?T5n)dnW_ycQ2b#uFC}O-1)emZah&74uNDQ z=hy-&Xn#TcPC;w`UW%x;Ss|?n&lI(Th??)#d~$pA-Nxv**E(KUch&Tq%-urd3w*0b z%)jqNv2?^3$@Zr$-dXrKGuLoo-e%D!r3Uzwr_V}_F^Xnk7hqpE6#0L}-#c9 z(FMZehX<}f#=JkUk!JrGog9~D+Cx+&fjaYC186TZ;BVO}T2@&C zi8IbU6E#H=PrFGbwLHo1h=S~ z6nD!SIDMsxn3>^wWwC}|FYRsGZGzp*G0JU&=;veenoTdFnvA>G?u-Otm8fCJ#q#^w z^XOqq>Ti%D&WR>vt&hH8tmwHJIGklaW$QE_yD1(6W$+VGf#|{iGJRab_MYXJcuJU* z_zJP={3RJ~qXq-Of&2*7Hn=nHftq%R<9MM))YXm&^25i3Vas%(lI**O!h)wU$e)fj+2`T4-2Uc6{$-LWufoS+EMIpi;q7I zLK{Dclv92w;%|7ci`d?j$TiEf{YF>?ONZokkJ--+{vK!?yd(6?b{w7K#6wa}vI2~3 z=;~)gL6QM>V>2cbwc3LpBN%>>=+?OGbw7GPLb}pGQf^hCp?ia9t&!L0TzVWDZN7$7 z$xk*jS{7|Sj8sX-hRoY2LOomjFf1oapOY8upMV57n^L!k1bvaO=3=Ig1nqfp*_D*M zO(6NIVt*lFvw{X5-bC=ZFMo!S<^ezor?U#J(!RkKF_o-(e0!ImVn0`!;|=+ep$B{V zuwe31wF<|44iI?ZM2Ky{{41}j4P#J86N^E(&ISwip#J;2kfPZO>;fwrDu_7K=$eah zWh91FT-eFM$KqXPr0}IY%qg()^->`m+8g0e3UkK?C%?yz7OiS-* z4*FK!y-;qD;KBuXjk?^CfIS!azPgyEtKl(95PU^aR^RxnI#BQ&ORAbVpqLy!7-ofk z)qVVx#ZHdUuQrEXYf)>qXtW_3AaAt+k>}TJa9e-kRBljaYC-JM3g%as`0(G9W#tWf ztd#Gtf;DG|MzxKB2biKo(+=jr3MporSQgE#^?@2YT&9dF+UPFzaVo)5q1W3^QR8Gq zV<$M^p)be>j*GJ3#g0uQvk51^_J@-%juex!^@()t3om9PgHSAARG@gbF8YVk#WQP~ z=#W>VQs_%VlAOs@+Jkj{yK#A z4UP^P#2p!OtAJ*;jd%A^Nk}v7Gaj%?&E&QVIOk==1Xiz!sWEoYDCOxSgcqzRoB`$2 z4Fran+fG3`%p5`sbD9T)@>i_cF(Dd8*m5PtP_lIW+&*9|`$?rsi_Nyde>SwSTA&L*(00t-!jf)ft;*M{KIor%tzp=wWQ-lu5dq*3Lkzw2Pqfb3` zkRzC|vMA>?R=8DxrZpPugy)Y3QP}fpr=+A+>F6Y%YuBgjuAHEIUr&J%ju4R3=pbIF z?TU0`H#0~&e^BR){TP}PiYQJBw_LBLWnFDOF%|b-?&W;mWeD7d`ozcGT_QZ2$XOIC z?cB`b?^r$AG32&=7^~-y6G&42eCLlObXnE()p8ht!zQ`KY*uGmW*MWCn7n9aw#i&$ z(Rc-EP=z2T&`fKtr^;wHOkGTLMPs(hB(T>H~+N} z_o$beYNw*+{N2xUg&tj?pXj7Z5Wd%qfoOMTn@6V6EF>i*juRVp-Rq~7Z;`}FMyO$! z58ysS1D7)R&}?_AHrF&1W(zO!w&t-v3`Q3hi`q)7U1P>)WnXbVTad8OVlkkTU%NIC z$OGI@arA)tpN+HWpxc2Qmgyae3HaElq9=+_<8X6dG~7v&TBj?#*>_jdVO0 zC7l#eV1oG4&{+X(`6g3b1kKWmS<}8|xTFZK5MrTBHbh!&bt-9U9hRGR@oF?FP;|DP z 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

Ei(XwNOEtpM1u_&thFjw8IQWXi0_ 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 UqTY8I_-Dh4fe{3}4Flu<0l%(&;s5{u literal 0 HcmV?d00001 diff --git a/data/word_cloud.zip b/data/word_cloud.zip new file mode 100644 index 0000000000000000000000000000000000000000..fd395c5bd2c2770fb8a1f2482d16a2a3cd9f0686 GIT binary patch literal 54076 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 An*aa+ literal 0 HcmV?d00001 diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index d4947a8f8..a1e93be40 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -117,9 +117,22 @@ class Engineer(Role): action = WriteCodeReview(context=coding_context, llm=self.llm) self._init_action_system_message(action) coding_context = await action.run() + + # Get dependencies + if guideline: + dependencies = { + coding_context.design_doc.root_relative_path, + coding_context.task_doc.root_relative_path, + "code_guideline.json", + } + else: + dependencies = { + coding_context.design_doc.root_relative_path, + coding_context.task_doc.root_relative_path, + } await src_file_repo.save( coding_context.filename, - dependencies={coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path}, + dependencies=dependencies, content=coding_context.code_doc.content, ) msg = Message( @@ -347,6 +360,10 @@ class Engineer(Role): context = CODE_GUIDELINE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) node = await WriteCodeGuideline().run(context=context) guideline = node.instruct_content.model_dump_json() + + await CONFIG.git_repo.new_file_repository(CONFIG.git_repo.workdir).save( + filename="code_guideline.json", content=guideline + ) return guideline @staticmethod diff --git a/tests/metagpt/test_increment.py b/tests/metagpt/test_incremental_dev.py similarity index 63% rename from tests/metagpt/test_increment.py rename to tests/metagpt/test_incremental_dev.py index 25769ff6a..a4131f1ef 100644 --- a/tests/metagpt/test_increment.py +++ b/tests/metagpt/test_incremental_dev.py @@ -3,11 +3,14 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_increment.py +@File : test_incremental_dev.py """ +import os + import pytest from typer.testing import CliRunner +from metagpt.const import DATA_PATH from metagpt.logs import logger from metagpt.startup import app @@ -15,11 +18,14 @@ runner = CliRunner() def test_refined_simple_calculator(): + project_path = f"{DATA_PATH}/simple_add_calculator" + check_or_create_base_tag(project_path) + args = [ "Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator", "--inc", "--project-path", - "data/simple_add_calculator", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -27,11 +33,14 @@ def test_refined_simple_calculator(): def test_refined_number_guessing_game(): + project_path = f"{DATA_PATH}/number_guessing_game" + check_or_create_base_tag(project_path) + args = [ "Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal", "--inc", "--project-path", - "data/number_guessing_game", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -39,11 +48,14 @@ def test_refined_number_guessing_game(): def test_refined_dice_simulator_1(): + project_path = f"{DATA_PATH}/dice_simulator_new" + check_or_create_base_tag(project_path) + args = [ "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores", "--inc", "--project-path", - "data/dice_simulator_new", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -51,11 +63,14 @@ def test_refined_dice_simulator_1(): def test_refined_dice_simulator_2(): + project_path = f"{DATA_PATH}/dice_simulator_new" + check_or_create_base_tag(project_path) + args = [ "Add functionality to view the history of scores and perform statistical analysis on them. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores and display the statistical analysis results of the current score", "--inc", "--project-path", - "data/dice_simulator_new", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -63,11 +78,14 @@ def test_refined_dice_simulator_2(): def test_refined_dice_simulator_3(): + project_path = f"{DATA_PATH}/dice_simulator_new" + check_or_create_base_tag(project_path) + args = [ "Add functionality to set the number of sides on a die; Add functionality to view the history of scores; Add functionality to perform statistical analysis on all scores. The original dice rolling game could roll the dice multiple times and only display the current game result. But the new requirement add function that players to customize the number of sides of the dice and to view the history of scores and display the statistical analysis", "--inc", "--project-path", - "data/dice_simulator_new", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -75,11 +93,14 @@ def test_refined_dice_simulator_3(): def test_refined_pygame_2048_1(): + project_path = f"{DATA_PATH}/pygame_2048" + check_or_create_base_tag(project_path) + args = [ "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8", "--inc", "--project-path", - "data/pygame_2048", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -87,11 +108,14 @@ def test_refined_pygame_2048_1(): def test_refined_pygame_2048_2(): + project_path = f"{DATA_PATH}/pygame_2048" + check_or_create_base_tag(project_path) + args = [ "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores", "--inc", "--project-path", - "data/pygame_2048", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -99,11 +123,14 @@ def test_refined_pygame_2048_2(): def test_refined_pygame_2048_3(): + project_path = f"{DATA_PATH}/pygame_2048" + check_or_create_base_tag(project_path) + args = [ "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game.", "--inc", "--project-path", - "data/pygame_2048", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -111,11 +138,14 @@ def test_refined_pygame_2048_3(): def test_refined_word_cloud_1(): + project_path = f"{DATA_PATH}/word_cloud" + check_or_create_base_tag(project_path) + args = [ "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove.", "--inc", "--project-path", - "data/word_cloud", + project_path, ] result = runner.invoke(app, args) logger.info(result) @@ -123,16 +153,61 @@ def test_refined_word_cloud_1(): def test_refined_word_cloud_2(): + project_path = f"{DATA_PATH}/word_cloud" + check_or_create_base_tag(project_path) + args = [ "Add a feature to customize the resolution of the word cloud.The new version allows users to customize the size and resolution of the generated word cloud after uploading a text file, and then generate the word cloud.", "--inc", "--project-path", - "data/word_cloud", + project_path, ] result = runner.invoke(app, args) logger.info(result) logger.info(result.output) +def check_or_create_base_tag(project_path): + # Change the current working directory to the specified project path + os.chdir(project_path) + + # Initialize a Git repository + os.system("git init") + + # Check if the 'base' tag exists + check_base_tag_cmd = "git show-ref --verify --quiet refs/tags/base" + has_base_tag = os.system(check_base_tag_cmd) == 0 + + if has_base_tag: + logger.info("base tag exists") + # Switch to the 'base' branch if it exists + switch_to_base_branch_cmd = "git checkout base" + if os.system(switch_to_base_branch_cmd) == 0: + logger.info("switched to base branch") + else: + logger.debug("Failed to switch to base branch.") + else: + logger.info("Base tag doesn't exist.") + # Add and commit the current code if 'base' tag doesn't exist + add_cmd = "git add ." + commit_cmd = 'git commit -m "Initial commit"' + add_and_commit_success = os.system(add_cmd) == 0 & os.system(commit_cmd) == 0 + + if add_and_commit_success: + logger.info("Added and committed all files with the message 'Initial commit'.") + else: + logger.debug("Failed to add and commit all files.") + + # Add 'base' tag + add_base_tag_cmd = "git tag base" + + # Check if the 'git tag' command was successful + tag_cmd_success = os.system(add_base_tag_cmd) == 0 + if tag_cmd_success: + logger.info("Successfully added 'base' tag.") + else: + logger.debug("Failed to add 'base' tag.") + + if __name__ == "__main__": pytest.main([__file__, "-s"]) From f8dd30f1334dc89da5edabb2979b64ffd5c163fe Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Thu, 4 Jan 2024 20:16:29 +0800 Subject: [PATCH 265/637] fixbug: invalid default project name --- config/config.yaml | 1 - docs/.well-known/openapi.yaml | 2 +- metagpt/actions/prepare_documents.py | 1 - metagpt/actions/write_code.py | 2 +- metagpt/actions/write_prd.py | 5 ++- metagpt/tools/metagpt_oas3_api_svc.py | 8 ++++- metagpt/tools/openapi_v3_hello.py | 2 +- metagpt/utils/file_repository.py | 2 ++ setup.py | 1 + tests/conftest.py | 5 +-- ...test_hello.py => test_openapi_v3_hello.py} | 17 +++++----- tests/metagpt/utils/test_redis.py | 26 ++++++++++++---- tests/metagpt/utils/test_s3.py | 31 +++++++++++++++---- 13 files changed, 74 insertions(+), 29 deletions(-) rename tests/metagpt/tools/{test_hello.py => test_openapi_v3_hello.py} (65%) diff --git a/config/config.yaml b/config/config.yaml index 28a312a9e..6dff55b4e 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -14,7 +14,6 @@ OPENAI_BASE_URL: "https://api.openai.com/v1" OPENAI_API_MODEL: "gpt-4-1106-preview" MAX_TOKENS: 4096 RPM: 10 -LLM_TYPE: OpenAI # Except for these three major models – OpenAI, MetaGPT LLM, and Azure – other large models can be distinguished based on the validity of the key. TIMEOUT: 60 # Timeout for llm invocation #DEFAULT_PROVIDER: openai diff --git a/docs/.well-known/openapi.yaml b/docs/.well-known/openapi.yaml index bc291b7db..47ca04b23 100644 --- a/docs/.well-known/openapi.yaml +++ b/docs/.well-known/openapi.yaml @@ -11,7 +11,7 @@ paths: post: summary: Generate greeting description: Generates a greeting message. - operationId: hello.post_greeting + operationId: openapi_v3_hello.post_greeting responses: 200: description: greeting response diff --git a/metagpt/actions/prepare_documents.py b/metagpt/actions/prepare_documents.py index a936ea655..5c5798d95 100644 --- a/metagpt/actions/prepare_documents.py +++ b/metagpt/actions/prepare_documents.py @@ -35,7 +35,6 @@ class PrepareDocuments(Action): if path.exists() and not CONFIG.inc: shutil.rmtree(path) CONFIG.project_path = path - CONFIG.project_name = path.name CONFIG.git_repo = GitRepository(local_path=path, auto_init=True) async def run(self, with_messages, **kwargs): diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 25c4912c3..7377442b5 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -130,7 +130,7 @@ class WriteCode(Action): if not coding_context.code_doc: # avoid root_path pydantic ValidationError if use WriteCode alone root_path = CONFIG.src_workspace if CONFIG.src_workspace else "" - coding_context.code_doc = Document(filename=coding_context.filename, root_path=root_path) + coding_context.code_doc = Document(filename=coding_context.filename, root_path=str(root_path)) coding_context.code_doc.content = code return coding_context diff --git a/metagpt/actions/write_prd.py b/metagpt/actions/write_prd.py index d51c0a7be..073d8c076 100644 --- a/metagpt/actions/write_prd.py +++ b/metagpt/actions/write_prd.py @@ -14,6 +14,7 @@ from __future__ import annotations import json +import uuid from pathlib import Path from typing import Optional @@ -117,7 +118,7 @@ class WritePRD(Action): # if sas.result: # logger.info(sas.result) # logger.info(rsp) - project_name = CONFIG.project_name if CONFIG.project_name else "" + project_name = CONFIG.project_name or "" context = CONTEXT_TEMPLATE.format(requirements=requirements, project_name=project_name) exclude = [PROJECT_NAME.key] if project_name else [] node = await WRITE_PRD_NODE.fill(context=context, llm=self.llm, exclude=exclude) # schema=schema @@ -183,6 +184,8 @@ class WritePRD(Action): ws_name = CodeParser.parse_str(block="Project Name", text=prd) if ws_name: CONFIG.project_name = ws_name + if not CONFIG.project_name: # The LLM failed to provide a project name, and the user didn't provide one either. + CONFIG.project_name = "app" + uuid.uuid4().hex[:16] CONFIG.git_repo.rename_root(CONFIG.project_name) async def _is_bugfix(self, context) -> bool: diff --git a/metagpt/tools/metagpt_oas3_api_svc.py b/metagpt/tools/metagpt_oas3_api_svc.py index 319e7efb2..8e9f4a0da 100644 --- a/metagpt/tools/metagpt_oas3_api_svc.py +++ b/metagpt/tools/metagpt_oas3_api_svc.py @@ -5,6 +5,12 @@ @Author : mashenquan @File : metagpt_oas3_api_svc.py @Desc : MetaGPT OpenAPI Specification 3.0 REST API service + + curl -X 'POST' \ + 'http://localhost:8080/openapi/greeting/dave' \ + -H 'accept: text/plain' \ + -H 'Content-Type: application/json' \ + -d '{}' """ from pathlib import Path @@ -15,7 +21,7 @@ import connexion def oas_http_svc(): """Start the OAS 3.0 OpenAPI HTTP service""" print("http://localhost:8080/oas3/ui/") - specification_dir = Path(__file__).parent.parent.parent / ".well-known" + specification_dir = Path(__file__).parent.parent.parent / "docs/.well-known" app = connexion.AsyncApp(__name__, specification_dir=str(specification_dir)) app.add_api("metagpt_oas3_api.yaml") app.add_api("openapi.yaml") diff --git a/metagpt/tools/openapi_v3_hello.py b/metagpt/tools/openapi_v3_hello.py index c8f5de42d..d1c83eac2 100644 --- a/metagpt/tools/openapi_v3_hello.py +++ b/metagpt/tools/openapi_v3_hello.py @@ -23,7 +23,7 @@ async def post_greeting(name: str) -> str: if __name__ == "__main__": - specification_dir = Path(__file__).parent.parent.parent / ".well-known" + specification_dir = Path(__file__).parent.parent.parent / "docs/.well-known" app = connexion.AsyncApp(__name__, specification_dir=str(specification_dir)) app.add_api("openapi.yaml", arguments={"title": "Hello World Example"}) app.run(port=8082) diff --git a/metagpt/utils/file_repository.py b/metagpt/utils/file_repository.py index ff750fbbb..0ddca414d 100644 --- a/metagpt/utils/file_repository.py +++ b/metagpt/utils/file_repository.py @@ -138,6 +138,8 @@ class FileRepository: files = self._git_repo.changed_files relative_files = {} for p, ct in files.items(): + if ct.value == "D": # deleted + continue try: rf = Path(p).relative_to(self._relative_path) except ValueError: diff --git a/setup.py b/setup.py index 17fe8815e..94e9a35c2 100644 --- a/setup.py +++ b/setup.py @@ -46,6 +46,7 @@ extras_require["test"] = [ "chromadb==0.4.14", "gradio==3.0.0", "grpcio-status==1.48.2", + "mock==5.1.0", ] extras_require["pyppeteer"] = [ diff --git a/tests/conftest.py b/tests/conftest.py index 1f4a73030..419ac7d0c 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -11,6 +11,7 @@ import json import logging import os import re +import uuid from typing import Optional import pytest @@ -151,9 +152,9 @@ def loguru_caplog(caplog): # init & dispose git repo -@pytest.fixture(scope="session", autouse=True) +@pytest.fixture(scope="function", autouse=True) def setup_and_teardown_git_repo(request): - CONFIG.git_repo = GitRepository(local_path=DEFAULT_WORKSPACE_ROOT / "unittest") + CONFIG.git_repo = GitRepository(local_path=DEFAULT_WORKSPACE_ROOT / f"unittest/{uuid.uuid4().hex}") CONFIG.git_reinit = True # Destroy git repo at the end of the test session. diff --git a/tests/metagpt/tools/test_hello.py b/tests/metagpt/tools/test_openapi_v3_hello.py similarity index 65% rename from tests/metagpt/tools/test_hello.py rename to tests/metagpt/tools/test_openapi_v3_hello.py index 7e61532ab..5726cf8e0 100644 --- a/tests/metagpt/tools/test_hello.py +++ b/tests/metagpt/tools/test_openapi_v3_hello.py @@ -3,7 +3,7 @@ """ @Time : 2023/12/26 @Author : mashenquan -@File : test_hello.py +@File : test_openapi_v3_hello.py """ import asyncio import subprocess @@ -24,13 +24,14 @@ async def test_hello(): process = subprocess.Popen(["python", str(script_pathname)], cwd=workdir, env=env) await asyncio.sleep(5) - url = "http://localhost:8082/openapi/greeting/dave" - headers = {"accept": "text/plain", "Content-Type": "application/json"} - data = {} - response = requests.post(url, headers=headers, json=data) - assert response.text == "Hello dave\n" - - process.terminate() + try: + url = "http://localhost:8082/openapi/greeting/dave" + headers = {"accept": "text/plain", "Content-Type": "application/json"} + data = {} + response = requests.post(url, headers=headers, json=data) + assert response.text == "Hello dave\n" + finally: + process.terminate() if __name__ == "__main__": diff --git a/tests/metagpt/utils/test_redis.py b/tests/metagpt/utils/test_redis.py index b93ff0cdb..d499418ac 100644 --- a/tests/metagpt/utils/test_redis.py +++ b/tests/metagpt/utils/test_redis.py @@ -6,20 +6,34 @@ @File : test_redis.py """ +import mock import pytest from metagpt.config import CONFIG from metagpt.utils.redis import Redis +async def async_mock_from_url(*args, **kwargs): + mock_client = mock.AsyncMock() + mock_client.set.return_value = None + mock_client.get.side_effect = [b"test", b""] + return mock_client + + @pytest.mark.asyncio -async def test_redis(): +@mock.patch("aioredis.from_url", return_value=async_mock_from_url()) +async def test_redis(mock_from_url): + # Mock + # mock_client = mock.AsyncMock() + # mock_client.set.return_value=None + # mock_client.get.side_effect = [b'test', b''] + # mock_from_url.return_value = mock_client + # Prerequisites - assert CONFIG.REDIS_HOST and CONFIG.REDIS_HOST != "YOUR_REDIS_HOST" - assert CONFIG.REDIS_PORT and CONFIG.REDIS_PORT != "YOUR_REDIS_PORT" - # assert CONFIG.REDIS_USER - assert CONFIG.REDIS_PASSWORD is not None and CONFIG.REDIS_PASSWORD != "YOUR_REDIS_PASSWORD" - assert CONFIG.REDIS_DB is not None and CONFIG.REDIS_DB != "YOUR_REDIS_DB_INDEX, str, 0-based" + CONFIG.REDIS_HOST = "MOCK_REDIS_HOST" + CONFIG.REDIS_PORT = "MOCK_REDIS_PORT" + CONFIG.REDIS_PASSWORD = "MOCK_REDIS_PASSWORD" + CONFIG.REDIS_DB = 0 conn = Redis() assert not conn.is_valid diff --git a/tests/metagpt/utils/test_s3.py b/tests/metagpt/utils/test_s3.py index f74e7b52a..132aa0635 100644 --- a/tests/metagpt/utils/test_s3.py +++ b/tests/metagpt/utils/test_s3.py @@ -9,20 +9,36 @@ import uuid from pathlib import Path import aiofiles +import mock import pytest from metagpt.config import CONFIG +from metagpt.utils.common import aread from metagpt.utils.s3 import S3 @pytest.mark.asyncio -async def test_s3(): +@mock.patch("aioboto3.Session") +async def test_s3(mock_session_class): + # Set up the mock response + data = await aread(__file__, "utf-8") + mock_session_object = mock.Mock() + reader_mock = mock.AsyncMock() + reader_mock.read.side_effect = [data.encode("utf-8"), b"", data.encode("utf-8")] + type(reader_mock).url = mock.PropertyMock(return_value="https://mock") + mock_client = mock.AsyncMock() + mock_client.put_object.return_value = None + mock_client.get_object.return_value = {"Body": reader_mock} + mock_client.__aenter__.return_value = mock_client + mock_client.__aexit__.return_value = None + mock_session_object.client.return_value = mock_client + mock_session_class.return_value = mock_session_object + # Prerequisites - assert CONFIG.S3_ACCESS_KEY and CONFIG.S3_ACCESS_KEY != "YOUR_S3_ACCESS_KEY" - assert CONFIG.S3_SECRET_KEY and CONFIG.S3_SECRET_KEY != "YOUR_S3_SECRET_KEY" - assert CONFIG.S3_ENDPOINT_URL and CONFIG.S3_ENDPOINT_URL != "YOUR_S3_ENDPOINT_URL" - # assert CONFIG.S3_SECURE: true # true/false - assert CONFIG.S3_BUCKET and CONFIG.S3_BUCKET != "YOUR_S3_BUCKET" + # assert CONFIG.S3_ACCESS_KEY and CONFIG.S3_ACCESS_KEY != "YOUR_S3_ACCESS_KEY" + # assert CONFIG.S3_SECRET_KEY and CONFIG.S3_SECRET_KEY != "YOUR_S3_SECRET_KEY" + # assert CONFIG.S3_ENDPOINT_URL and CONFIG.S3_ENDPOINT_URL != "YOUR_S3_ENDPOINT_URL" + # assert CONFIG.S3_BUCKET and CONFIG.S3_BUCKET != "YOUR_S3_BUCKET" conn = S3() assert conn.is_valid @@ -42,6 +58,7 @@ async def test_s3(): assert "http" in res # Mock session env + type(reader_mock).url = mock.PropertyMock(return_value="") old_options = CONFIG.options.copy() new_options = old_options.copy() new_options["S3_ACCESS_KEY"] = "YOUR_S3_ACCESS_KEY" @@ -54,6 +71,8 @@ async def test_s3(): finally: CONFIG.set_context(old_options) + await reader.close() + if __name__ == "__main__": pytest.main([__file__, "-s"]) From 174da4f0e3cfa908a21f28bc756d3d0a64e229dd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Thu, 4 Jan 2024 20:25:14 +0800 Subject: [PATCH 266/637] feat: ver +1 --- setup.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/setup.py b/setup.py index 94e9a35c2..10938c769 100644 --- a/setup.py +++ b/setup.py @@ -57,7 +57,7 @@ extras_require["dev"] = (["pylint~=3.0.3", "black~=23.3.0", "isort~=5.12.0", "pr setup( name="metagpt", - version="0.6.1", + version="0.6.2", description="The Multi-Agent Framework", long_description=long_description, long_description_content_type="text/markdown", From 86b9d11f243a4f0d1581fec4afcdc0f6434cfc0b Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 5 Jan 2024 17:03:24 +0800 Subject: [PATCH 267/637] 1. Update test_incremental_dev.py and data files example --- data/dice_simulator_new.zip | Bin 27923 -> 56384 bytes data/number_guessing_game.zip | Bin 65020 -> 53755 bytes data/pygame_2048.zip | Bin 19338 -> 60632 bytes data/simple_add_calculator.zip | Bin 53760 -> 8105 bytes data/word_cloud.zip | Bin 54076 -> 12762 bytes tests/metagpt/test_incremental_dev.py | 44 ++++++++++++++++++-------- 6 files changed, 31 insertions(+), 13 deletions(-) diff --git a/data/dice_simulator_new.zip b/data/dice_simulator_new.zip index 307e41541899d742a97d1209813169f672fe13cd..4b8d3f038addf84e74e0d1a15a5209d3a263e921 100644 GIT binary patch delta 25311 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 CD&JE8 delta 1647 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 O)^{N+tc`TY!2bbVvtEk; delta 38747 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 Z_u9A{+&;>N84D8&BL@u4GP)tk{{wf5;NAcL diff --git a/data/pygame_2048.zip b/data/pygame_2048.zip index 476d0b438b207f594418d2e8fb78a29371590f66..2f0457be626f4f60d27563c5bdc7c86940260c35 100644 GIT binary patch delta 44150 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 AJOBUy delta 5409 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 Huo3?O!dcjh literal 53760 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 z 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

Ei(XwNOEtpM1u_&thFjw8IQWXi0_ zyU5F-H#`R1z%^?@ng57az?o@m&yMBP7Fasw0kZfWty31roVK;08@{gDl+{b`*8sjDL)qCGk zROfJUYSD8!0d2@%`Yi%Gs|Q3LRKTN4Tsr*U2#etMvz20TGhCB!HH~BS>9+Yc zw21W55#6gwD=sn|X&X?0(DpM@fKB*<);pZ8j?ai|(W#=;I5t`)T&QEko~aN^6wW); z6U&1k$q(h$EwGCzs2%q=h2A9Q)zc|-Mv*~bP^egrLV;QbXvjbj@1y<4P_;)uvsjOu zHY%NHV}&?>mDkX<7r7uP=v_IXWgs&*ZyiUp@=$u$YuPovXoNx+Y%@&LlF-WoB?^hO+ zjZd-$rUbVM74{gGXj}IQlqt%gRr1^eTW(g82xm`x9~6eR#&60N@bbyT(l3B*eKce~ zLd&;C1Kqu)a$`PoMp~xKA;?Bl1$G339)C(Gm2=VbTLD2Wjb?kJX#vcDmrO}#F%0X3 zVB83WmVh$^Q%OO}pNgdeI2;&)RN^F7@JJj=0J?^5AsD(;VnRR37ZEsX{l<<+ie=uX z70GlKKf79^Z42P_e>(dPc&h&Ye`TJE#5lL27ivPK8{Y2$+zW;ZRb1oj;=j-)+z24`Xd(U~jUr+DkNmAjrvtu}O z(k3_s6c=^dQv~b8kNS7asa!W&^ksif|6J8JP(+QHnOKjuizt>az}iZOu;q=9CB@BK z!4lUo+i0Ssm}z*f(A5RIS zJtQE2kQyS7iA~OpfmwEm0$09aiwyg=4PK8QGE;zFmC?S+-8yFIM6j#wyP>u|*E{R2 zHCf3_#K>0fiywVd>6|AcEs%nA@aKlD5T$iJreTkB&Vt227Y=;`QY@Tq2zQa|9e{g>y`2487Ftji-+b&mzmY`jtW%GH<^!ti? z?dB%1C64*7B#!ND3HWaG$@%WAk9n_~j%}|M)o#8w-1*utw%z2uvxL~$O7&Z z`)*9heP1l{Td6Yj+mfsOzUsI0y|cbQ&{_|gW~?lcm6B~@8yw~7f~e3fs~eWH_Q;_q zs=Hs$NW!k!IU#isZ{>;?LyBicN(Hkj1Kj@dN__?i5rtd;Jm_i0a0|g_4@wp70@7Mi zLSHJpnhUGjS8YZ)Z;3VIEw76cG>hL=NTr_LWbv8}J@<@&!0C4Rsdp#T+uj;3y&O;- zaxvB0NU-Y3vlOcjE`9LQtV~wsq9;UJ_TBIp9VzMh8fj?D3c1fTF-FU)4={?F*NQC{ zC1Ok_dI&#f&6!r)_ea@3BG?|K#<}EjQ$;i8J>GIliYG85CYQa=LX_T%N8){b=%asq`c?YX z{-a8!EOhz&+O;Ov#~xNv4;e#^=NN`fOlAC@!wc6pgQa|G-kQoit+G32Z}@>?SZ(`S znIZ$FY=dgU?N(}J(BvD9Q%hCjn@OV37KXy{?hFKFlnRd;=N6TIk?KZ8X2I*onE|Qq z>KwDKuKTFBBqaS}N7gIl@>psC z`o0LG2T1)_RaY?c4C(@5nRi$DspA??PIh*)O=g8Q&yMLeWksrn>vhK7m}{!)UQYK` zy5!9>l6LCVz4KFyRG)KI(gW05`J@9*4W3*tvGcqZk>hPwm5^0r)b9Swj?R$Ea;e$< z4olV9(H8@&iUV`h*@al=4DhN#Y^&;SQ#HfQ2YDc{_!8r!>p0m3NVlUs@BsC`Zut8_ zUPo9{xlr6}@wF(2)32WmPiNn~_nLK3oM@``Y?1TtZ=TaB7Z zux1UVCq&w9*V+Zi!y%Iykwy(Fx8Y+1G_V&5FO(XQq|!>QFtVfJWW(5ax3t~s98T&9Z1%@{dXMkvLt$99#Q6?afX zrA(fBhfOcH<5=2}0J38&XSaTI1+>Hnx^M*`h^cOW7(GgCRdUVw&8u{O@|~)-ZRIbf zt>0T8jNL0R&19L1G<9fO7acP749b(u9Pz-$cY#X|e^f}QX^Zn?(zAfqpNZ_WQ9JJy zlbJ6j;X9~ z0n$}ndfxj9_4BMbkK^ZL1*SD4&Qw6DO1^p2BCb|jY$u7n&XJj3QrGj+DN}jiwng0Z zXjM$Uz>^HuOy1)osq|dbRQk?yRh!B2x;4+l%dULymLC3k+Ug5MPO88npS>mdDd7Q~RP;jbA3+>sh?}Hs;M-cXQb#!+SRi zWN9aaal9+FMcY$?5%UmXXy<^SHDojEbbB&iTM$Ma*D;E2T2iFmf_M-GF=s0=A!YTQ z4z5>YYuO*z=BNeaAGWtwoXDh^$Yom0kG#MVDH3_ ze?)JFG5I9Oc786OO_?1sP4qQJXb60dEqniCrixNO^8u`lr^R{Qf{^nvq@Bq86lJEf6`gC)#Kb~YO|+sR{|{J>BBeWi|-PKZ(b?Z^T~94 zY*Z;}aVnbjD%*5-SelTsOcR&Km>?2e20ook5Amxc** znk-WTt)m9M3J8wcD4JWvvhs+{nkK^=&^zlqzG@OO^fR#`q$%r+R5H2KOIsf9?!i&17HI^-9F~3fR*#s<6 zLWXarM-497KX()7O=i{L3QKr|38#OBeWqBwESG}Vz2Jy9!$@+|%x#V$i}$UkyYntf zJqq!Bj(j%3T!bwhytK-_kk3IvPyLvfRO7mUrS_Oee(Hk>QYMR>z>^JBSq$~4VMdBR;G4Jlb(m)AV;^ijy!h5CM;7;9|(Y!kLFghG7hkMZWo4xBN? zbG26f}US_UfUn6E%<1j;}T-U$J{!FssY1B6>?6=Tw1kt{Hla1J$Uh+cI zYHiaw%<5;TjeJrno_U_s)=GFa!O-r);GU6!hAW8twf^z6O#9q15Sg&U7y!_ zC!=2C*yV|&Qu3)7VVDR+7JGk#eq)Q#T6EA?dN4k+=si0zO2A@s{StFy_wKKVOS<337Foyb+);51MY3`mzQ(a*iws@QkLjqv4h zjpKE|=`)1_qbMQirigkM<{*HAMP!2VcI z`=sn%Iu?;GJF||e z99tE}$VFjdHs97J%sC;?^H~>eY0jzAeuP)M>1o!~h^G#9l0@Gp(Fx7eR!}G)nv_P! zjYvz--GBVLfw9}wrlQIZ{~H1>vo(8`inE%|;3kWp3ia`~YfCGP69aO?a>_#)E<-^s zPL3|7{MtzFrzTId+sjGGGQm?%`H9=PEnc+Agrxwd#tjGH2q|~2Z@jE&e78pO>b}5F}`iPc(-1=>}XVV$)}sSTlCxKiyCLDkA#$- z;(`xbl_Z(x8dTOy(EG)tL%O((w(2V)9@f&Bbw*#xG4Jef7ri@fDtIo6MjBRrmuT?4 zrs#`s%#H;T7K@1QaDv>_O`S#A^=uo$#Prm*e(lQ;YaRXJbobm-pTy6m%E|GMoM!NS zj>*L-^kL9FQ1D5JcGg!-p>(}(cPI6U9@3TEOYjUa*OAH)AB}U)SQ69iP_up?-ujh; z>B6Uw;~vUXt7$gM&@*^T>HJ?_$>$K9I6vi_KWSJI7P9KhGL1}^NhiO6w>}eZp(Wj% zU{~+nkuT@pd0dmc8Lw$rBdC;fKH=k}tzO$gu!+~swR+h>C=D%N-;lI4 zcE2HsPVT4>);;T$94exW#YnZd6>AdfBzM0n)5M5Z7v_}K{oPEVYHS}xufaB7yN5Vx z))Sj93cNqdo~XuIgYk~krv?4xE?#A26}c}Q_evy5D%*C+v_p(x4b z`>9)Mt-L;63{>04>yrQ4r*={i8bLocB z@5;~IAsz)3sxl9SoyL_qxsQ(yP0?jMW&V7zwSPO(w#WM6>we~CO~du)GLo-#=1(eI z5bKHB>b|8(H!NApnmdzFf;_7HbV($Y2zSJL!};!l7T(roJ+eDRd%dcvr|W^#s9oZ1 z%qIrOxSoOL#z%!&BdozjQ?3(!3>SyrSfo-Wk`e)>ZOTJc3SqEjFFsPHwZ|m(?t?T?}x4YSo=S`pN3zFy{ucK<3m#>emDmcINVq zc1~GX)znk3h>YnqVm~|g>**UOh8NGrSX0G|ud9|SYP?Lt3XN9$Af?|iIaZ%9O$h%s zTkP2LREW%7K)JFaPfYD)^&ng1FfSGJlw0tmvjVY#A8lZ?rQf=qDwc^^Q#mxo;z6L9 zDmFL0ye<}{14qRBAc8=0Ts3ZdUil+F92miRqXQUBS6DR0Zw3;Ry?OA%h!WB`{=&eF z1(yn|@EF9LO)X3z_!2E3-4qwD0h7M6m-PL}7vc zgvXe1F!C#Qv^r`T4T)`&x3MLD#7=}$nlrm6pOq{YEN>6LGk%$%oMMtv$TDmpm(@uR z`a$pX1p9fI;zVYDgnF_*k^D$hqq>7-O-C~${Ss1NQIA1Us|lgONEM&2rJ=g1Eyyd& zldjhn&6`gCmUe(SOHnJ24wr;CFZgv)Vz>TTZJfyJ+{H=Sr#<58bb?%=h(f1{D1QS< z&!EctinCVu=iD(ZS6tJS$QN_JIVY#oh)wr+E<$H8*wX3e3TLSi&L43*oXwG0|?NzHe8l$j(eH*e(;p9ZL~x zxMTdvJecMlM$I*qa&H<4boq?aR=DDcTcqBsztmhdR-WT$WdsT$(y_73R)mMvvUO4_ zqWlYj%K~uD6sAxM^-+rw^*KBCnKN86Hu2MePOmhdV!xa=G;w}u3tGUbF&lh7j8q`2 z4fCzTy!A){ufE9k-6`=z&KSlUsilQ3Sjb+UdrcQ}8J4@QwqkrKRcshAkQfqnrBq2X zoGQfR{*btWa8|wmamTwbT>OzOS@uW3s*Ay`D#+41pWM>bT0372Ve`Tl*c5O~Uo&b! zmqBxS{~1j~V~Bt74@BBJr+tWbbSC3i_Ke#jp(^vSCNaK{ELyhsTi?G&IH%tD=oXc$ z)t-A=s~O%=CKH#7uO(=cq!Od55CtVw@R0U(xzcCoAv!#Z%briiQQ3RJZ=Q-eDB-gl;}4GixG((lE8ti)=OV}GF_8Vcd5pkg+q zKj*G{nqZr=R)OYzrQr;=^%GNpufh#q3_E@C)Q;Y7#99{*uB?4vvZ3RW$+mdLG}db> zbYP|wsqY4%$GzbZ96?#u-gs)ZUrhLCVzA;F*e_HjU-T_ z@5c}4H|?`3^ZF0lB>H$ZKCo9mI-gwE>0Vem;srH$p?bekNVbGZ(ob(dUQfVh=lbnc z$_|(}G4JT7!n`^hc+aO9{)gu+BwjV~6|mZvZ1h<<*sY%98dZz=;AKeM{$h}Ew6K7; zFrs)+weRhV(Qt3}qqinLMJci5Lv@20WBI>^+BD@mN?HfPMz?*sbrm9xugiZ5>Jh#i zsVLJcnC*tMB0HE6-qcFsaa0-OmC+ZsawMWUTKXu*uxrQ9{=NL;uQe3^SBZj;A!WyA zH0h{wi5OUmbq6RkllW(;$KR1G&Fw8+?ah%0XJDyBl%AGmBji{$&9@Uw^Dr7{febaJ zm5kT11rE`0;}+3ab;_OU*V(qEH;#7j2;W)1b*C44?U7eerEr4J_HE@0U++?w*WMl~ zsiRapi5!8DJem__FCXlnVNvWTU@u~1dTc56pt)3EnzE*1jl9jMB1q{op<;OaQ76{t zO*1hI-J?&>4mu_-TX#II@QTbiQ<+g);{effGVb1O)e5azZu%-XX!V5)jYpX?7+VZ4{qs9%7BHw#6 z5O~Cy*;yXf8ckiu>W>|&_ypOc5pJ^!8Vh~v-S|E$D5I`)J78xs#aHLTcdWf~nuGO1 zZ2ycg66S(5MYtjzU3V3QF|jByD1rZsj}ZpcY5BDl^rZs7kpTt1sC7%Ifw{Y;1@+V4 z({suS<_LpZ!Qhs_LXzFmnX{U|Q2$&F#ef3Z9jMhZQ^58QRMZNqgFq955+ufC>aQQ~@Pwpka}R!&)NX z*0weW;lys>oj{c{sRJ9T*bOz%aHulNL%am)tu38x9g(&O_#P%YguJ#cIWzzcYCC}f z1O^&2{%}l`X81uc^xM(CL0k~{)-DbRxGivo;&(?8aWIGfgU4N>^|}ZE zyBgR~Wr(POwxdM?!j2=t1)#b`R zVio+^NOuC5xU3WhbAfprB)T&Uwf6BJ__2)}3F<%|m;na+C#*0~lL{&q&M-@ut=qp0 z^x)zzP6NmA7~#QWzvcjA00<2~62T9wrnIyJE^GgwF|ha@HvzDs2pnaovQ^YTPbmUk z5LOO7q9;lOZ2oV260{(Lu z^8RzPyUc5rLg2lNz{Z7!`pLTi1RBi4(%#kTx0CUIxmXS?AUMQDd&KTq-Tf{uNaGVufH$4-h=-$Rcyc9-o4Z>9uEI&artRLDvcVkIanYHcklVhI%&x3Go^ zL&d?fVFWHbheZGX*hu`rrjc7l`R{DZL1T(SAtFK+P#8?mTnuU{BrFVQ#EMD)>Rh6L zlpsU`A_2&7N{E4HLjZi|0>MVm`VTf?%?cu@YcOgX4CoMIE@~wTwFKl*#YL>etwmuFQ7ds#5ivm3S4c$E3JQUMXM+n&77&(qW$u5S zInf9AL81G-+-QKt1SfkBo=xa!P;7QJqxZ8ps2aV;<`rm6a2nF!*&viavDsCI-p}Ts zHuN5wQqY*-l%fx^IjHr!XZf-!ST^94e-BzlTSD*mUmIX*pe*ld?e34`LABjIHhG{i z!D;V;XJZX~vj=g+?h5DbXLC?GcaKdzXiRWIy5QO9X@FycQkvY)=Abz49-9QvnBb&x z4_ZcBLhtv_G#O1$Y>uf4j#f05m2zHC*s)@U_6PK?%U_XLC>jc8|>&(3s!^ zVGmmVcW(N?S>9EE-OuKr2J9XiI&Dx90<8!Oo(;ARD9gJFt^3&=)M(vfqY4@moMJ0@ zHhFHK*z9()+t21;FS|W9^Pn-ob+!Y~hR_2X8&uo4{cH|4j@x6S4H^?%8@Yp)(U#Er z{pYkcPf%=jo2cz)bFhut9vhNtU@^fpRXdE$uO|uU?}?${EFZX?@#ojs{iv_Oq5kZ$ zwm(Y0N1plrenH(Ypzu00O8AyQhEavbGn4szYJXJ=cCmJNU3{@0k=T z=SP7~KY#ypx7!=%FrbqtpLPcdNa=8(KRU4OMe(46%xF+xkHU>S9QS8OwqIy}@5zP& zL{FeL5NHRM+Fy0~{Q(s|#@|{b?HNc8JT#uaHGM$$eE8+~pUq8vO(t|0%!3Za-QO%= zFKhU_-5U;eG1_xC_kV#sG%$Y>K;1S$4^$o{Lk}hJtLenAc%U6W*sTQxhfbs?7`Xp% z_;0;B&<-;LFC*dqa`?9-{_NcG%j-XbfI>nKR1jDq{*AfU$>ZN>zn)2;=Y%N;w7+)t z*fa3A-X3UonSqzb|9=J!gdNrCW6$Ejqy5)Sa5N?I8~+_3bmxyf#Dg4Ay+6={8FS(D+FZ>eNdf^{8#k{NZm^ zG-xWQ$$uMoH1k&+fxrKCZ`S<1s%Fpc(f<*7UyaS4nLm$SHnjJ^9z{U0=wAd5l$qe* UqTY8I_-Dh4fe{3}4Flu<0l%(&;s5{u diff --git a/data/word_cloud.zip b/data/word_cloud.zip index fd395c5bd2c2770fb8a1f2482d16a2a3cd9f0686..d15c1cf74cea5a7ac15ed29f75574351949df1cc 100644 GIT binary patch delta 732 zcmZWmUuaTM9QOY1Ty%A6xpH|mUD7sn10^tulv+;_Q3OTwvLGsAP~;v8Dvj_#tC8#r z%rK(C8Z;kbMi2}sdTIWf+Cwxcu8$4E=P0Oq&do)19`5h^e&;*i?>px%Jrh^Yrkj{u z?nf?UwqCq&>2mbkvBP3dzmBcNq6c#^zmUb#ga;=|o20)xxDs<>A}$p|@p-+dDho zlD+s<(x&d2mnkn{urf{xD+9P$3Dd9vO45Z^MH8`kXAB75vB6O_@(BK_Owd{r)i8bP zLMxV1R;TYb_~Jp7-JZ=JB)1dOWPD1tQ~U1KnUTwQa1UXw{O@nK(Td+A6QS}=BR zGgJNNyzL+7xS5|lI^B!h$_TyE=;xI|o}0aleG6Bt%^0x1K0+roIs|z=!-A^1y_Ks1 z4HLfAee|wN{rn$JHcTYgalJ9bZP_#Kw&QJMoF-WrfU~)ePCGfAY6hw3;_&|CK9rk2 YI=+)r^9Ps7rE}?88+E#`!|WQr0nI-P3;+NC literal 54076 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 An*aa+ diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index cdee82d2a..c69515b86 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -28,6 +28,7 @@ def test_refined_simple_calculator(): project_path, ] result = runner.invoke(app, args) + os.system("git tag refine") logger.info(result) logger.info(result.output) @@ -43,6 +44,7 @@ def test_refined_number_guessing_game(): project_path, ] result = runner.invoke(app, args) + os.system("git tag refine") logger.info(result) logger.info(result.output) @@ -56,8 +58,10 @@ def test_refined_dice_simulator_1(): "--inc", "--project-path", project_path, + "--no-code-review", ] result = runner.invoke(app, args) + os.system("git tag refine_1") logger.info(result) logger.info(result.output) @@ -71,8 +75,10 @@ def test_refined_dice_simulator_2(): "--inc", "--project-path", project_path, + "--no-code-review", ] result = runner.invoke(app, args) + os.system("git tag refine_2") logger.info(result) logger.info(result.output) @@ -86,8 +92,10 @@ def test_refined_dice_simulator_3(): "--inc", "--project-path", project_path, + "--no-code-review", ] result = runner.invoke(app, args) + os.system("git tag refine_3") logger.info(result) logger.info(result.output) @@ -103,6 +111,7 @@ def test_refined_pygame_2048_1(): project_path, ] result = runner.invoke(app, args) + os.system("git tag refine_1") logger.info(result) logger.info(result.output) @@ -118,6 +127,7 @@ def test_refined_pygame_2048_2(): project_path, ] result = runner.invoke(app, args) + os.system("git tag refine_2") logger.info(result) logger.info(result.output) @@ -133,6 +143,7 @@ def test_refined_pygame_2048_3(): project_path, ] result = runner.invoke(app, args) + os.system("git tag refine_3") logger.info(result) logger.info(result.output) @@ -148,6 +159,7 @@ def test_refined_word_cloud_1(): project_path, ] result = runner.invoke(app, args) + os.system("git tag refine_1") logger.info(result) logger.info(result.output) @@ -163,6 +175,7 @@ def test_refined_word_cloud_2(): project_path, ] result = runner.invoke(app, args) + os.system("git tag refine_2") logger.info(result) logger.info(result.output) @@ -182,31 +195,36 @@ def check_or_create_base_tag(project_path): logger.info("Base tag exists") # Switch to the 'base' branch if it exists switch_to_base_branch_cmd = "git checkout base" - if os.system(switch_to_base_branch_cmd) == 0: + try: + os.system(switch_to_base_branch_cmd) logger.info("Switched to base branch") - else: - logger.debug("Failed to switch to base branch.") + except Exception as e: + logger.info("Failed to switch to base branch") + raise e + else: logger.info("Base tag doesn't exist.") # Add and commit the current code if 'base' tag doesn't exist add_cmd = "git add ." commit_cmd = 'git commit -m "Initial commit"' - add_and_commit_success = os.system(add_cmd) == 0 & os.system(commit_cmd) == 0 - - if add_and_commit_success: + try: + os.system(add_cmd) + os.system(commit_cmd) logger.info("Added and committed all files with the message 'Initial commit'.") - else: - logger.debug("Failed to add and commit all files.") + except Exception as e: + logger.info("Failed to add and commit all files.") + raise e # Add 'base' tag add_base_tag_cmd = "git tag base" # Check if the 'git tag' command was successful - tag_cmd_success = os.system(add_base_tag_cmd) == 0 - if tag_cmd_success: - logger.info("Successfully added 'base' tag.") - else: - logger.debug("Failed to add 'base' tag.") + try: + os.system(add_base_tag_cmd) + logger.info("Added 'base' tag.") + except Exception as e: + logger.info("Failed to add 'base' tag.") + raise e if __name__ == "__main__": From 8851147df74f2111b392fb88802d19b46145b4bd Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 5 Jan 2024 17:20:15 +0800 Subject: [PATCH 268/637] Update test_incremental_dev.py --- tests/metagpt/test_incremental_dev.py | 65 +++++++++++++++++++++------ 1 file changed, 52 insertions(+), 13 deletions(-) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index c69515b86..216b54952 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -28,9 +28,13 @@ def test_refined_simple_calculator(): project_path, ] result = runner.invoke(app, args) - os.system("git tag refine") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine") + assert True def test_refined_number_guessing_game(): @@ -44,9 +48,13 @@ def test_refined_number_guessing_game(): project_path, ] result = runner.invoke(app, args) - os.system("git tag refine") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine") + assert True def test_refined_dice_simulator_1(): @@ -58,12 +66,15 @@ def test_refined_dice_simulator_1(): "--inc", "--project-path", project_path, - "--no-code-review", ] result = runner.invoke(app, args) - os.system("git tag refine_1") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine_1") + assert True def test_refined_dice_simulator_2(): @@ -75,12 +86,15 @@ def test_refined_dice_simulator_2(): "--inc", "--project-path", project_path, - "--no-code-review", ] result = runner.invoke(app, args) - os.system("git tag refine_2") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine_2") + assert True def test_refined_dice_simulator_3(): @@ -92,12 +106,15 @@ def test_refined_dice_simulator_3(): "--inc", "--project-path", project_path, - "--no-code-review", ] result = runner.invoke(app, args) - os.system("git tag refine_3") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine_3") + assert True def test_refined_pygame_2048_1(): @@ -111,9 +128,13 @@ def test_refined_pygame_2048_1(): project_path, ] result = runner.invoke(app, args) - os.system("git tag refine_1") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine_1") + assert True def test_refined_pygame_2048_2(): @@ -127,9 +148,13 @@ def test_refined_pygame_2048_2(): project_path, ] result = runner.invoke(app, args) - os.system("git tag refine_2") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine_2") + assert True def test_refined_pygame_2048_3(): @@ -143,9 +168,13 @@ def test_refined_pygame_2048_3(): project_path, ] result = runner.invoke(app, args) - os.system("git tag refine_3") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine_3") + assert True def test_refined_word_cloud_1(): @@ -159,9 +188,13 @@ def test_refined_word_cloud_1(): project_path, ] result = runner.invoke(app, args) - os.system("git tag refine_1") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine_1") + assert True def test_refined_word_cloud_2(): @@ -175,9 +208,13 @@ def test_refined_word_cloud_2(): project_path, ] result = runner.invoke(app, args) - os.system("git tag refine_2") logger.info(result) logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + os.system("git tag refine_2") + assert True def check_or_create_base_tag(project_path): @@ -194,8 +231,10 @@ def check_or_create_base_tag(project_path): if has_base_tag: logger.info("Base tag exists") # Switch to the 'base' branch if it exists + stash_cmd = "git stash" switch_to_base_branch_cmd = "git checkout base" try: + os.system(stash_cmd) os.system(switch_to_base_branch_cmd) logger.info("Switched to base branch") except Exception as e: From e975a43dad2b216ea2d3160d90320f5ec5e59321 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Sat, 6 Jan 2024 23:35:33 +0800 Subject: [PATCH 269/637] fixbug: an unexpected UserRequirement type message is thrown when there is nothing to do. --- metagpt/roles/role.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 356b9e33f..3bcd600fc 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -418,7 +418,7 @@ class Role(SerializationMixin, is_polymorphic_base=True): Use llm to select actions in _think dynamically """ actions_taken = 0 - rsp = Message(content="No actions taken yet") # will be overwritten after Role _act + rsp = Message(content="No actions taken yet", cause_by=Action) # will be overwritten after Role _act while actions_taken < self.rc.max_react_loop: # think await self._think() From 7f04eaafae184443b44a01afd1a49248a6cd0af1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Mon, 8 Jan 2024 14:45:23 +0800 Subject: [PATCH 270/637] fixbug: unit test --- .gitignore | 1 + tests/metagpt/actions/test_skill_action.py | 10 ++++++++-- tests/metagpt/learn/test_text_to_image.py | 10 +++++++++- 3 files changed, 18 insertions(+), 3 deletions(-) diff --git a/.gitignore b/.gitignore index 6dd3608f1..a6f45d894 100644 --- a/.gitignore +++ b/.gitignore @@ -175,3 +175,4 @@ htmlcov.* *.pkl *-structure.csv *-structure.json +*.dot \ No newline at end of file diff --git a/tests/metagpt/actions/test_skill_action.py b/tests/metagpt/actions/test_skill_action.py index 0e0d5d5aa..529ed632a 100644 --- a/tests/metagpt/actions/test_skill_action.py +++ b/tests/metagpt/actions/test_skill_action.py @@ -6,6 +6,7 @@ @File : test_skill_action.py @Desc : Unit tests. """ + import pytest from metagpt.actions.skill_action import ArgumentsParingAction, SkillAction @@ -47,7 +48,11 @@ class TestSkillAction: assert args.get("size_type") == "512x512" @pytest.mark.asyncio - async def test_parser_action(self): + async def test_parser_action(self, mocker): + # mock + mock_text_2_image = mocker.patch("metagpt.learn.text_to_image") + mock_text_2_image.return_value = "https://mock.com/xxx" + parser_action = ArgumentsParingAction(skill=self.skill, ask="Draw an apple") rsp = await parser_action.run() assert rsp @@ -80,7 +85,8 @@ class TestSkillAction: @pytest.mark.asyncio async def test_skill_action_error(self): action = SkillAction(skill=self.skill, args={}) - await action.run() + rsp = await action.run() + assert "Error" in rsp.content if __name__ == "__main__": diff --git a/tests/metagpt/learn/test_text_to_image.py b/tests/metagpt/learn/test_text_to_image.py index 760b9d09c..1485df5c6 100644 --- a/tests/metagpt/learn/test_text_to_image.py +++ b/tests/metagpt/learn/test_text_to_image.py @@ -12,10 +12,18 @@ import pytest from metagpt.config import CONFIG from metagpt.learn.text_to_image import text_to_image +from metagpt.tools.metagpt_text_to_image import MetaGPTText2Image +from metagpt.tools.openai_text_to_image import OpenAIText2Image +from metagpt.utils.s3 import S3 @pytest.mark.asyncio -async def test_metagpt_llm(): +async def test_text_to_image(mocker): + # mock + mocker.patch.object(MetaGPTText2Image, "text_2_image", return_value=b"mock MetaGPTText2Image") + mocker.patch.object(OpenAIText2Image, "text_2_image", return_value=b"mock OpenAIText2Image") + mocker.patch.object(S3, "cache", return_value="http://mock/s3") + # Prerequisites assert CONFIG.METAGPT_TEXT_TO_IMAGE_MODEL_URL assert CONFIG.OPENAI_API_KEY From 2ec2e71c4d3013635574785072aa69ba7ac7cd5c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Mon, 8 Jan 2024 17:38:53 +0800 Subject: [PATCH 271/637] fixbug: rename folder does not work in windows os --- metagpt/roles/engineer.py | 3 ++- metagpt/utils/git_repository.py | 7 +++++++ 2 files changed, 9 insertions(+), 1 deletion(-) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index e05e69cbb..b2a909400 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -204,7 +204,8 @@ class Engineer(Role): async def _think(self) -> Action | None: if not CONFIG.src_workspace: - CONFIG.src_workspace = CONFIG.git_repo.workdir / CONFIG.git_repo.workdir.name + project_name = CONFIG.project_name or CONFIG.git_repo.workdir.name + CONFIG.src_workspace = CONFIG.git_repo.workdir / project_name write_code_filters = any_to_str_set([WriteTasks, SummarizeCode, FixBug]) summarize_code_filters = any_to_str_set([WriteCode, WriteCodeReview]) if not self.rc.news: diff --git a/metagpt/utils/git_repository.py b/metagpt/utils/git_repository.py index e9855df05..4feed89d5 100644 --- a/metagpt/utils/git_repository.py +++ b/metagpt/utils/git_repository.py @@ -199,10 +199,17 @@ class GitRepository: if new_path.exists(): logger.info(f"Delete directory {str(new_path)}") shutil.rmtree(new_path) + if new_path.exists(): # Recheck for windows os + logger.warning(f"Failed to delete directory {str(new_path)}") + return try: shutil.move(src=str(self.workdir), dst=str(new_path)) except Exception as e: logger.warning(f"Move {str(self.workdir)} to {str(new_path)} error: {e}") + finally: + if not new_path.exists(): # Recheck for windows os + logger.warning(f"Failed to move {str(self.workdir)} to {str(new_path)}") + return logger.info(f"Rename directory {str(self.workdir)} to {str(new_path)}") self._repository = Repo(new_path) self._gitignore_rules = parse_gitignore(full_path=str(new_path / ".gitignore")) From 12ac57af4c66bfb0e92e36120d65eccc9af77e2d Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Mon, 8 Jan 2024 20:54:52 +0800 Subject: [PATCH 272/637] release 0.6.3 --- setup.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/setup.py b/setup.py index 10938c769..d997b5f62 100644 --- a/setup.py +++ b/setup.py @@ -57,7 +57,7 @@ extras_require["dev"] = (["pylint~=3.0.3", "black~=23.3.0", "isort~=5.12.0", "pr setup( name="metagpt", - version="0.6.2", + version="0.6.3", description="The Multi-Agent Framework", long_description=long_description, long_description_content_type="text/markdown", From 7a5485bc8bbe1e8c3044311042e8f2748c2b6bb6 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 9 Jan 2024 10:49:10 +0800 Subject: [PATCH 273/637] Revert changes to file conftest.py --- tests/metagpt/test_incremental_dev.py | 191 ++++++++++++++++---------- 1 file changed, 119 insertions(+), 72 deletions(-) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 216b54952..e86e6d5ff 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -6,6 +6,7 @@ @File : test_incremental_dev.py """ import os +import subprocess import pytest from typer.testing import CliRunner @@ -33,8 +34,15 @@ def test_refined_simple_calculator(): if "Aborting" in result.output: assert False else: - os.system("git tag refine") - assert True + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine"], check=True) + except subprocess.CalledProcessError as e: + raise e def test_refined_number_guessing_game(): @@ -53,8 +61,42 @@ def test_refined_number_guessing_game(): if "Aborting" in result.output: assert False else: - os.system("git tag refine") - assert True + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine"], check=True) + except subprocess.CalledProcessError as e: + raise e + + +def test_refined_word_cloud(): + project_path = f"{DATA_PATH}/word_cloud" + check_or_create_base_tag(project_path) + + args = [ + "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove.", + "--inc", + "--project-path", + project_path, + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine"], check=True) + except subprocess.CalledProcessError as e: + raise e def test_refined_dice_simulator_1(): @@ -73,8 +115,15 @@ def test_refined_dice_simulator_1(): if "Aborting" in result.output: assert False else: - os.system("git tag refine_1") - assert True + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine_1"], check=True) + except subprocess.CalledProcessError as e: + raise e def test_refined_dice_simulator_2(): @@ -93,8 +142,15 @@ def test_refined_dice_simulator_2(): if "Aborting" in result.output: assert False else: - os.system("git tag refine_2") - assert True + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine_2"], check=True) + except subprocess.CalledProcessError as e: + raise e def test_refined_dice_simulator_3(): @@ -113,8 +169,15 @@ def test_refined_dice_simulator_3(): if "Aborting" in result.output: assert False else: - os.system("git tag refine_3") - assert True + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine_3"], check=True) + except subprocess.CalledProcessError as e: + raise e def test_refined_pygame_2048_1(): @@ -133,8 +196,15 @@ def test_refined_pygame_2048_1(): if "Aborting" in result.output: assert False else: - os.system("git tag refine_1") - assert True + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine_1"], check=True) + except subprocess.CalledProcessError as e: + raise e def test_refined_pygame_2048_2(): @@ -153,8 +223,15 @@ def test_refined_pygame_2048_2(): if "Aborting" in result.output: assert False else: - os.system("git tag refine_2") - assert True + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine_2"], check=True) + except subprocess.CalledProcessError as e: + raise e def test_refined_pygame_2048_3(): @@ -173,48 +250,15 @@ def test_refined_pygame_2048_3(): if "Aborting" in result.output: assert False else: - os.system("git tag refine_3") - assert True - - -def test_refined_word_cloud_1(): - project_path = f"{DATA_PATH}/word_cloud" - check_or_create_base_tag(project_path) - - args = [ - "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove.", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - os.system("git tag refine_1") - assert True - - -def test_refined_word_cloud_2(): - project_path = f"{DATA_PATH}/word_cloud" - check_or_create_base_tag(project_path) - - args = [ - "Add a feature to customize the resolution of the word cloud.The new version allows users to customize the size and resolution of the generated word cloud after uploading a text file, and then generate the word cloud.", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - os.system("git tag refine_2") - assert True + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine_3"], check=True) + except subprocess.CalledProcessError as e: + raise e def check_or_create_base_tag(project_path): @@ -222,47 +266,50 @@ def check_or_create_base_tag(project_path): os.chdir(project_path) # Initialize a Git repository - os.system("git init") + subprocess.run(["git", "init"], check=True) # Check if the 'base' tag exists - check_base_tag_cmd = "git show-ref --verify --quiet refs/tags/base" - has_base_tag = os.system(check_base_tag_cmd) == 0 + check_base_tag_cmd = ["git", "show-ref", "--verify", "--quiet", "refs/tags/base"] + if subprocess.run(check_base_tag_cmd).returncode == 0: + has_base_tag = True + else: + has_base_tag = False if has_base_tag: logger.info("Base tag exists") # Switch to the 'base' branch if it exists - stash_cmd = "git stash" - switch_to_base_branch_cmd = "git checkout base" + stash_cmd = ["git", "stash"] + switch_to_base_branch_cmd = ["git", "checkout", "-f", "base"] try: - os.system(stash_cmd) - os.system(switch_to_base_branch_cmd) + subprocess.run(stash_cmd, check=True) + subprocess.run(switch_to_base_branch_cmd, check=True) logger.info("Switched to base branch") except Exception as e: - logger.info("Failed to switch to base branch") + logger.error("Failed to switch to base branch") raise e else: logger.info("Base tag doesn't exist.") # Add and commit the current code if 'base' tag doesn't exist - add_cmd = "git add ." - commit_cmd = 'git commit -m "Initial commit"' + add_cmd = ["git", "add", "."] + commit_cmd = ["git", "commit", "-m", "Initial commit"] try: - os.system(add_cmd) - os.system(commit_cmd) + subprocess.run(add_cmd, check=True) + subprocess.run(commit_cmd, check=True) logger.info("Added and committed all files with the message 'Initial commit'.") except Exception as e: - logger.info("Failed to add and commit all files.") + logger.error("Failed to add and commit all files.") raise e # Add 'base' tag - add_base_tag_cmd = "git tag base" + add_base_tag_cmd = ["git", "tag", "base"] # Check if the 'git tag' command was successful try: - os.system(add_base_tag_cmd) + subprocess.run(add_base_tag_cmd, check=True) logger.info("Added 'base' tag.") except Exception as e: - logger.info("Failed to add 'base' tag.") + logger.error("Failed to add 'base' tag.") raise e From fd9f2416ff672e32ee0d5a8aa4251e9ec3795662 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Tue, 9 Jan 2024 16:14:44 +0800 Subject: [PATCH 274/637] add timeout and retry when code execution --- metagpt/actions/execute_code.py | 57 +++++++++++++++++----- tests/metagpt/actions/test_execute_code.py | 9 ++++ 2 files changed, 54 insertions(+), 12 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 6e4a6fd6e..ab8019e23 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -12,6 +12,8 @@ import re import nbformat from nbclient import NotebookClient +from nbclient.exceptions import DeadKernelError, CellTimeoutError +from nbformat import NotebookNode from nbformat.v4 import new_code_cell, new_output from rich.console import Console from rich.syntax import Syntax @@ -46,13 +48,23 @@ class ExecuteCode(ABC): class ExecutePyCode(ExecuteCode, Action): """execute code, return result to llm, and display it.""" - def __init__(self, name: str = "python_executor", context=None, llm=None, nb=None): + def __init__( + self, + name: str = "python_executor", + context=None, + llm=None, + nb=None, + timeout: int = 600, + max_tries: int = 3 + ): super().__init__(name, context, llm) if nb is None: self.nb = nbformat.v4.new_notebook() else: self.nb = nb - self.nb_client = NotebookClient(self.nb) + self.timeout = timeout + self.max_tries = max_tries + self.nb_client = NotebookClient(self.nb, timeout=self.timeout) self.console = Console() self.interaction = "ipython" if self.is_ipython() else "terminal" @@ -69,7 +81,8 @@ class ExecutePyCode(ExecuteCode, Action): async def reset(self): """reset NotebookClient""" await self.terminate() - self.nb_client = NotebookClient(self.nb) + await self.build() + self.nb_client = NotebookClient(self.nb, timeout=self.timeout) def add_code_cell(self, code): self.nb.cells.append(new_code_cell(source=code)) @@ -160,6 +173,19 @@ class ExecutePyCode(ExecuteCode, Action): return code, language + async def run_cell(self, cell: NotebookNode, cell_index: int) -> Tuple[bool, str]: + """set timeout for run code""" + try: + await self.nb_client.async_execute_cell(cell, cell_index) + return True, "" + except CellTimeoutError: + return False, "TimeoutError" + except DeadKernelError: + await self.reset() + return False, "DeadKernelError" + except Exception as e: + return False, f"{traceback.format_exc()}" + async def run(self, code: Union[str, Dict, Message], language: str = "python") -> Tuple[str, bool]: code, language = self._process_code(code, language) @@ -168,19 +194,26 @@ class ExecutePyCode(ExecuteCode, Action): if language == "python": # add code to the notebook self.add_code_cell(code=code) - try: + + tries = 0 + success = False + outputs = "" + while tries < self.max_tries and not success: # build code executor await self.build() # run code - # TODO: add max_tries for run code. cell_index = len(self.nb.cells) - 1 - await self.nb_client.async_execute_cell(self.nb.cells[-1], cell_index) - outputs = self.parse_outputs(self.nb.cells[-1].outputs) - success = True - except Exception as e: - outputs = traceback.format_exc() - success = False - return truncate(remove_escape_and_color_codes(outputs)), success + success, error_message = await self.run_cell(self.nb.cells[-1], cell_index) + + if success: + outputs = self.parse_outputs(self.nb.cells[-1].outputs) + else: + tries += 1 + + if success: + return truncate(remove_escape_and_color_codes(outputs)), True + else: + return error_message, False else: # TODO: markdown raise NotImplementedError(f"Not support this code type : {language}, Only support code!") diff --git a/tests/metagpt/actions/test_execute_code.py b/tests/metagpt/actions/test_execute_code.py index 95f883e12..8340272e4 100644 --- a/tests/metagpt/actions/test_execute_code.py +++ b/tests/metagpt/actions/test_execute_code.py @@ -88,3 +88,12 @@ def test_truncate(): assert truncate(output) == output output = "hello world" assert truncate(output, 5) == "Truncated to show only the last 5 characters\nworld" + + +@pytest.mark.asyncio +async def test_run_with_timeout(): + pi = ExecutePyCode(timeout=1) + code = "import time; time.sleep(2)" + message, success = await pi.run(code) + assert not success + assert message == "TimeoutError" From 851ec41380490571018d38e46bd9a500bdc82496 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 9 Jan 2024 16:54:36 +0800 Subject: [PATCH 275/637] fix task type issue; add TaskResult data type --- metagpt/actions/write_plan.py | 4 ++-- metagpt/plan/planner.py | 24 ++++++++++----------- metagpt/prompts/ml_engineer.py | 2 +- metagpt/roles/code_interpreter.py | 17 ++++++--------- metagpt/roles/ml_engineer.py | 2 +- metagpt/roles/role.py | 17 ++++++++------- metagpt/schema.py | 14 ++++++++++++ tests/metagpt/roles/run_code_interpreter.py | 13 ++++------- 8 files changed, 49 insertions(+), 44 deletions(-) diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index 11a3f3e1e..d90138d46 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -10,7 +10,7 @@ from copy import deepcopy import traceback from metagpt.actions import Action -from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_PROMPT, ASSIGN_TASK_TYPE +from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_PROMPT, ASSIGN_TASK_TYPE_CONFIG from metagpt.schema import Message, Task, Plan from metagpt.utils.common import CodeParser, create_func_config from metagpt.logs import logger @@ -50,7 +50,7 @@ class WritePlan(Action): [f"Task {task['task_id']}: {task['instruction']}" for task in tasks] ) prompt = ASSIGN_TASK_TYPE_PROMPT.format(task_list=task_list) - tool_config = create_func_config(ASSIGN_TASK_TYPE) + tool_config = create_func_config(ASSIGN_TASK_TYPE_CONFIG) rsp = await self.llm.aask_code(prompt, **tool_config) task_type_list = rsp["task_type"] for task, task_type in zip(tasks, task_type_list): diff --git a/metagpt/plan/planner.py b/metagpt/plan/planner.py index c2b430817..86b197256 100644 --- a/metagpt/plan/planner.py +++ b/metagpt/plan/planner.py @@ -2,7 +2,7 @@ import json from metagpt.logs import logger from metagpt.memory import Memory -from metagpt.schema import Message, Plan, Task +from metagpt.schema import Message, Plan, Task, TaskResult from metagpt.actions.ask_review import AskReview, ReviewConst from metagpt.actions.write_plan import WritePlan, update_plan_from_rsp, precheck_update_plan_from_rsp @@ -20,9 +20,10 @@ STRUCTURAL_CONTEXT = """ class Planner: - def __init__(self, goal: str, working_memory: Memory, auto_run: bool = False): + def __init__(self, goal: str, working_memory: Memory, auto_run: bool = False, use_tools: bool = False): self.plan = Plan(goal=goal) self.auto_run = auto_run + self.use_tools = use_tools # memory for working on each task, discarded each time a task is done self.working_memory = working_memory @@ -35,7 +36,7 @@ class Planner: def current_task_id(self): return self.plan.current_task_id - async def ask_review(self, task_to_review: Task = None, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): + async def ask_review(self, task_result: TaskResult = None, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): """ Ask to review the task result, reviewer needs to provide confirmation or request change. If human confirms the task result, then we deem the task completed, regardless of whether the code run succeeds; @@ -48,12 +49,11 @@ class Planner: if not confirmed: self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) return review, confirmed - confirmed = task_to_review.is_success if task_to_review else True + confirmed = task_result.is_success if task_result else True return "", confirmed - async def confirm_task(self, task, updated_task, review): - assert updated_task.task_id == task.task_id - self.plan.replace_task(updated_task) + async def confirm_task(self, task: Task, task_result: TaskResult, review: str): + self.plan.update_task_result(task=task, task_result=task_result) self.plan.finish_current_task() self.working_memory.clear() @@ -63,13 +63,11 @@ class Planner: self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) await self.update_plan(review) - async def update_plan(self, review: str = "", max_tasks: int = 3, max_retries: int = 3, **kwargs): + async def update_plan(self, max_tasks: int = 3, max_retries: int = 3): plan_confirmed = False while not plan_confirmed: context = self.get_useful_memories() - rsp = await WritePlan().run( - context, max_tasks=max_tasks, **kwargs - ) + rsp = await WritePlan().run(context, max_tasks=max_tasks, use_tools=self.use_tools) self.working_memory.add( Message(content=rsp, role="assistant", cause_by=WritePlan) ) @@ -85,10 +83,10 @@ class Planner: _, plan_confirmed = await self.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - update_plan_from_rsp(rsp, self.plan) + update_plan_from_rsp(rsp=rsp, current_plan=self.plan) self.working_memory.clear() - + def get_useful_memories(self, task_exclude_field=None) -> list[Message]: """find useful memories only to reduce context length and improve performance""" # TODO dataset description , code steps diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index c4b0ad8ae..8fde85d86 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -61,7 +61,7 @@ Please assign a task type to each task in the list below from the given categori - **other**: Any tasks that do not fit into the previous categories, such as visualization, summarizing findings, etc. """ -ASSIGN_TASK_TYPE = { +ASSIGN_TASK_TYPE_CONFIG = { "name": "assign_task_type", "description": "Assign task type to each task by order.", "parameters": { diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 32f530548..437f15698 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -6,16 +6,16 @@ from metagpt.actions.ask_review import ReviewConst from metagpt.actions.write_analysis_code import WriteCodeByGenerate from metagpt.logs import logger from metagpt.roles import Role -from metagpt.schema import Message, Task +from metagpt.schema import Message, Task, TaskResult from metagpt.utils.save_code import save_code_file class CodeInterpreter(Role): def __init__( - self, name="Charlie", profile="CodeInterpreter", goal="", auto_run=False, + self, name="Charlie", profile="CodeInterpreter", goal="", auto_run=False, use_tools=False, ): super().__init__(name=name, profile=profile, goal=goal) - self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run) + self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run, use_tools=use_tools) self.execute_code = ExecutePyCode() @property @@ -32,13 +32,10 @@ class CodeInterpreter(Role): return rsp - async def _act_on_task(self, current_task) -> Task: - code, result, success = await self._write_and_exec_code() - task_copy_with_result = current_task.copy( - update={"code": code, "result": result, "is_success": success}, - deep=True - ) - return task_copy_with_result + async def _act_on_task(self, current_task: Task) -> TaskResult: + code, result, is_success = await self._write_and_exec_code() + task_result = TaskResult(code=code, result=result, is_success=is_success) + return task_result async def _write_and_exec_code(self, max_retry: int = 3): diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index e29d8fce5..eef6dbd21 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -20,7 +20,7 @@ class MLEngineer(CodeInterpreter): self, name="Mark", profile="MLEngineer", goal="", auto_run=False, use_tools=False, use_code_steps=False, make_udfs=False, use_udfs=False ): - super().__init__(name=name, profile=profile, goal=goal, auto_run=auto_run) + super().__init__(name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools) self._watch([DownloadData, SubmitResult]) self.use_tools = use_tools diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 8c68a7ab4..8f1536d39 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -18,7 +18,7 @@ from metagpt.actions import Action, ActionOutput from metagpt.llm import LLM, HumanProvider from metagpt.logs import logger from metagpt.memory import Memory, LongTermMemory -from metagpt.schema import Message, Task +from metagpt.schema import Message, Task, TaskResult from metagpt.plan.planner import Planner PREFIX_TEMPLATE = """You are a {profile}, named {name}, your goal is {goal}, and the constraint is {constraints}. """ @@ -137,7 +137,7 @@ class Role: self._actions.append(i) self._states.append(f"{idx}. {action}") - def _set_react_mode(self, react_mode: str, max_react_loop: int = 1, auto_run: bool = True): + def _set_react_mode(self, react_mode: str, max_react_loop: int = 1, auto_run: bool = True, use_tools: bool = False): """Set strategy of the Role reacting to observed Message. Variation lies in how this Role elects action to perform during the _think stage, especially if it is capable of multiple Actions. @@ -158,7 +158,7 @@ class Role: if react_mode == RoleReactMode.REACT: self._rc.max_react_loop = max_react_loop elif react_mode == RoleReactMode.PLAN_AND_ACT: - self.planner = Planner(goal=self._setting.goal, working_memory=self._rc.working_memory, auto_run=auto_run) + self.planner = Planner(goal=self._setting.goal, working_memory=self._rc.working_memory, auto_run=auto_run, use_tools=use_tools) def _watch(self, actions: Iterable[Type[Action]]): """Listen to the corresponding behaviors""" @@ -285,18 +285,19 @@ class Role: await self.planner.update_plan() while self.planner.current_task: + task = self.planner.current_task logger.info(f"ready to take on task {task}") # take on current task - task_copy_with_result = await self._act_on_task(task) + task_result = await self._act_on_task(task) # ask for acceptance, users can other refuse and change tasks in the plan - review, task_result_confirmed = await self.planner.ask_review(task_copy_with_result) + review, task_result_confirmed = await self.planner.ask_review(task_result) if task_result_confirmed: # tick off this task and record progress - await self.planner.confirm_task(task, task_copy_with_result, review) + await self.planner.confirm_task(task, task_result, review) elif "redo" in review: # Ask the Role to redo this task with help of review feedback, @@ -315,7 +316,7 @@ class Role: return rsp - async def _act_on_task(self, current_task: Task) -> Task: + async def _act_on_task(self, current_task: Task) -> TaskResult: """Taking specific action to handle one task in plan Args: @@ -325,7 +326,7 @@ class Role: NotImplementedError: Specific Role must implement this method if expected to use planner Returns: - Task: A copy of the current task with result from actions + TaskResult: Result from the actions """ raise NotImplementedError diff --git a/metagpt/schema.py b/metagpt/schema.py index f46da0fde..adf30dffe 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -85,6 +85,14 @@ class Task(BaseModel): is_finished: bool = False +class TaskResult(BaseModel): + """Result of taking a task, with result and is_success required to be filled""" + code_steps: str = "" + code: str = "" + result: str + is_success: bool + + class Plan(BaseModel): goal: str context: str = "" @@ -215,6 +223,12 @@ class Plan(BaseModel): self.tasks.append(new_task) self.task_map[new_task.task_id] = new_task self._update_current_task() + + def update_task_result(self, task: Task, task_result: TaskResult): + task.code_steps = task_result.code_steps + task.code = task_result.code + task.result = task_result.result + task.is_success = task_result.is_success def has_task_id(self, task_id: str) -> bool: return task_id in self.task_map diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py index daa6bbe05..51506e7e5 100644 --- a/tests/metagpt/roles/run_code_interpreter.py +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -23,7 +23,7 @@ async def run_code_interpreter(role_class, requirement, auto_run, use_tools, use """ if role_class == "ci": - role = CodeInterpreter(goal=requirement, auto_run=auto_run) + role = CodeInterpreter(goal=requirement, auto_run=auto_run, use_tools=use_tools) else: role = MLEngineer( goal=requirement, auto_run=auto_run, use_tools=use_tools, use_code_steps=use_code_steps, @@ -62,17 +62,12 @@ if __name__ == "__main__": # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." save_dir = "" - # save_dir = DATA_PATH / "output" / "2023-12-14_20-40-34" - role_class = "ci" - # role_class = "mle" + # role_class = "ci" + role_class = "mle" auto_run = True - # auto_run = False - # use_tools = True - use_tools = False - # make_udfs = True + use_tools = True make_udfs = False - # use_udfs = True use_udfs = False async def main( From 3eee6eff8c7b2112cede5084cbe8b8fd81c4190b Mon Sep 17 00:00:00 2001 From: lidanyang Date: Tue, 9 Jan 2024 17:45:46 +0800 Subject: [PATCH 276/637] drop retry --- metagpt/actions/execute_code.py | 21 ++++++--------------- 1 file changed, 6 insertions(+), 15 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index ab8019e23..d192ca79a 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -55,7 +55,6 @@ class ExecutePyCode(ExecuteCode, Action): llm=None, nb=None, timeout: int = 600, - max_tries: int = 3 ): super().__init__(name, context, llm) if nb is None: @@ -63,7 +62,6 @@ class ExecutePyCode(ExecuteCode, Action): else: self.nb = nb self.timeout = timeout - self.max_tries = max_tries self.nb_client = NotebookClient(self.nb, timeout=self.timeout) self.console = Console() self.interaction = "ipython" if self.is_ipython() else "terminal" @@ -195,22 +193,15 @@ class ExecutePyCode(ExecuteCode, Action): # add code to the notebook self.add_code_cell(code=code) - tries = 0 - success = False - outputs = "" - while tries < self.max_tries and not success: - # build code executor - await self.build() - # run code - cell_index = len(self.nb.cells) - 1 - success, error_message = await self.run_cell(self.nb.cells[-1], cell_index) + # build code executor + await self.build() - if success: - outputs = self.parse_outputs(self.nb.cells[-1].outputs) - else: - tries += 1 + # run code + cell_index = len(self.nb.cells) - 1 + success, error_message = await self.run_cell(self.nb.cells[-1], cell_index) if success: + outputs = self.parse_outputs(self.nb.cells[-1].outputs) return truncate(remove_escape_and_color_codes(outputs)), True else: return error_message, False From a9e0b67b7187474efe1272b3c294b6c1a3b75110 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 10 Jan 2024 14:13:48 +0800 Subject: [PATCH 277/637] update test_incremental_dev.py --- tests/metagpt/test_incremental_dev.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index e86e6d5ff..2ff33dff3 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -278,11 +278,11 @@ def check_or_create_base_tag(project_path): if has_base_tag: logger.info("Base tag exists") # Switch to the 'base' branch if it exists - stash_cmd = ["git", "stash"] - switch_to_base_branch_cmd = ["git", "checkout", "-f", "base"] try: - subprocess.run(stash_cmd, check=True) - subprocess.run(switch_to_base_branch_cmd, check=True) + status = subprocess.run(["git", "status", "-s"], capture_output=True, text=True).stdout.strip() + if status: + subprocess.run(["git", "clean", "-df"]) + subprocess.run(["git", "checkout", "-f", "base"], check=True) logger.info("Switched to base branch") except Exception as e: logger.error("Failed to switch to base branch") From 767c99388f64d0017452b48955d586c99d9e9046 Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 10 Jan 2024 14:15:30 +0800 Subject: [PATCH 278/637] format using precommit --- kaggle_team.py | 7 +- metagpt/actions/ask_review.py | 13 +- metagpt/actions/debug_code.py | 12 +- metagpt/actions/execute_code.py | 24 ++-- metagpt/actions/ml_da_action.py | 11 +- metagpt/actions/write_analysis_code.py | 72 +++++------ metagpt/actions/write_code_steps.py | 22 ++-- metagpt/actions/write_plan.py | 20 ++- metagpt/plan/__init__.py | 1 - metagpt/plan/planner.py | 44 ++++--- metagpt/prompts/ml_engineer.py | 2 +- metagpt/provider/openai_api.py | 1 - metagpt/roles/code_interpreter.py | 40 +++--- metagpt/roles/kaggle_manager.py | 44 +++---- metagpt/roles/ml_engineer.py | 116 ++++++++++-------- metagpt/roles/ml_engineer_simple.py | 41 +++---- metagpt/roles/role.py | 28 ++--- metagpt/schema.py | 31 ++--- .../tools/functions/libs/data_preprocess.py | 64 +++++++--- .../functions/libs/feature_engineering.py | 35 +++--- metagpt/tools/functions/libs/udf/__init__.py | 65 +++++----- metagpt/utils/common.py | 1 + metagpt/utils/recovery_util.py | 22 ++-- metagpt/utils/save_code.py | 9 +- tests/metagpt/test_schema.py | 31 +++-- 25 files changed, 376 insertions(+), 380 deletions(-) diff --git a/kaggle_team.py b/kaggle_team.py index 50a8f7288..e9f3e67de 100644 --- a/kaggle_team.py +++ b/kaggle_team.py @@ -1,6 +1,5 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- -import asyncio import fire @@ -8,6 +7,7 @@ from metagpt.roles.kaggle_manager import KaggleManager from metagpt.roles.ml_engineer import MLEngineer from metagpt.team import Team + async def main( # competition: str, # data_desc: str, @@ -21,7 +21,7 @@ async def main( "Training set is train.csv.\nTest set is test.csv. We also include gender_submission.csv, a set of predictions that assume all and only female passengers survive, as an example of what a submission file should look like.", # "Run EDA on the train dataset, train a model to predict survival (20% as validation) and save it, predict the test set using saved model, save the test result according to format", # "generate a random prediction, replace the Survived column of gender_submission.csv, and save the prediction to a new submission file", - "Score as high as possible for the provided dataset, save the test prediction to a csv with two columns PassengerId and Survived" + "Score as high as possible for the provided dataset, save the test prediction to a csv with two columns PassengerId and Survived", ) team = Team() @@ -36,5 +36,6 @@ async def main( team.start_project(requirement) await team.run(n_round=n_round) -if __name__ == '__main__': + +if __name__ == "__main__": fire.Fire(main) diff --git a/metagpt/actions/ask_review.py b/metagpt/actions/ask_review.py index eec5e49aa..85ac33bd8 100644 --- a/metagpt/actions/ask_review.py +++ b/metagpt/actions/ask_review.py @@ -1,8 +1,8 @@ from typing import List from metagpt.actions import Action -from metagpt.schema import Message, Plan from metagpt.logs import logger +from metagpt.schema import Message, Plan class ReviewConst: @@ -23,17 +23,10 @@ class ReviewConst: class AskReview(Action): - async def run( - self, context: List[Message], plan: Plan = None, trigger: str = "task" - ): + async def run(self, context: List[Message], plan: Plan = None, trigger: str = "task"): logger.info("Current overall plan:") logger.info( - "\n".join( - [ - f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" - for task in plan.tasks - ] - ) + "\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks]) ) logger.info("most recent context:") diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 3e1705d8e..be09f3493 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -1,9 +1,9 @@ -from typing import Dict, List, Union, Tuple, Optional, Any +from typing import Any, List, Optional -from metagpt.logs import logger -from metagpt.schema import Message, Plan -from metagpt.utils.common import CodeParser, create_func_config from metagpt.actions.write_analysis_code import BaseWriteAnalysisCode +from metagpt.logs import logger +from metagpt.schema import Message +from metagpt.utils.common import create_func_config DEBUG_REFLECTION_EXAMPLE = ''' Example 1: @@ -113,9 +113,7 @@ class DebugCode(BaseWriteAnalysisCode): # msg = messages_to_str(info) # resp = await self.llm.aask(msg=msg) - resp = await self.llm.aask_code( - messages=info, **create_func_config(CODE_REFLECTION) - ) + resp = await self.llm.aask_code(messages=info, **create_func_config(CODE_REFLECTION)) logger.info(f"reflection is {resp}") return resp diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index d192ca79a..b2f6067ab 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -4,23 +4,23 @@ @Author : orange-crow @File : code_executor.py """ +import re +import traceback from abc import ABC, abstractmethod from pathlib import Path from typing import Dict, List, Tuple, Union -import traceback -import re import nbformat from nbclient import NotebookClient -from nbclient.exceptions import DeadKernelError, CellTimeoutError +from nbclient.exceptions import CellTimeoutError, DeadKernelError from nbformat import NotebookNode from nbformat.v4 import new_code_cell, new_output from rich.console import Console from rich.syntax import Syntax from metagpt.actions import Action -from metagpt.schema import Message from metagpt.logs import logger +from metagpt.schema import Message class ExecuteCode(ABC): @@ -113,7 +113,9 @@ class ExecutePyCode(ExecuteCode, Action): if "image/png" in output["data"]: self.show_bytes_figure(output["data"]["image/png"], self.interaction) else: - logger.info(f"{i}th output['data'] from nbclient outputs dont have image/png, continue next output ...") + logger.info( + f"{i}th output['data'] from nbclient outputs dont have image/png, continue next output ..." + ) elif output["output_type"] == "execute_result": parsed_output += output["data"]["text/plain"] return parsed_output @@ -148,7 +150,7 @@ class ExecutePyCode(ExecuteCode, Action): return False def _process_code(self, code: Union[str, Dict, Message], language: str = None) -> Tuple: - language = language or 'python' + language = language or "python" if isinstance(code, str) and Path(code).suffix in (".py", ".txt"): code = Path(code).read_text(encoding="utf-8") return code, language @@ -158,11 +160,11 @@ class ExecutePyCode(ExecuteCode, Action): if isinstance(code, dict): assert "code" in code if "language" not in code: - code['language'] = 'python' + code["language"] = "python" code, language = code["code"], code["language"] elif isinstance(code, Message): if isinstance(code.content, dict) and "language" not in code.content: - code.content["language"] = 'python' + code.content["language"] = "python" code, language = code.content["code"], code.content["language"] elif isinstance(code.content, str): code, language = code.content, language @@ -181,7 +183,7 @@ class ExecutePyCode(ExecuteCode, Action): except DeadKernelError: await self.reset() return False, "DeadKernelError" - except Exception as e: + except Exception: return False, f"{traceback.format_exc()}" async def run(self, code: Union[str, Dict, Message], language: str = "python") -> Tuple[str, bool]: @@ -224,6 +226,6 @@ def truncate(result: str, keep_len: int = 2000) -> str: def remove_escape_and_color_codes(input_str): # 使用正则表达式去除转义字符和颜色代码 - pattern = re.compile(r'\x1b\[[0-9;]*[mK]') - result = pattern.sub('', input_str) + pattern = re.compile(r"\x1b\[[0-9;]*[mK]") + result = pattern.sub("", input_str) return result diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py index 50d1d2420..3ab5e0429 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_da_action.py @@ -1,14 +1,9 @@ import json -from typing import Dict, List, Union from metagpt.actions import Action -from metagpt.schema import Message, Plan -from metagpt.utils.common import CodeParser, remove_comments, create_func_config -from metagpt.logs import logger -from metagpt.prompts.ml_engineer import ( - UPDATE_DATA_COLUMNS, - PRINT_DATA_COLUMNS -) +from metagpt.prompts.ml_engineer import PRINT_DATA_COLUMNS, UPDATE_DATA_COLUMNS +from metagpt.schema import Plan +from metagpt.utils.common import CodeParser, create_func_config, remove_comments class SummarizeAnalysis(Action): diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 21add3159..b0c8dab3b 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -4,25 +4,24 @@ @Author : orange-crow @File : write_code_v2.py """ -from typing import Dict, List, Union, Tuple -from tenacity import retry, stop_after_attempt, wait_fixed -from pathlib import Path import re -import json +from pathlib import Path +from typing import Dict, List, Tuple, Union import yaml +from tenacity import retry, stop_after_attempt, wait_fixed from metagpt.actions import Action from metagpt.llm import LLM from metagpt.logs import logger from metagpt.prompts.ml_engineer import ( - TOOL_RECOMMENDATION_PROMPT, - SELECT_FUNCTION_TOOLS, CODE_GENERATOR_WITH_TOOLS, - TOOL_USAGE_PROMPT, - ML_SPECIFIC_PROMPT, - ML_MODULE_MAP, GENERATE_CODE_PROMPT, + ML_MODULE_MAP, + ML_SPECIFIC_PROMPT, + SELECT_FUNCTION_TOOLS, + TOOL_RECOMMENDATION_PROMPT, + TOOL_USAGE_PROMPT, ) from metagpt.schema import Message, Plan from metagpt.utils.common import create_func_config, remove_comments @@ -52,24 +51,16 @@ class BaseWriteAnalysisCode(Action): messages.append(p.content["code"]) # 添加默认的提示词 - if ( - default_system_msg not in messages[0]["content"] - and messages[0]["role"] != "system" - ): + if default_system_msg not in messages[0]["content"] and messages[0]["role"] != "system": messages.insert(0, {"role": "system", "content": default_system_msg}) - elif ( - default_system_msg not in messages[0]["content"] - and messages[0]["role"] == "system" - ): + elif default_system_msg not in messages[0]["content"] and messages[0]["role"] == "system": messages[0] = { "role": "system", "content": messages[0]["content"] + default_system_msg, } return messages - async def run( - self, context: List[Message], plan: Plan = None, code_steps: str = "" - ) -> str: + async def run(self, context: List[Message], plan: Plan = None, code_steps: str = "") -> str: """Run of a code writing action, used in data analysis or modeling Args: @@ -115,7 +106,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): def _load_tools(self, schema_path, schema_module=None): """Load tools from yaml file""" if isinstance(schema_path, dict): - schema_module = schema_module or 'udf' + schema_module = schema_module or "udf" self.available_tools.update({schema_module: schema_path}) else: if isinstance(schema_path, list): @@ -197,9 +188,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): available_tools = {k: v["description"] for k, v in available_tools.items()} recommend_tools = await self._tool_recommendation( - plan.current_task.instruction, - code_steps, - available_tools + plan.current_task.instruction, code_steps, available_tools ) tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) logger.info(f"Recommended tools: \n{recommend_tools}") @@ -216,8 +205,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name=module_name, tool_catalog=tool_catalog, ) - - + else: prompt = GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, @@ -245,7 +233,7 @@ class MakeTools(WriteCodeByGenerate): 5. Only use the imported packages** """ - def __init__(self, name: str = '', context: list[Message] = None, llm: LLM = None, workspace: str = None): + def __init__(self, name: str = "", context: list[Message] = None, llm: LLM = None, workspace: str = None): """ :param str name: name, defaults to '' :param list[Message] context: context, defaults to None @@ -254,12 +242,12 @@ class MakeTools(WriteCodeByGenerate): """ super().__init__(name, context, llm) self.workspace = workspace or str(Path(__file__).parents[1].joinpath("./tools/functions/libs/udf")) - self.file_suffix: str = '.py' + self.file_suffix: str = ".py" self.context = [] def parse_function_name(self, function_code: str) -> str: # 定义正则表达式模式 - pattern = r'\bdef\s+([a-zA-Z_]\w*)\s*\(' + pattern = r"\bdef\s+([a-zA-Z_]\w*)\s*\(" # 在代码中搜索匹配的模式 match = re.search(pattern, function_code) # 如果找到匹配项,则返回匹配的函数名;否则返回None @@ -272,9 +260,9 @@ class MakeTools(WriteCodeByGenerate): func_name = self.parse_function_name(tool_code) if func_name is None: raise ValueError(f"No function name found in {tool_code}") - saved_path = Path(self.workspace).joinpath(func_name+self.file_suffix) + saved_path = Path(self.workspace).joinpath(func_name + self.file_suffix) logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") - saved_path.write_text(tool_code, encoding='utf-8') + saved_path.write_text(tool_code, encoding="utf-8") @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) async def run(self, code: Union[str, List[dict]], code_desc: str = None, **kwargs) -> str: @@ -287,27 +275,31 @@ class MakeTools(WriteCodeByGenerate): logger.info(f"\n\nAsk to Make tools:\n{'-'*60}\n {self.context[-1]}") # 更新kwargs - if 'code' in kwargs: - kwargs.pop('code') - if 'code_desc' in kwargs: - kwargs.pop('code_desc') + if "code" in kwargs: + kwargs.pop("code") + if "code_desc" in kwargs: + kwargs.pop("code_desc") max_tries, current_try = 3, 0 while True: tool_code = await self.llm.aask_code(self.context, **kwargs) - func_name = self.parse_function_name(tool_code['code']) + func_name = self.parse_function_name(tool_code["code"]) current_try += 1 # make tools failed, add error message to context. if not func_name: logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") logger.error(f"No function name found in code, we will retry make tools.\n{tool_code['code']}\n") - self.context.append({'role': 'user', 'content': 'We need a general function in above code,but not found function.'}) + self.context.append( + {"role": "user", "content": "We need a general function in above code,but not found function."} + ) # end make tools if func_name is not None or current_try >= max_tries: if current_try >= max_tries: - logger.error(f"We have tried the maximum number of attempts {max_tries}\ - and still have not created tools successfully, we will skip it.") + logger.error( + f"We have tried the maximum number of attempts {max_tries}\ + and still have not created tools successfully, we will skip it." + ) break logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") - self.save(tool_code['code']) + self.save(tool_code["code"]) return tool_code["code"] diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py index 79f3e5902..7ba22fde4 100644 --- a/metagpt/actions/write_code_steps.py +++ b/metagpt/actions/write_code_steps.py @@ -1,9 +1,7 @@ - import json -from typing import Dict, List, Union from metagpt.actions import Action -from metagpt.schema import Message, Task, Plan +from metagpt.schema import Plan from metagpt.utils.common import CodeParser # CODE_STEPS_PROMPT_TEMPLATE = """ @@ -79,7 +77,6 @@ STRUCTURAL_CONTEXT = """ class WriteCodeSteps(Action): - async def run(self, plan: Plan) -> str: """Run of a task guide writing action, used in ml engineer @@ -91,9 +88,7 @@ class WriteCodeSteps(Action): """ context = self.get_context(plan) - code_steps_prompt = CODE_STEPS_PROMPT_TEMPLATE.replace( - "{context}", context - ) + code_steps_prompt = CODE_STEPS_PROMPT_TEMPLATE.replace("{context}", context) code_steps = await self._aask(code_steps_prompt) code_steps = CodeParser.parse_code(block=None, text=code_steps) return code_steps @@ -102,19 +97,16 @@ class WriteCodeSteps(Action): user_requirement = plan.goal # select_task_keys = ['task_id', 'instruction', 'is_finished', 'code'] # select_task_keys = ['task_id','instruction'] - + def process_task(task): task_dict = task.dict() # ptask = {k: task_dict[k] for k in task_dict if k in select_task_keys } ptask = f"task_id_{task_dict['task_id']}:{task_dict['instruction']}" return ptask - - - tasks = json.dumps( - [process_task(task) for task in plan.tasks], indent=4, ensure_ascii=False - ) - - code_lists = [task.code for task in plan.tasks if task.is_finished==True] + + tasks = json.dumps([process_task(task) for task in plan.tasks], indent=4, ensure_ascii=False) + + code_lists = [task.code for task in plan.tasks if task.is_finished == True] codes = "\n\n".join(code_lists) current_task = json.dumps(process_task(plan.current_task)) if plan.current_task else {} context = STRUCTURAL_CONTEXT.format( diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index d90138d46..d2553e609 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -4,16 +4,15 @@ @Author : orange-crow @File : plan.py """ -from typing import List, Dict, Tuple import json from copy import deepcopy -import traceback +from typing import Dict, List, Tuple from metagpt.actions import Action -from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_PROMPT, ASSIGN_TASK_TYPE_CONFIG -from metagpt.schema import Message, Task, Plan -from metagpt.utils.common import CodeParser, create_func_config from metagpt.logs import logger +from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_CONFIG, ASSIGN_TASK_TYPE_PROMPT +from metagpt.schema import Message, Plan, Task +from metagpt.utils.common import CodeParser, create_func_config class WritePlan(Action): @@ -46,9 +45,7 @@ class WritePlan(Action): Returns: List[Dict]: tasks with task type assigned """ - task_list = "\n".join( - [f"Task {task['task_id']}: {task['instruction']}" for task in tasks] - ) + task_list = "\n".join([f"Task {task['task_id']}: {task['instruction']}" for task in tasks]) prompt = ASSIGN_TASK_TYPE_PROMPT.format(task_list=task_list) tool_config = create_func_config(ASSIGN_TASK_TYPE_CONFIG) rsp = await self.llm.aask_code(prompt, **tool_config) @@ -57,9 +54,7 @@ class WritePlan(Action): task["task_type"] = task_type return json.dumps(tasks) - async def run( - self, context: List[Message], max_tasks: int = 5, use_tools: bool = False - ) -> str: + async def run(self, context: List[Message], max_tasks: int = 5, use_tools: bool = False) -> str: prompt = ( self.PROMPT_TEMPLATE.replace("__context__", "\n".join([str(ct) for ct in context])) # .replace("__current_plan__", current_plan) @@ -71,11 +66,13 @@ class WritePlan(Action): rsp = await self.assign_task_type(json.loads(rsp)) return rsp + def rsp_to_tasks(rsp: str) -> List[Task]: rsp = json.loads(rsp) tasks = [Task(**task_config) for task_config in rsp] return tasks + def update_plan_from_rsp(rsp: str, current_plan: Plan): tasks = rsp_to_tasks(rsp) if len(tasks) == 1 or tasks[0].dependent_task_ids: @@ -97,6 +94,7 @@ def update_plan_from_rsp(rsp: str, current_plan: Plan): # add tasks in general current_plan.add_tasks(tasks) + def precheck_update_plan_from_rsp(rsp: str, current_plan: Plan) -> Tuple[bool, str]: temp_plan = deepcopy(current_plan) try: diff --git a/metagpt/plan/__init__.py b/metagpt/plan/__init__.py index 5ad35e100..e69de29bb 100644 --- a/metagpt/plan/__init__.py +++ b/metagpt/plan/__init__.py @@ -1 +0,0 @@ -from metagpt.plan.planner import Planner \ No newline at end of file diff --git a/metagpt/plan/planner.py b/metagpt/plan/planner.py index 86b197256..dadc2e563 100644 --- a/metagpt/plan/planner.py +++ b/metagpt/plan/planner.py @@ -1,11 +1,14 @@ import json +from metagpt.actions.ask_review import AskReview, ReviewConst +from metagpt.actions.write_plan import ( + WritePlan, + precheck_update_plan_from_rsp, + update_plan_from_rsp, +) from metagpt.logs import logger from metagpt.memory import Memory from metagpt.schema import Message, Plan, Task, TaskResult -from metagpt.actions.ask_review import AskReview, ReviewConst -from metagpt.actions.write_plan import WritePlan, update_plan_from_rsp, precheck_update_plan_from_rsp - STRUCTURAL_CONTEXT = """ ## User Requirement @@ -27,16 +30,18 @@ class Planner: # memory for working on each task, discarded each time a task is done self.working_memory = working_memory - + @property def current_task(self): return self.plan.current_task - + @property def current_task_id(self): return self.plan.current_task_id - async def ask_review(self, task_result: TaskResult = None, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): + async def ask_review( + self, task_result: TaskResult = None, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER + ): """ Ask to review the task result, reviewer needs to provide confirmation or request change. If human confirms the task result, then we deem the task completed, regardless of whether the code run succeeds; @@ -51,27 +56,26 @@ class Planner: return review, confirmed confirmed = task_result.is_success if task_result else True return "", confirmed - + async def confirm_task(self, task: Task, task_result: TaskResult, review: str): self.plan.update_task_result(task=task, task_result=task_result) self.plan.finish_current_task() self.working_memory.clear() - - confirmed_and_more = (ReviewConst.CONTINUE_WORD[0] in review.lower() - and review.lower() not in ReviewConst.CONTINUE_WORD[0]) # "confirm, ... (more content, such as changing downstream tasks)" + + confirmed_and_more = ( + ReviewConst.CONTINUE_WORD[0] in review.lower() and review.lower() not in ReviewConst.CONTINUE_WORD[0] + ) # "confirm, ... (more content, such as changing downstream tasks)" if confirmed_and_more: self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) await self.update_plan(review) - + async def update_plan(self, max_tasks: int = 3, max_retries: int = 3): plan_confirmed = False while not plan_confirmed: context = self.get_useful_memories() rsp = await WritePlan().run(context, max_tasks=max_tasks, use_tools=self.use_tools) - self.working_memory.add( - Message(content=rsp, role="assistant", cause_by=WritePlan) - ) - + self.working_memory.add(Message(content=rsp, role="assistant", cause_by=WritePlan)) + # precheck plan before asking reviews is_plan_valid, error = precheck_update_plan_from_rsp(rsp, self.plan) if not is_plan_valid and max_retries > 0: @@ -80,11 +84,11 @@ class Planner: self.working_memory.add(Message(content=error_msg, role="assistant", cause_by=WritePlan)) max_retries -= 1 continue - + _, plan_confirmed = await self.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - + update_plan_from_rsp(rsp=rsp, current_plan=self.plan) - + self.working_memory.clear() def get_useful_memories(self, task_exclude_field=None) -> list[Message]: @@ -93,7 +97,7 @@ class Planner: if task_exclude_field is None: # Shorten the context as we don't need code steps after we get the codes. # This doesn't affect current_task below, which should hold the code steps - task_exclude_field = {'code_steps'} + task_exclude_field = {"code_steps"} user_requirement = self.plan.goal context = self.plan.context tasks = [task.dict(exclude=task_exclude_field) for task in self.plan.tasks] @@ -103,5 +107,5 @@ class Planner: user_requirement=user_requirement, context=context, tasks=tasks, current_task=current_task ) context_msg = [Message(content=context, role="user")] - + return context_msg + self.working_memory.get() diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 8fde85d86..9b873d39f 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -259,7 +259,7 @@ for col in num_cols: - Always copy the DataFrame before processing it and use the copy to process. - The output code should contain all steps implemented correctly in 'Code Steps'. """ -#- If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. +# - If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. DATA_PREPROCESS_PROMPT = """ The current task is about data preprocessing, please note the following: diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 85362fca9..747e36480 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -22,7 +22,6 @@ from tenacity import ( retry_if_exception_type, stop_after_attempt, wait_random_exponential, - wait_fixed, ) from metagpt.config import CONFIG, Config, LLMProviderEnum diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 437f15698..25890bc93 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -1,8 +1,7 @@ -import json from datetime import datetime -from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.ask_review import ReviewConst +from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.write_analysis_code import WriteCodeByGenerate from metagpt.logs import logger from metagpt.roles import Role @@ -12,7 +11,12 @@ from metagpt.utils.save_code import save_code_file class CodeInterpreter(Role): def __init__( - self, name="Charlie", profile="CodeInterpreter", goal="", auto_run=False, use_tools=False, + self, + name="Charlie", + profile="CodeInterpreter", + goal="", + auto_run=False, + use_tools=False, ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run, use_tools=use_tools) @@ -21,9 +25,8 @@ class CodeInterpreter(Role): @property def working_memory(self): return self._rc.working_memory - - async def _plan_and_act(self): + async def _plan_and_act(self): rsp = await super()._plan_and_act() # save code using datetime.now or keywords related to the goal of your project (plan.goal). @@ -31,47 +34,40 @@ class CodeInterpreter(Role): save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") return rsp - + async def _act_on_task(self, current_task: Task) -> TaskResult: code, result, is_success = await self._write_and_exec_code() task_result = TaskResult(code=code, result=result, is_success=is_success) return task_result async def _write_and_exec_code(self, max_retry: int = 3): - counter = 0 success = False - + while not success and counter < max_retry: context = self.planner.get_useful_memories() logger.info("Write code with pure generation") - code = await WriteCodeByGenerate().run( - context=context, plan=self.planner.plan, temperature=0.0 - ) + code = await WriteCodeByGenerate().run(context=context, plan=self.planner.plan, temperature=0.0) cause_by = WriteCodeByGenerate - self.working_memory.add( - Message(content=code, role="assistant", cause_by=cause_by) - ) - + self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) + result, success = await self.execute_code.run(code) print(result) - self.working_memory.add( - Message(content=result, role="user", cause_by=ExecutePyCode) - ) - + self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) + if "!pip" in code: success = False - + counter += 1 - + if not success and counter >= max_retry: logger.info("coding failed!") review, _ = await self.planner.ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions - + return code, result, success diff --git a/metagpt/roles/kaggle_manager.py b/metagpt/roles/kaggle_manager.py index cad12a16a..e12f47051 100644 --- a/metagpt/roles/kaggle_manager.py +++ b/metagpt/roles/kaggle_manager.py @@ -1,25 +1,23 @@ -from typing import Dict, List, Union, Tuple import json -import subprocess import os +import subprocess import fire import pandas as pd +from metagpt.actions import Action, BossRequirement +from metagpt.actions.ml_da_action import SummarizeAnalysis from metagpt.config import CONFIG from metagpt.const import WORKSPACE_ROOT -from metagpt.roles import Role -from metagpt.actions import Action, BossRequirement -from metagpt.actions.ask_review import AskReview -from metagpt.actions.ml_da_action import SummarizeAnalysis -from metagpt.schema import Message, Task, Plan from metagpt.logs import logger +from metagpt.roles import Role +from metagpt.schema import Message from metagpt.utils.common import CodeParser - os.environ["KAGGLE_USERNAME"] = CONFIG.kaggle_username os.environ["KAGGLE_KEY"] = CONFIG.kaggle_key + def run_command(cmd): print(cmd) output = subprocess.run(cmd, shell=True, capture_output=True, text=True) @@ -30,21 +28,21 @@ def run_command(cmd): print(output.stdout) return output.stdout -class DownloadData(Action): +class DownloadData(Action): async def run(self, competition, data_desc="") -> str: data_path = WORKSPACE_ROOT / competition - + output = run_command(f"kaggle competitions list --search {competition}") assert output != "No competitions found", "You must provide the correct competition name" - + run_command(f"kaggle competitions download {competition} --path {WORKSPACE_ROOT}") - + if not os.path.exists(data_path): - # if True: + # if True: # run_command(f"rm -r {data_path / '*'}") run_command(f"unzip -o {WORKSPACE_ROOT / '*.zip'} -d {data_path}") # FIXME: not safe - + file_list = run_command(f"ls {data_path}") rsp = f""" @@ -55,6 +53,7 @@ class DownloadData(Action): """ return rsp + class SubmitResult(Action): PROMPT_TEMPLATE = """ # Summary @@ -85,9 +84,9 @@ class SubmitResult(Action): run_command(f"kaggle competitions submit {competition} -f {submit_file_path} -m '{submit_message}'") run_command(f"kaggle competitions leaderboard --show --csv {competition} > {data_path / 'leaderboard.csv'}") run_command(f"kaggle competitions submissions --csv {competition} > {data_path / 'submission.csv'}") - - leaderboard = pd.read_csv(data_path / 'leaderboard.csv') - submission = pd.read_csv(data_path / 'submission.csv') + + leaderboard = pd.read_csv(data_path / "leaderboard.csv") + submission = pd.read_csv(data_path / "submission.csv") print(submission) # submission.to_json(orient="records") submission_score = submission.loc[0, "publicScore"] @@ -106,9 +105,7 @@ class SubmitResult(Action): class KaggleManager(Role): - def __init__( - self, name="ABC", profile="KaggleManager", goal="", competition="titanic", data_desc="" - ): + def __init__(self, name="ABC", profile="KaggleManager", goal="", competition="titanic", data_desc=""): super().__init__(name=name, profile=profile, goal=goal) self._init_actions([DownloadData, SubmitResult]) self._watch([BossRequirement, SummarizeAnalysis]) @@ -130,13 +127,16 @@ class KaggleManager(Role): rsp = await todo.run(self.competition, self.data_desc) elif isinstance(todo, SubmitResult): - submit_message = self.get_memories()[-1].content # use analysis summary from MLEngineer as submission message + submit_message = self.get_memories()[ + -1 + ].content # use analysis summary from MLEngineer as submission message rsp = await todo.run(competition=self.competition, submit_message=submit_message) msg = Message(content=rsp, role="user", cause_by=type(todo)) return msg + if __name__ == "__main__": competition, data_desc, requirement = ( "titanic", @@ -151,4 +151,4 @@ if __name__ == "__main__": # await role.run(Message(content="", cause_by=BossRequirement)) await role.run(Message(content=summary, cause_by=SummarizeAnalysis)) - fire.Fire(main) \ No newline at end of file + fire.Fire(main) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index eef6dbd21..a631daa47 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,36 +1,46 @@ import json +from metagpt.actions.ask_review import ReviewConst from metagpt.actions.debug_code import DebugCode from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.ask_review import ReviewConst -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools, MakeTools +from metagpt.actions.ml_da_action import Reflect, SummarizeAnalysis, UpdateDataColumns +from metagpt.actions.write_analysis_code import ( + MakeTools, + WriteCodeByGenerate, + WriteCodeWithTools, +) from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.const import PROJECT_ROOT from metagpt.logs import logger -from metagpt.schema import Message -from metagpt.utils.common import remove_comments -from metagpt.actions.ml_da_action import SummarizeAnalysis, Reflect, UpdateDataColumns from metagpt.roles.code_interpreter import CodeInterpreter from metagpt.roles.kaggle_manager import DownloadData, SubmitResult +from metagpt.schema import Message from metagpt.tools.functions.libs.udf import UDFS_YAML +from metagpt.utils.common import remove_comments class MLEngineer(CodeInterpreter): def __init__( - self, name="Mark", profile="MLEngineer", goal="", auto_run=False, use_tools=False, use_code_steps=False, - make_udfs=False, use_udfs=False + self, + name="Mark", + profile="MLEngineer", + goal="", + auto_run=False, + use_tools=False, + use_code_steps=False, + make_udfs=False, + use_udfs=False, ): super().__init__(name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools) self._watch([DownloadData, SubmitResult]) self.use_tools = use_tools self.use_code_steps = use_code_steps - self.make_udfs = make_udfs # user-defined functions + self.make_udfs = make_udfs # user-defined functions self.use_udfs = use_udfs self.data_desc = {} - + async def _plan_and_act(self): - ### Actions in a multi-agent multi-turn setting, a new attempt on the data ### memories = self.get_memories() if memories: @@ -40,64 +50,62 @@ class MLEngineer(CodeInterpreter): elif latest_event == SubmitResult: # self reflect on previous plan outcomes and think about how to improve the plan, add to working memory await self._reflect() - + # get feedback for improvement from human, add to working memory await self.planner.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - + ### general plan process ### await super()._plan_and_act() - + ### summarize analysis ### summary = await SummarizeAnalysis().run(self.planner.plan) rsp = Message(content=summary, cause_by=SummarizeAnalysis) self._rc.memory.add(rsp) - + return rsp async def _write_and_exec_code(self, max_retry: int = 3): self.planner.current_task.code_steps = ( - await WriteCodeSteps().run(self.planner.plan) - if self.use_code_steps - else "" + await WriteCodeSteps().run(self.planner.plan) if self.use_code_steps else "" ) - + counter = 0 success = False debug_context = [] - - while not success and counter < max_retry: + while not success and counter < max_retry: context = self.planner.get_useful_memories() if counter > 0 and (self.use_tools or self.use_udfs): - logger.warning('We got a bug code, now start to debug...') + logger.warning("We got a bug code, now start to debug...") code = await DebugCode().run( plan=self.planner.current_task.instruction, code=code, runtime_result=self.working_memory.get(), - context=debug_context + context=debug_context, ) logger.info(f"new code \n{code}") cause_by = DebugCode - + elif (not self.use_tools and not self.use_udfs) or ( - self.planner.current_task.task_type == 'other' and not self.use_udfs): + self.planner.current_task.task_type == "other" and not self.use_udfs + ): logger.info("Write code with pure generation") - code = await WriteCodeByGenerate().run( - context=context, plan=self.planner.plan, temperature=0.0 - ) - debug_context = [self.planner.get_useful_memories(task_exclude_field={'result', 'code_steps'})[0]] + code = await WriteCodeByGenerate().run(context=context, plan=self.planner.plan, temperature=0.0) + debug_context = [self.planner.get_useful_memories(task_exclude_field={"result", "code_steps"})[0]] cause_by = WriteCodeByGenerate - + else: logger.info("Write code with tools") if self.use_udfs: # use user-defined function tools. logger.warning("Writing code with user-defined function tools by WriteCodeWithTools.") - logger.info(f"Local user defined function as following:\ - \n{json.dumps(list(UDFS_YAML.keys()), indent=2, ensure_ascii=False)}") + logger.info( + f"Local user defined function as following:\ + \n{json.dumps(list(UDFS_YAML.keys()), indent=2, ensure_ascii=False)}" + ) # set task_type to `udf` - self.planner.current_task.task_type = 'udf' + self.planner.current_task.task_type = "udf" schema_path = UDFS_YAML else: schema_path = PROJECT_ROOT / "metagpt/tools/functions/schemas" @@ -108,26 +116,22 @@ class MLEngineer(CodeInterpreter): ) debug_context = tool_context cause_by = WriteCodeWithTools - - self.working_memory.add( - Message(content=code, role="assistant", cause_by=cause_by) - ) - + + self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) + result, success = await self.execute_code.run(code) print(result) # make tools for successful code and long code. - if success and self.make_udfs and len(remove_comments(code).split('\n')) > 4: - logger.info('Execute code successfully. Now start to make tools ...') + if success and self.make_udfs and len(remove_comments(code).split("\n")) > 4: + logger.info("Execute code successfully. Now start to make tools ...") await self.make_tools(code=code) - self.working_memory.add( - Message(content=result, role="user", cause_by=ExecutePyCode) - ) - + self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) + if "!pip" in code: success = False - + counter += 1 - + if not success and counter >= max_retry: logger.info("coding failed!") review, _ = await self.planner.ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) @@ -135,13 +139,15 @@ class MLEngineer(CodeInterpreter): counter = 0 # redo the task again with help of human suggestions if success: - if (self.use_tools and self.planner.current_task.task_type not in ['model_train', 'model_evaluate']) or self.use_udfs: + if ( + self.use_tools and self.planner.current_task.task_type not in ["model_train", "model_evaluate"] + ) or self.use_udfs: update_success, new_code = await self._update_data_columns() if update_success: code = code + "\n\n" + new_code return code, result, success - + async def _update_data_columns(self): logger.info("Check columns in updated data") rsp = await UpdateDataColumns().run(self.planner.plan) @@ -153,11 +159,11 @@ class MLEngineer(CodeInterpreter): print(result) self.data_desc["column_info"] = result return success, code - + async def _reflect(self): context = self.get_memories() context = "\n".join([str(msg) for msg in context]) - + reflection = await Reflect().run(context=context) self.working_memory.add(Message(content=reflection, role="assistant")) self.working_memory.add(Message(content=Reflect.REWRITE_PLAN_INSTRUCTION, role="user")) @@ -168,8 +174,10 @@ class MLEngineer(CodeInterpreter): Args: code (str): pure generation code by class WriteCodeByGenerate. """ - logger.warning(f"Making tools for task_id {self.planner.current_task_id}: \ - `{self.planner.current_task.instruction}` \n code: \n {code}") + logger.warning( + f"Making tools for task_id {self.planner.current_task_id}: \ + `{self.planner.current_task.instruction}` \n code: \n {code}" + ) make_tools = MakeTools() make_tool_retries, make_tool_current_retry = 3, 0 while True: @@ -185,9 +193,11 @@ class MLEngineer(CodeInterpreter): # end make tools if execute_success or make_tool_current_retry >= make_tool_retries: if make_tool_current_retry >= make_tool_retries: - logger.error(f"We have tried the maximum number of attempts {make_tool_retries}\ + logger.error( + f"We have tried the maximum number of attempts {make_tool_retries}\ and still have not created tools for task_id {self.planner.current_task_id} successfully,\ - we will skip it.") + we will skip it." + ) break # save successful tool code in udf if execute_success: diff --git a/metagpt/roles/ml_engineer_simple.py b/metagpt/roles/ml_engineer_simple.py index 7214e37c2..1006a4262 100644 --- a/metagpt/roles/ml_engineer_simple.py +++ b/metagpt/roles/ml_engineer_simple.py @@ -1,18 +1,17 @@ import re -from typing import List -import json from datetime import datetime +from typing import List import fire -from metagpt.roles import Role -from metagpt.schema import Message -from metagpt.memory import Memory -from metagpt.logs import logger -from metagpt.actions.write_analysis_code import WriteCodeByGenerate from metagpt.actions.ask_review import AskReview, ReviewConst from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.write_analysis_code import WriteCodeByGenerate +from metagpt.logs import logger +from metagpt.memory import Memory +from metagpt.roles import Role from metagpt.roles.kaggle_manager import DownloadData +from metagpt.schema import Message from metagpt.utils.save_code import save_code_file STRUCTURAL_CONTEXT_SIMPLE = """ @@ -40,9 +39,7 @@ Next Steps: class MLEngineerSimple(Role): - def __init__( - self, name="ABC", profile="MLEngineerSimple", goal="", auto_run: bool = False - ): + def __init__(self, name="ABC", profile="MLEngineerSimple", goal="", auto_run: bool = False): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="react") self._watch([DownloadData]) @@ -78,19 +75,13 @@ class MLEngineerSimple(Role): context = self.get_useful_memories() print(f"memories数量:{len(context)}") # print("===\n" +str(context) + "\n===") - code = await WriteCodeByGenerate().run( - context=context, temperature=0.0 - ) + code = await WriteCodeByGenerate().run(context=context, temperature=0.0) cause_by = WriteCodeByGenerate - self.working_memory.add( - Message(content=code, role="assistant", cause_by=cause_by) - ) + self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) result, success = await self.execute_code.run(code) print(result) - self.working_memory.add( - Message(content=result, role="user", cause_by=ExecutePyCode) - ) + self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) if "!pip" in code: success = False @@ -107,12 +98,10 @@ class MLEngineerSimple(Role): self._rc.memory.add(completed_plan_memory[0]) # add to persistent memory prompt = JUDGE_PROMPT_TEMPLATE.format(user_requirement=self.goal, context=completed_plan_memory) rsp = await self._llm.aask(prompt) - self.working_memory.add( - Message(content=rsp, role="system") - ) + self.working_memory.add(Message(content=rsp, role="system")) - matches = re.findall(r'\b(True|False)\b', rsp) - state = False if 'False' in matches else True + matches = re.findall(r"\b(True|False)\b", rsp) + state = False if "False" in matches else True async def _ask_review(self, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): auto_run = auto_run or self.auto_run @@ -127,9 +116,7 @@ class MLEngineerSimple(Role): def get_useful_memories(self) -> List[Message]: """find useful memories only to reduce context length and improve performance""" user_requirement = self.goal - context = STRUCTURAL_CONTEXT_SIMPLE.format( - user_requirement=user_requirement, data_desc=self.data_desc - ) + context = STRUCTURAL_CONTEXT_SIMPLE.format(user_requirement=user_requirement, data_desc=self.data_desc) context_msg = [Message(content=context, role="user")] return context_msg + self.get_working_memories(6) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index cb1d2eef3..0ea6d6ee6 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -35,10 +35,9 @@ from metagpt.const import SERDESER_PATH from metagpt.llm import LLM, HumanProvider from metagpt.logs import logger from metagpt.memory import Memory -from metagpt.provider.base_llm import BaseLLM -from metagpt.schema import Message, MessageQueue, SerializationMixin -from metagpt.schema import Task, TaskResult from metagpt.plan.planner import Planner +from metagpt.provider.base_llm import BaseLLM +from metagpt.schema import Message, MessageQueue, SerializationMixin, Task, TaskResult from metagpt.utils.common import ( any_to_name, any_to_str, @@ -270,7 +269,9 @@ class Role(SerializationMixin, is_polymorphic_base=True): if react_mode == RoleReactMode.REACT: self.rc.max_react_loop = max_react_loop elif react_mode == RoleReactMode.PLAN_AND_ACT: - self.planner = Planner(goal=self._setting.goal, working_memory=self.rc.working_memory, auto_run=auto_run, use_tools=use_tools) + self.planner = Planner( + goal=self._setting.goal, working_memory=self.rc.working_memory, auto_run=auto_run, use_tools=use_tools + ) def _watch(self, actions: Iterable[Type[Action]] | Iterable[Action]): """Watch Actions of interest. Role will select Messages caused by these Actions from its personal message @@ -450,35 +451,34 @@ class Role(SerializationMixin, is_polymorphic_base=True): async def _plan_and_act(self) -> Message: """first plan, then execute an action sequence, i.e. _think (of a plan) -> _act -> _act -> ... Use llm to come up with the plan dynamically.""" - + ### Common Procedure in both single- and multi-agent setting ### # create initial plan and update until confirmation await self.planner.update_plan() - - while self.planner.current_task: + while self.planner.current_task: task = self.planner.current_task logger.info(f"ready to take on task {task}") - + # take on current task task_result = await self._act_on_task(task) - + # ask for acceptance, users can other refuse and change tasks in the plan review, task_result_confirmed = await self.planner.ask_review(task_result) - + if task_result_confirmed: # tick off this task and record progress await self.planner.confirm_task(task, task_result, review) - + elif "redo" in review: # Ask the Role to redo this task with help of review feedback, # useful when the code run is successful but the procedure or result is not what we want continue - + else: # update plan according to user's feedback and to take on changed tasks await self.planner.update_plan(review) - + completed_plan_memory = self.planner.get_useful_memories() # completed plan as a outcome rsp = completed_plan_memory[0] @@ -486,7 +486,7 @@ class Role(SerializationMixin, is_polymorphic_base=True): self.rc.memory.add(rsp) # add to persistent memory return rsp - + async def _act_on_task(self, current_task: Task) -> TaskResult: """Taking specific action to handle one task in plan diff --git a/metagpt/schema.py b/metagpt/schema.py index 402b3e93f..31a83e5dd 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -308,12 +308,12 @@ class AIMessage(Message): """ def __init__(self, content: str): - super().__init__(content, 'assistant') + super().__init__(content, "assistant") class Task(BaseModel): task_id: str = "" - dependent_task_ids: list[str] = [] # Tasks prerequisite to this Task + dependent_task_ids: list[str] = [] # Tasks prerequisite to this Task instruction: str = "" task_type: str = "" code_steps: str = "" @@ -325,6 +325,7 @@ class Task(BaseModel): class TaskResult(BaseModel): """Result of taking a task, with result and is_success required to be filled""" + code_steps: str = "" code: str = "" result: str @@ -360,12 +361,12 @@ class Plan(BaseModel): def add_tasks(self, tasks: list[Task]): """ Integrates new tasks into the existing plan, ensuring dependency order is maintained. - + This method performs two primary functions based on the current state of the task list: - 1. If there are no existing tasks, it topologically sorts the provided tasks to ensure + 1. If there are no existing tasks, it topologically sorts the provided tasks to ensure correct execution order based on dependencies, and sets these as the current tasks. - 2. If there are existing tasks, it merges the new tasks with the existing ones. It maintains - any common prefix of tasks (based on task_id and instruction) and appends the remainder + 2. If there are existing tasks, it merges the new tasks with the existing ones. It maintains + any common prefix of tasks (based on task_id and instruction) and appends the remainder of the new tasks. The current task is updated to the first unfinished task in this merged list. Args: @@ -395,13 +396,13 @@ class Plan(BaseModel): # Combine the common prefix with the remainder of the new tasks final_tasks = self.tasks[:prefix_length] + new_tasks[prefix_length:] self.tasks = final_tasks - + # Update current_task_id to the first unfinished task in the merged list self._update_current_task() # Update the task map for quick access to tasks by ID self.task_map = {task.task_id: task for task in self.tasks} - + def reset_task(self, task_id: str): """ Clear code and result of the task based on task_id, and set the task as unfinished. @@ -448,20 +449,21 @@ class Plan(BaseModel): Args: new_task (Task): The new task to be appended to the existing task sequence - + Returns: None """ assert not self.has_task_id(new_task.task_id), "Task already in current plan, use replace_task instead" - assert all([self.has_task_id(dep_id) for dep_id in new_task.dependent_task_ids]), \ - "New task has unknown dependencies" + assert all( + [self.has_task_id(dep_id) for dep_id in new_task.dependent_task_ids] + ), "New task has unknown dependencies" # Existing tasks do not depend on the new task, it's fine to put it to the end of the sorted task sequence self.tasks.append(new_task) self.task_map[new_task.task_id] = new_task self._update_current_task() - + def update_task_result(self, task: Task, task_result: TaskResult): task.code_steps = task_result.code_steps task.code = task_result.code @@ -478,7 +480,7 @@ class Plan(BaseModel): current_task_id = task.task_id break self.current_task_id = current_task_id # all tasks finished - + @property def current_task(self) -> Task: """Find current task to execute @@ -489,8 +491,7 @@ class Plan(BaseModel): return self.task_map.get(self.current_task_id, None) def finish_current_task(self): - """Finish current task, set Task.is_finished=True, set current task to next task - """ + """Finish current task, set Task.is_finished=True, set current task to next task""" if self.current_task_id: self.current_task.is_finished = True self._update_current_task() # set to next task diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index f1665b405..5d1cd97d8 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -3,19 +3,26 @@ import json import numpy as np import pandas as pd from sklearn.impute import SimpleImputer -from sklearn.preprocessing import LabelEncoder -from sklearn.preprocessing import MaxAbsScaler -from sklearn.preprocessing import MinMaxScaler -from sklearn.preprocessing import OneHotEncoder -from sklearn.preprocessing import OrdinalEncoder -from sklearn.preprocessing import RobustScaler -from sklearn.preprocessing import StandardScaler +from sklearn.preprocessing import ( + LabelEncoder, + MaxAbsScaler, + MinMaxScaler, + OneHotEncoder, + OrdinalEncoder, + RobustScaler, + StandardScaler, +) from metagpt.tools.functions.libs.base import MLProcess class FillMissingValue(MLProcess): - def __init__(self, features: list, strategy: str = 'mean', fill_value=None,): + def __init__( + self, + features: list, + strategy: str = "mean", + fill_value=None, + ): self.features = features self.strategy = strategy self.fill_value = fill_value @@ -35,7 +42,10 @@ class FillMissingValue(MLProcess): class MinMaxScale(MLProcess): - def __init__(self, features: list,): + def __init__( + self, + features: list, + ): self.features = features self.mms = None @@ -49,7 +59,10 @@ class MinMaxScale(MLProcess): class StandardScale(MLProcess): - def __init__(self, features: list,): + def __init__( + self, + features: list, + ): self.features = features self.ss = None @@ -63,7 +76,10 @@ class StandardScale(MLProcess): class MaxAbsScale(MLProcess): - def __init__(self, features: list,): + def __init__( + self, + features: list, + ): self.features = features self.mas = None @@ -77,7 +93,10 @@ class MaxAbsScale(MLProcess): class RobustScale(MLProcess): - def __init__(self, features: list,): + def __init__( + self, + features: list, + ): self.features = features self.rs = None @@ -91,7 +110,10 @@ class RobustScale(MLProcess): class OrdinalEncode(MLProcess): - def __init__(self, features: list,): + def __init__( + self, + features: list, + ): self.features = features self.oe = None @@ -105,7 +127,10 @@ class OrdinalEncode(MLProcess): class OneHotEncode(MLProcess): - def __init__(self, features: list,): + def __init__( + self, + features: list, + ): self.features = features self.ohe = None @@ -123,7 +148,10 @@ class OneHotEncode(MLProcess): class LabelEncode(MLProcess): - def __init__(self, features: list,): + def __init__( + self, + features: list, + ): self.features = features self.le_encoders = [] @@ -131,7 +159,7 @@ class LabelEncode(MLProcess): if len(self.features) == 0: return for col in self.features: - le = LabelEncoder().fit(df[col].astype(str).unique().tolist() + ['unknown']) + le = LabelEncoder().fit(df[col].astype(str).unique().tolist() + ["unknown"]) self.le_encoders.append(le) def transform(self, df: pd.DataFrame): @@ -141,7 +169,7 @@ class LabelEncode(MLProcess): data_list = df[self.features[i]].astype(str).tolist() for unique_item in np.unique(df[self.features[i]].astype(str)): if unique_item not in self.le_encoders[i].classes_: - data_list = ['unknown' if x == unique_item else x for x in data_list] + data_list = ["unknown" if x == unique_item else x for x in data_list] df[self.features[i]] = self.le_encoders[i].transform(data_list) return df @@ -165,5 +193,5 @@ def get_column_info(df: pd.DataFrame) -> dict: column_info["Others"].append(col) if len(json.dumps(column_info)) > 2000: - column_info['Numeric'] = column_info['Numeric'][0:5] + ['Too many cols, omission here...'] + column_info["Numeric"] = column_info["Numeric"][0:5] + ["Too many cols, omission here..."] return column_info diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index df36752b9..534c5b8e4 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -13,7 +13,7 @@ from joblib import Parallel, delayed from pandas.core.dtypes.common import is_object_dtype from sklearn.feature_selection import VarianceThreshold from sklearn.model_selection import KFold -from sklearn.preprocessing import PolynomialFeatures, KBinsDiscretizer +from sklearn.preprocessing import KBinsDiscretizer, PolynomialFeatures from metagpt.tools.functions.libs.base import MLProcess @@ -91,9 +91,7 @@ class KFoldTargetMeanEncoder(MLProcess): col_name = f"{self.col}_kf_target_mean" for trn_idx, val_idx in kf.split(tmp, tmp[self.label]): _trn, _val = tmp.iloc[trn_idx], tmp.iloc[val_idx] - tmp.loc[tmp.index[val_idx], col_name] = _val[self.col].map( - _trn.groupby(self.col)[self.label].mean() - ) + tmp.loc[tmp.index[val_idx], col_name] = _val[self.col].map(_trn.groupby(self.col)[self.label].mean()) tmp[col_name].fillna(global_mean, inplace=True) self.encoder_dict = tmp.groupby(self.col)[col_name].mean().to_dict() @@ -111,7 +109,7 @@ class CatCross(MLProcess): @staticmethod def cross_two(comb, df): - new_col = f'{comb[0]}_{comb[1]}' + new_col = f"{comb[0]}_{comb[1]}" new_col_combs = list(itertools.product(df[comb[0]].unique(), df[comb[1]].unique())) ll = list(range(len(new_col_combs))) comb_map = dict(zip(new_col_combs, ll)) @@ -122,13 +120,12 @@ class CatCross(MLProcess): if df[col].nunique() > self.max_cat_num: self.cols.remove(col) self.combs = list(itertools.combinations(self.cols, 2)) - res = Parallel(n_jobs=4, require='sharedmem')( - delayed(self.cross_two)(comb, df) for comb in self.combs) + res = Parallel(n_jobs=4, require="sharedmem")(delayed(self.cross_two)(comb, df) for comb in self.combs) self.combs_map = dict(res) def transform(self, df: pd.DataFrame) -> pd.DataFrame: for comb in self.combs: - new_col = f'{comb[0]}_{comb[1]}' + new_col = f"{comb[0]}_{comb[1]}" _map = self.combs_map[new_col] df[new_col] = pd.Series(zip(df[comb[0]], df[comb[1]])).map(_map) # set the unknown value to a new number @@ -157,13 +154,13 @@ class GroupStat(MLProcess): class SplitBins(MLProcess): - def __init__(self, cols: str, strategy: str = 'quantile'): + def __init__(self, cols: str, strategy: str = "quantile"): self.cols = cols self.strategy = strategy self.encoder = None def fit(self, df: pd.DataFrame): - self.encoder = KBinsDiscretizer(strategy=self.strategy, encode='ordinal') + self.encoder = KBinsDiscretizer(strategy=self.strategy, encode="ordinal") self.encoder.fit(df[self.cols].fillna(0)) def transform(self, df: pd.DataFrame) -> pd.DataFrame: @@ -296,10 +293,7 @@ class GeneralSelection(MLProcess): if df[col].nunique() == 1: feats.remove(col) - if ( - df.loc[df[col] == np.inf].shape[0] != 0 - or df.loc[df[col] == np.inf].shape[0] != 0 - ): + if df.loc[df[col] == np.inf].shape[0] != 0 or df.loc[df[col] == np.inf].shape[0] != 0: feats.remove(col) if is_object_dtype(df[col]) and df[col].nunique() == df.shape[0]: @@ -320,10 +314,10 @@ class TreeBasedSelection(MLProcess): def fit(self, df: pd.DataFrame): params = { - 'boosting_type': 'gbdt', - 'objective': 'binary', - 'learning_rate': 0.1, - 'num_leaves': 31, + "boosting_type": "gbdt", + "objective": "binary", + "learning_rate": 0.1, + "num_leaves": 31, } if self.task_type == "cls": @@ -342,12 +336,11 @@ class TreeBasedSelection(MLProcess): dtrain = lgb.Dataset(df[cols], df[self.label_col]) model = lgb.train(params, dtrain, num_boost_round=100) - df_imp = pd.DataFrame({'feature_name': dtrain.feature_name, - 'importance': model.feature_importance("gain")}) + df_imp = pd.DataFrame({"feature_name": dtrain.feature_name, "importance": model.feature_importance("gain")}) df_imp.sort_values("importance", ascending=False, inplace=True) df_imp = df_imp[df_imp["importance"] > 0] - self.feats = df_imp['feature_name'].tolist() + self.feats = df_imp["feature_name"].tolist() self.feats.append(self.label_col) def transform(self, df: pd.DataFrame) -> pd.DataFrame: diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py index 5d9c35b27..6644565d7 100644 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ b/metagpt/tools/functions/libs/udf/__init__.py @@ -5,12 +5,12 @@ import yaml import inspect import importlib from pathlib import Path -from typing import Dict, List +from typing import List from metagpt.logs import logger def extract_function_signatures(file_path): - with open(file_path, 'r', encoding='utf-8') as file: + with open(file_path, "r", encoding="utf-8") as file: source_code = file.read() tree = ast.parse(source_code) @@ -19,7 +19,7 @@ def extract_function_signatures(file_path): for node in ast.walk(tree): if isinstance(node, ast.FunctionDef): # 只提取用户自定义函数,排除内置函数 - if not (node.name.startswith('__') and node.name.endswith('__')): + if not (node.name.startswith("__") and node.name.endswith("__")): # 获取函数名 function_name = node.name # 获取参数列表 @@ -27,36 +27,37 @@ def extract_function_signatures(file_path): # 获取函数签名 function_signature = f"{function_name}({', '.join(args)})" # 导入函数 - module_name = Path(file_path).parts[-1][:-len(Path(file_path).suffix)] + module_name = Path(file_path).parts[-1][: -len(Path(file_path).suffix)] module = importlib.import_module(f"metagpt.tools.functions.libs.udf.{module_name}") # 将函数导入到当前命名空间 globals().update({function_name: getattr(module, function_name)}) # 获取函数注释和函数路径 - function_schema = {'udf_name': function_signature, - 'udf_path': f'from metagpt.tools.functions.libs.udf.{module_name} import {function_name}', - 'udf_doc': inspect.getdoc(getattr(module, function_name))} + function_schema = { + "udf_name": function_signature, + "udf_path": f"from metagpt.tools.functions.libs.udf.{module_name} import {function_name}", + "udf_doc": inspect.getdoc(getattr(module, function_name)), + } function_signatures.append(function_schema) # 获取函数返回变量名 source_lines, _ = inspect.getsourcelines(getattr(module, function_name)) for line in source_lines: if line.strip().startswith("return "): - function_returns.append({ - 'udf_name': function_name, - 'udf_returns': [var.strip() for var in line.strip()[len("return "):].split(',')] - }) + function_returns.append( + { + "udf_name": function_name, + "udf_returns": [var.strip() for var in line.strip()[len("return ") :].split(",")], + } + ) break # 没有返回值的函数 - if not function_returns or function_returns[-1]['udf_name'] != function_name: - function_returns.append({ - 'udf_name': function_name, - 'udf_returns': [None] - }) + if not function_returns or function_returns[-1]["udf_name"] != function_name: + function_returns.append({"udf_name": function_name, "udf_returns": [None]}) return function_signatures, function_returns def get_function_signatures_in_folder(folder_path): - python_files = [f for f in os.listdir(folder_path) if f.endswith('.py') and f != '__init__.py'] + python_files = [f for f in os.listdir(folder_path) if f.endswith(".py") and f != "__init__.py"] all_function_signatures = [] all_function_returns = [] @@ -74,31 +75,33 @@ def docstring_to_yaml(docstring: str, return_vars: List[str] = None): if docstring is None: return {} # 匹配简介部分 - description_match = re.search(r'^(.*?)(?:Args:|Returns:|Raises:|$)', docstring, re.DOTALL) + description_match = re.search(r"^(.*?)(?:Args:|Returns:|Raises:|$)", docstring, re.DOTALL) description = description_match.group(1).strip() if description_match else "" # 匹配Args部分 - args_match = re.search(r'Args:\s*(.*?)(?:Returns:|Raises:|$)', docstring, re.DOTALL) + args_match = re.search(r"Args:\s*(.*?)(?:Returns:|Raises:|$)", docstring, re.DOTALL) _args = args_match.group(1).strip() if args_match else "" - variable_pattern = re.compile(r'(\w+)\s*\((.*?)\):\s*(.*)') + variable_pattern = re.compile(r"(\w+)\s*\((.*?)\):\s*(.*)") params = variable_pattern.findall(_args) if not params: params = ((None, None, None),) # 匹配Returns部分 - returns_match = re.search(r'Returns:\s*(.*?)(?:Raises:|$)', docstring, re.DOTALL) + returns_match = re.search(r"Returns:\s*(.*?)(?:Raises:|$)", docstring, re.DOTALL) returns = returns_match.group(1).strip() if returns_match else "" - return_pattern = re.compile(r'^(.*)\s*:\s*(.*)$') + return_pattern = re.compile(r"^(.*)\s*:\s*(.*)$") # 添加返回值变量名 return_vars = return_vars if isinstance(return_vars, list) else [return_vars] returns = [(r, *r_desc) for r_desc, r in zip(return_pattern.findall(returns), return_vars)] # 构建YAML字典 yaml_data = { - 'description': description.strip('.').strip(), - 'parameters': { - 'properties': {param[0]: {'type': param[1], 'description': param[2]} for param in params if param[0] is not None}, - 'required': [param[0] for param in params if param[0] is not None] + "description": description.strip(".").strip(), + "parameters": { + "properties": { + param[0]: {"type": param[1], "description": param[2]} for param in params if param[0] is not None + }, + "required": [param[0] for param in params if param[0] is not None], }, - 'returns': {ret[0]: {'type': ret[1], 'description': ret[2]} for ret in returns} + "returns": {ret[0]: {"type": ret[1], "description": ret[2]} for ret in returns}, } return yaml_data @@ -107,10 +110,10 @@ def extract_function_schema_yaml_in_folder(folder_path: str): function_signatures, function_returns = get_function_signatures_in_folder(folder_path) function_schema_yaml_data = {} for func_docstring, func_returns in zip(function_signatures, function_returns): - if func_docstring['udf_doc']: - fun_yaml_data = docstring_to_yaml(func_docstring['udf_doc'], func_returns['udf_returns']) - fun_yaml_data.update({'type': 'function'}) - function_schema_yaml_data.update({func_returns['udf_name']: fun_yaml_data}) + if func_docstring["udf_doc"]: + fun_yaml_data = docstring_to_yaml(func_docstring["udf_doc"], func_returns["udf_returns"]) + fun_yaml_data.update({"type": "function"}) + function_schema_yaml_data.update({func_returns["udf_name"]: fun_yaml_data}) return yaml.dump(function_schema_yaml_data, default_flow_style=False) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index bf112f820..b20b4acd2 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -361,6 +361,7 @@ def create_func_config(func_schema: dict) -> dict: def remove_comments(code_str): """Remove comments from code.""" pattern = r"(\".*?\"|\'.*?\')|(\#.*?$)" + def replace_func(match): if match.group(2) is not None: return "" diff --git a/metagpt/utils/recovery_util.py b/metagpt/utils/recovery_util.py index cef302d6b..3405b9587 100644 --- a/metagpt/utils/recovery_util.py +++ b/metagpt/utils/recovery_util.py @@ -2,15 +2,17 @@ # @Date : 12/20/2023 11:07 AM # @Author : stellahong (stellahong@fuzhi.ai) # @Desc : -import nbformat -from pathlib import Path import json from datetime import datetime +from pathlib import Path + +import nbformat -from metagpt.roles.role import Role from metagpt.const import DATA_PATH +from metagpt.roles.role import Role from metagpt.utils.save_code import save_code_file + def load_history(save_dir: str = ""): """ Load history from the specified save directory. @@ -21,7 +23,7 @@ def load_history(save_dir: str = ""): Returns: Tuple: A tuple containing the loaded plan and notebook. """ - + plan_path = Path(save_dir) / "plan.json" nb_path = Path(save_dir) / "history_nb" / "code.ipynb" plan = json.load(open(plan_path, "r", encoding="utf-8")) @@ -40,16 +42,16 @@ def save_history(role: Role, save_dir: str = ""): Returns: Path: The path to the saved history directory. """ - record_time = datetime.now().strftime('%Y-%m-%d_%H-%M-%S') + record_time = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") save_path = DATA_PATH / "output" / f"{record_time}" - + # overwrite exist trajectory save_path.mkdir(parents=True, exist_ok=True) - + plan = role.planner.plan.dict() - + with open(save_path / "plan.json", "w", encoding="utf-8") as plan_file: json.dump(plan, plan_file, indent=4, ensure_ascii=False) - + save_code_file(name=Path(record_time) / "history_nb", code_context=role.execute_code.nb, file_format="ipynb") - return save_path \ No newline at end of file + return save_path diff --git a/metagpt/utils/save_code.py b/metagpt/utils/save_code.py index 96c310336..adf136316 100644 --- a/metagpt/utils/save_code.py +++ b/metagpt/utils/save_code.py @@ -2,13 +2,14 @@ # @Date : 12/12/2023 4:14 PM # @Author : stellahong (stellahong@fuzhi.ai) # @Desc : -import os import json +import os import nbformat from metagpt.const import DATA_PATH + def save_code_file(name: str, code_context: str, file_format: str = "py") -> None: """ Save code files to a specified path. @@ -36,10 +37,6 @@ def save_code_file(name: str, code_context: str, file_format: str = "py") -> Non with open(file_path, "w", encoding="utf-8") as fp: json.dump(data, fp, indent=2) elif file_format == "ipynb": - nbformat.write(code_context, file_path) + nbformat.write(code_context, file_path) else: raise ValueError("Unsupported file format. Please choose 'py', 'json', or 'ipynb'.") - - - - diff --git a/tests/metagpt/test_schema.py b/tests/metagpt/test_schema.py index f4dc56bdd..ab2e206a4 100644 --- a/tests/metagpt/test_schema.py +++ b/tests/metagpt/test_schema.py @@ -26,10 +26,11 @@ from metagpt.schema import ( Document, Message, MessageQueue, + Plan, SystemMessage, + Task, UserMessage, ) -from metagpt.schema import Task, Plan from metagpt.utils.common import any_to_str @@ -53,7 +54,7 @@ class TestPlan: tasks = [ Task(task_id="1", dependent_task_ids=["2", "3"], instruction="Third"), Task(task_id="2", instruction="First"), - Task(task_id="3", dependent_task_ids=["2"], instruction="Second") + Task(task_id="3", dependent_task_ids=["2"], instruction="Second"), ] # 2 -> 3 -> 1 plan.add_tasks(tasks) @@ -65,7 +66,7 @@ class TestPlan: tasks = [ Task(task_id="1", dependent_task_ids=["2", "3"], instruction="Third"), Task(task_id="2", instruction="First"), - Task(task_id="3", dependent_task_ids=["2"], instruction="Second", is_finished=True) + Task(task_id="3", dependent_task_ids=["2"], instruction="Second", is_finished=True), ] # 2 -> 3 -> 1 plan.add_tasks(tasks) @@ -81,7 +82,7 @@ class TestPlan: tasks = [ Task(task_id="1", dependent_task_ids=["2", "3"], instruction="Third"), Task(task_id="2", instruction="First"), - Task(task_id="3", dependent_task_ids=["2"], instruction="Second") + Task(task_id="3", dependent_task_ids=["2"], instruction="Second"), ] # 2 -> 3 -> 1 plan.add_tasks(tasks) plan.finish_current_task() # finish 2 @@ -90,19 +91,21 @@ class TestPlan: new_tasks = [ Task(task_id="4", dependent_task_ids=["3"], instruction="Third"), Task(task_id="2", instruction="First"), - Task(task_id="3", dependent_task_ids=["2"], instruction="Second") + Task(task_id="3", dependent_task_ids=["2"], instruction="Second"), ] # 2 -> 3 -> 4, so the common prefix is 2 -> 3, and these two should be obtained from the existing tasks plan.add_tasks(new_tasks) assert [task.task_id for task in plan.tasks] == ["2", "3", "4"] - assert plan.tasks[0].is_finished and plan.tasks[1].is_finished # "2" and "3" should be the original finished one + assert ( + plan.tasks[0].is_finished and plan.tasks[1].is_finished + ) # "2" and "3" should be the original finished one assert plan.current_task_id == "4" def test_current_task(self): plan = Plan(goal="") tasks = [ Task(task_id="1", dependent_task_ids=["2"], instruction="Second"), - Task(task_id="2", instruction="First") + Task(task_id="2", instruction="First"), ] plan.add_tasks(tasks) assert plan.current_task.task_id == "2" @@ -111,7 +114,7 @@ class TestPlan: plan = Plan(goal="") tasks = [ Task(task_id="1", instruction="First"), - Task(task_id="2", dependent_task_ids=["1"], instruction="Second") + Task(task_id="2", dependent_task_ids=["1"], instruction="Second"), ] plan.add_tasks(tasks) plan.finish_current_task() @@ -121,7 +124,7 @@ class TestPlan: plan = Plan(goal="") tasks = [ Task(task_id="1", instruction="First"), - Task(task_id="2", dependent_task_ids=["1"], instruction="Second") + Task(task_id="2", dependent_task_ids=["1"], instruction="Second"), ] plan.add_tasks(tasks) plan.finish_current_task() @@ -149,8 +152,10 @@ class TestPlan: def test_replace_task_with_dependents(self): plan = Plan(goal="") - tasks = [Task(task_id="1", instruction="First Task", finished=True), - Task(task_id="2", instruction="Second Task", dependent_task_ids=["1"], finished=True)] + tasks = [ + Task(task_id="1", instruction="First Task", finished=True), + Task(task_id="2", instruction="Second Task", dependent_task_ids=["1"], finished=True), + ] plan.add_tasks(tasks) new_task = Task(task_id="1", instruction="Updated First Task") plan.replace_task(new_task) @@ -168,7 +173,7 @@ class TestPlan: plan.replace_task(new_task) # Task with ID 2 does not exist in plan assert "1" in plan.task_map assert "2" not in plan.task_map - + def test_append_task_with_valid_dependencies(self): plan = Plan(goal="Test") existing_task = [Task(task_id="1")] @@ -183,7 +188,7 @@ class TestPlan: plan = Plan(goal="Test") with pytest.raises(AssertionError): plan.append_task(new_task) - + def test_append_task_without_dependencies(self): plan = Plan(goal="Test") existing_task = [Task(task_id="1")] From 4ec615169162daa545947c84c2dccc30402ddd34 Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 10 Jan 2024 14:16:04 +0800 Subject: [PATCH 279/637] format using precommit --- tests/metagpt/actions/test_make_tools.py | 14 ++--- .../actions/test_write_analysis_code.py | 54 ++++++++++--------- tests/metagpt/actions/test_write_plan.py | 9 ++-- tests/metagpt/roles/run_code_interpreter.py | 42 +++++++++------ tests/metagpt/roles/test_daml.py | 16 +++--- tests/metagpt/tools/functions/test_udf.py | 36 ++++++------- tests/metagpt/utils/test_save_code.py | 15 +++--- 7 files changed, 102 insertions(+), 84 deletions(-) diff --git a/tests/metagpt/actions/test_make_tools.py b/tests/metagpt/actions/test_make_tools.py index cf7986b82..8e94c6eee 100644 --- a/tests/metagpt/actions/test_make_tools.py +++ b/tests/metagpt/actions/test_make_tools.py @@ -8,7 +8,7 @@ from metagpt.logs import logger @pytest.mark.asyncio async def test_make_tools(): code = "import yfinance as yf\n\n# Collect Alibaba stock data\nalibaba = yf.Ticker('BABA')\ndata = alibaba.history(period='1d', start='2022-01-01', end='2022-12-31')\nprint(data.head())" - msgs = [{'role': 'assistant', 'content': code}] + msgs = [{"role": "assistant", "content": code}] mt = MakeTools() tool_code = await mt.run(msgs) logger.debug(tool_code) @@ -21,10 +21,10 @@ async def test_make_tools(): @pytest.mark.asyncio async def test_make_tools2(): - code = '''import pandas as pd\npath = "./tests/data/test.csv"\ndf = pd.read_csv(path)\ndata = df.copy()\n + code = """import pandas as pd\npath = "./tests/data/test.csv"\ndf = pd.read_csv(path)\ndata = df.copy()\n data['started_at'] = data['started_at'].apply(lambda r: pd.to_datetime(r))\n - data['ended_at'] = data['ended_at'].apply(lambda r: pd.to_datetime(r))\ndata.head()''' - msgs = [{'role': 'assistant', 'content': code}] + data['ended_at'] = data['ended_at'].apply(lambda r: pd.to_datetime(r))\ndata.head()""" + msgs = [{"role": "assistant", "content": code}] mt = MakeTools() tool_code = await mt.run(msgs) logger.debug(tool_code) @@ -37,11 +37,11 @@ async def test_make_tools2(): @pytest.mark.asyncio async def test_make_tools3(): - code = '''import pandas as pd\npath = "./tests/data/test.csv"\ndf = pd.read_csv(path)\ndata = df.copy()\n + code = """import pandas as pd\npath = "./tests/data/test.csv"\ndf = pd.read_csv(path)\ndata = df.copy()\n data['started_at'] = data['started_at'].apply(lambda r: pd.to_datetime(r))\n data['ended_at'] = data['ended_at'].apply(lambda r: pd.to_datetime(r))\n - data['duration_hour'] = (data['ended_at'] - data['started_at']).dt.seconds/3600\ndata.head()''' - msgs = [{'role': 'assistant', 'content': code}] + data['duration_hour'] = (data['ended_at'] - data['started_at']).dt.seconds/3600\ndata.head()""" + msgs = [{"role": "assistant", "content": code}] mt = MakeTools() tool_code = await mt.run(msgs) logger.debug(tool_code) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index 1a568cdcd..df1d39603 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -1,10 +1,11 @@ import asyncio + import pytest -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.actions.execute_code import ExecutePyCode -from metagpt.schema import Message, Plan, Task +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.logs import logger +from metagpt.schema import Message, Plan, Task @pytest.mark.asyncio @@ -15,9 +16,9 @@ async def test_write_code_by_list_plan(): plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] for task in plan: print(f"\n任务: {task}\n\n") - messages.append(Message(task, role='assistant')) + messages.append(Message(task, role="assistant")) code = await write_code.run(messages) - messages.append(Message(code, role='assistant')) + messages.append(Message(code, role="assistant")) assert len(code) > 0 output = await execute_code.run(code) print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") @@ -48,11 +49,11 @@ async def test_write_code_with_tools(): messages = [] task_map = { "1": Task( - task_id="1", - instruction="随机生成一个pandas DataFrame数据集", - task_type="other", - dependent_task_ids=[], - code=""" + task_id="1", + instruction="随机生成一个pandas DataFrame数据集", + task_type="other", + dependent_task_ids=[], + code=""" import pandas as pd df = pd.DataFrame({ 'a': [1, 2, 3, 4, 5], @@ -61,18 +62,18 @@ async def test_write_code_with_tools(): 'd': [1, 2, 3, 4, 5] }) """, - is_finished=True, - ), + is_finished=True, + ), "2": Task( - task_id="2", - instruction="对数据集进行数据清洗", - task_type="data_preprocess", - dependent_task_ids=["1"], - code_steps=""" + task_id="2", + instruction="对数据集进行数据清洗", + task_type="data_preprocess", + dependent_task_ids=["1"], + code_steps=""" {"Step 1": "对数据集进行去重", "Step 2": "对数据集进行缺失值处理"} - """ - ), + """, + ), } plan = Plan( goal="构造数据集并进行数据清洗", @@ -89,7 +90,6 @@ async def test_write_code_with_tools(): @pytest.mark.asyncio async def test_write_code_to_correct_error(): - structural_context = """ ## User Requirement read a dataset test.csv and print its head @@ -136,7 +136,8 @@ async def test_write_code_to_correct_error(): ] new_code = await WriteCodeByGenerate().run(context=context) print(new_code) - assert "read_csv" in new_code # should correct read_excel to read_csv + assert "read_csv" in new_code # should correct read_excel to read_csv + @pytest.mark.asyncio async def test_write_code_reuse_code_simple(): @@ -174,7 +175,8 @@ async def test_write_code_reuse_code_simple(): ] code = await WriteCodeByGenerate().run(context=context) print(code) - assert "pandas" not in code and "read_csv" not in code # should reuse import and read statement from previous one + assert "pandas" not in code and "read_csv" not in code # should reuse import and read statement from previous one + @pytest.mark.asyncio async def test_write_code_reuse_code_long(): @@ -227,8 +229,9 @@ async def test_write_code_reuse_code_long(): trials = [WriteCodeByGenerate().run(context=context, temperature=0.0) for _ in range(trials_num)] trial_results = await asyncio.gather(*trials) print(*trial_results, sep="\n\n***\n\n") - success = ["load_iris" not in result and "iris_data" in result \ - for result in trial_results] # should reuse iris_data from previous tasks + success = [ + "load_iris" not in result and "iris_data" in result for result in trial_results + ] # should reuse iris_data from previous tasks success_rate = sum(success) / trials_num logger.info(f"success rate: {success_rate :.2f}") assert success_rate >= 0.8 @@ -299,8 +302,9 @@ async def test_write_code_reuse_code_long_for_wine(): trials = [WriteCodeByGenerate().run(context=context, temperature=0.0) for _ in range(trials_num)] trial_results = await asyncio.gather(*trials) print(*trial_results, sep="\n\n***\n\n") - success = ["load_wine" not in result and "wine_data" in result\ - for result in trial_results] # should reuse iris_data from previous tasks + success = [ + "load_wine" not in result and "wine_data" in result for result in trial_results + ] # should reuse iris_data from previous tasks success_rate = sum(success) / trials_num logger.info(f"success rate: {success_rate :.2f}") assert success_rate >= 0.8 diff --git a/tests/metagpt/actions/test_write_plan.py b/tests/metagpt/actions/test_write_plan.py index 7766e0d51..6f2e7d430 100644 --- a/tests/metagpt/actions/test_write_plan.py +++ b/tests/metagpt/actions/test_write_plan.py @@ -1,6 +1,9 @@ -import pytest +from metagpt.actions.write_plan import ( + Plan, + Task, + precheck_update_plan_from_rsp, +) -from metagpt.actions.write_plan import WritePlan, precheck_update_plan_from_rsp, Plan, Task def test_precheck_update_plan_from_rsp(): plan = Plan(goal="") @@ -10,6 +13,6 @@ def test_precheck_update_plan_from_rsp(): assert success assert len(plan.tasks) == 1 and plan.tasks[0].task_id == "1" # precheck should not change the original one - invalid_rsp = 'wrong' + invalid_rsp = "wrong" success, _ = precheck_update_plan_from_rsp(invalid_rsp, plan) assert not success diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py index 51506e7e5..418270e25 100644 --- a/tests/metagpt/roles/run_code_interpreter.py +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -1,15 +1,16 @@ import fire from metagpt.actions.execute_code import ExecutePyCode -from metagpt.const import DATA_PATH from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter from metagpt.roles.ml_engineer import MLEngineer from metagpt.schema import Plan -from metagpt.utils.recovery_util import save_history, load_history +from metagpt.utils.recovery_util import load_history, save_history -async def run_code_interpreter(role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir): +async def run_code_interpreter( + role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir +): """ The main function to run the MLEngineer with optional history loading. @@ -26,26 +27,28 @@ async def run_code_interpreter(role_class, requirement, auto_run, use_tools, use role = CodeInterpreter(goal=requirement, auto_run=auto_run, use_tools=use_tools) else: role = MLEngineer( - goal=requirement, auto_run=auto_run, use_tools=use_tools, use_code_steps=use_code_steps, - make_udfs=make_udfs, use_udfs=use_udfs + goal=requirement, + auto_run=auto_run, + use_tools=use_tools, + use_code_steps=use_code_steps, + make_udfs=make_udfs, + use_udfs=use_udfs, ) - + if save_dir: logger.info("Resuming from history trajectory") plan, nb = load_history(save_dir) role.planner.plan = Plan(**plan) role.execute_code = ExecutePyCode(nb) - + else: logger.info("Run from scratch") - - + try: await role.run(requirement) except Exception as e: - save_path = save_history(role, save_dir) - + logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") @@ -60,7 +63,7 @@ if __name__ == "__main__": # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report AUC Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - + save_dir = "" # role_class = "ci" @@ -71,10 +74,17 @@ if __name__ == "__main__": use_udfs = False async def main( - role_class: str = role_class, requirement: str = requirement, auto_run: bool = auto_run, - use_tools: bool = use_tools, use_code_steps: bool = False, make_udfs: bool = make_udfs, use_udfs: bool = use_udfs, - save_dir: str = save_dir + role_class: str = role_class, + requirement: str = requirement, + auto_run: bool = auto_run, + use_tools: bool = use_tools, + use_code_steps: bool = False, + make_udfs: bool = make_udfs, + use_udfs: bool = use_udfs, + save_dir: str = save_dir, ): - await run_code_interpreter(role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir) + await run_code_interpreter( + role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir + ) fire.Fire(main) diff --git a/tests/metagpt/roles/test_daml.py b/tests/metagpt/roles/test_daml.py index dbb4fb38f..2e2c003d9 100644 --- a/tests/metagpt/roles/test_daml.py +++ b/tests/metagpt/roles/test_daml.py @@ -2,8 +2,9 @@ import pytest from tqdm import tqdm from metagpt.logs import logger +from metagpt.roles.ml_engineer import ExecutePyCode, MLEngineer from metagpt.schema import Plan -from metagpt.roles.ml_engineer import MLEngineer, ExecutePyCode + def reset(role): """Restart role with the same goal.""" @@ -11,6 +12,7 @@ def reset(role): role.planner.plan = Plan(goal=role.planner.plan.goal) role.execute_code = ExecutePyCode() + async def make_use_tools(requirement: str, auto_run: bool = True): """make and use tools for requirement.""" role = MLEngineer(goal=requirement, auto_run=auto_run) @@ -31,11 +33,13 @@ async def make_use_tools(requirement: str, auto_run: bool = True): @pytest.mark.asyncio async def test_make_use_tools(): - requirements = ["Run data analysis on sklearn Iris dataset, include a plot", - "Run data analysis on sklearn Diabetes dataset, include a plot", - "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy", - "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy", - "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: tests/data/titanic.csv"] + requirements = [ + "Run data analysis on sklearn Iris dataset, include a plot", + "Run data analysis on sklearn Diabetes dataset, include a plot", + "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy", + "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy", + "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: tests/data/titanic.csv", + ] success = 0 for requirement in tqdm(requirements, total=len(requirements)): try: diff --git a/tests/metagpt/tools/functions/test_udf.py b/tests/metagpt/tools/functions/test_udf.py index b4060ad13..741bd9a9f 100644 --- a/tests/metagpt/tools/functions/test_udf.py +++ b/tests/metagpt/tools/functions/test_udf.py @@ -1,15 +1,15 @@ -import pytest -import yaml import json -from metagpt.tools.functions.libs.udf import UDFS, docstring_to_yaml, UDFS_YAML +import yaml + from metagpt.logs import logger +from metagpt.tools.functions.libs.udf import UDFS, UDFS_YAML, docstring_to_yaml def test_udfs(): assert len(UDFS) > 0 - assert 'udf_name' in UDFS[0] - assert 'udf_doc' in UDFS[0] + assert "udf_name" in UDFS[0] + assert "udf_doc" in UDFS[0] logger.info(UDFS) @@ -23,27 +23,27 @@ def test_docstring2yaml(): pd.DataFrame: The dataframe with an additional column 'duration_hour' added. """ - yaml_result = docstring_to_yaml(docstring, return_vars='dataframe') - assert 'parameters' in yaml_result - assert 'properties' in yaml_result['parameters'] - assert 'dataframe' in yaml_result['parameters']['properties'] + yaml_result = docstring_to_yaml(docstring, return_vars="dataframe") + assert "parameters" in yaml_result + assert "properties" in yaml_result["parameters"] + assert "dataframe" in yaml_result["parameters"]["properties"] def test_UDFS_YAML(): assert len(UDFS_YAML) > 0 logger.info(f"\n\n{json.dumps(UDFS_YAML, indent=2, ensure_ascii=False)}") function_schema = UDFS_YAML - assert 'description' in function_schema[list(function_schema.keys())[0]] - assert 'type' in function_schema[list(function_schema.keys())[0]] - assert 'parameters' in function_schema[list(function_schema.keys())[0]] - assert 'properties' in function_schema[list(function_schema.keys())[0]]['parameters'] - assert 'required' in function_schema[list(function_schema.keys())[0]]['parameters'] - assert 'returns' in function_schema[list(function_schema.keys())[0]] + assert "description" in function_schema[list(function_schema.keys())[0]] + assert "type" in function_schema[list(function_schema.keys())[0]] + assert "parameters" in function_schema[list(function_schema.keys())[0]] + assert "properties" in function_schema[list(function_schema.keys())[0]]["parameters"] + assert "required" in function_schema[list(function_schema.keys())[0]]["parameters"] + assert "returns" in function_schema[list(function_schema.keys())[0]] # 指定要保存的文件路径 - file_path = './tests/data/function_schema.yaml' + file_path = "./tests/data/function_schema.yaml" # 使用 PyYAML 将字典保存为 YAML 文件 - with open(file_path, 'w') as file: + with open(file_path, "w") as file: yaml.dump(function_schema, file, default_flow_style=False) - print(f'Data has been saved to {file_path}') + print(f"Data has been saved to {file_path}") diff --git a/tests/metagpt/utils/test_save_code.py b/tests/metagpt/utils/test_save_code.py index 60a9e1ff4..278d9a539 100644 --- a/tests/metagpt/utils/test_save_code.py +++ b/tests/metagpt/utils/test_save_code.py @@ -2,15 +2,15 @@ # @Date : 12/12/2023 4:17 PM # @Author : stellahong (stellahong@fuzhi.ai) # @Desc : -import pytest -import os import json +import os + import nbformat +import pytest -from metagpt.actions.write_analysis_code import WriteCodeByGenerate from metagpt.actions.execute_code import ExecutePyCode - -from metagpt.utils.save_code import save_code_file, DATA_PATH +from metagpt.actions.write_analysis_code import WriteCodeByGenerate +from metagpt.utils.save_code import DATA_PATH, save_code_file def test_save_code_file_python(): @@ -36,12 +36,9 @@ def test_save_code_file_json(): assert data["code"] == "print('Hello, JSON!')", "JSON content does not match" - @pytest.mark.asyncio async def test_save_code_file_notebook(): - code = await WriteCodeByGenerate().run( - context="basic python, hello world", plan="", code_steps="", temperature=0.0 - ) + code = await WriteCodeByGenerate().run(context="basic python, hello world", plan="", code_steps="", temperature=0.0) executor = ExecutePyCode() await executor.run(code) # Save as a Notebook file From cd990fd5c9e8b59251f78e5f3a1e2aea09589ec7 Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 10 Jan 2024 17:20:01 +0800 Subject: [PATCH 280/637] code adapted to v0.6 --- metagpt/actions/ask_review.py | 2 +- metagpt/actions/debug_code.py | 7 +---- metagpt/actions/execute_code.py | 30 +++++++++++---------- metagpt/actions/ml_da_action.py | 9 +++---- metagpt/actions/write_analysis_code.py | 19 +++++++------ metagpt/actions/write_plan.py | 2 +- metagpt/plan/planner.py | 19 ++++++++----- metagpt/roles/code_interpreter.py | 7 +++-- metagpt/roles/kaggle_manager.py | 21 +++++++-------- metagpt/roles/ml_engineer.py | 20 +++++++++----- metagpt/roles/ml_engineer_simple.py | 2 +- metagpt/roles/role.py | 4 +-- metagpt/schema.py | 2 +- tests/metagpt/actions/test_write_plan.py | 6 +---- tests/metagpt/roles/run_code_interpreter.py | 7 ++--- 15 files changed, 80 insertions(+), 77 deletions(-) diff --git a/metagpt/actions/ask_review.py b/metagpt/actions/ask_review.py index 85ac33bd8..7eb553b7e 100644 --- a/metagpt/actions/ask_review.py +++ b/metagpt/actions/ask_review.py @@ -30,7 +30,7 @@ class AskReview(Action): ) logger.info("most recent context:") - latest_action = context[-1].cause_by.__name__ if context[-1].cause_by else "" + latest_action = context[-1].cause_by if context[-1].cause_by else "" review_instruction = ( ReviewConst.TASK_REVIEW_INSTRUCTION if trigger == ReviewConst.TASK_REVIEW_TRIGGER diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index be09f3493..26a84bcf2 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -1,4 +1,4 @@ -from typing import Any, List, Optional +from typing import List from metagpt.actions.write_analysis_code import BaseWriteAnalysisCode from metagpt.logs import logger @@ -82,11 +82,6 @@ def messages_to_str(messages: List[Message]) -> str: class DebugCode(BaseWriteAnalysisCode): name: str = "debugcode" - context: Optional[str] = None - llm: None - - def __init__(self, **kwargs: Any): - super().__init__(**kwargs) async def run_reflection( self, diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index b2f6067ab..8355d3aca 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -8,7 +8,7 @@ import re import traceback from abc import ABC, abstractmethod from pathlib import Path -from typing import Dict, List, Tuple, Union +from typing import Any, Dict, List, Tuple, Union import nbformat from nbclient import NotebookClient @@ -48,23 +48,25 @@ class ExecuteCode(ABC): class ExecutePyCode(ExecuteCode, Action): """execute code, return result to llm, and display it.""" + nb: Any + nb_client: Any + console: Console + interaction: str + timeout: int = 600 + def __init__( self, - name: str = "python_executor", - context=None, - llm=None, nb=None, - timeout: int = 600, + timeout=600, ): - super().__init__(name, context, llm) - if nb is None: - self.nb = nbformat.v4.new_notebook() - else: - self.nb = nb - self.timeout = timeout - self.nb_client = NotebookClient(self.nb, timeout=self.timeout) - self.console = Console() - self.interaction = "ipython" if self.is_ipython() else "terminal" + nb = nb or nbformat.v4.new_notebook() + super().__init__( + nb=nb, + nb_client=NotebookClient(nb, timeout=timeout), + timeout=timeout, + console=Console(), + interaction=("ipython" if self.is_ipython() else "terminal"), + ) async def build(self): if self.nb_client.kc is None or not await self.nb_client.kc.is_alive(): diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py index 3ab5e0429..d4e77773f 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_da_action.py @@ -7,16 +7,13 @@ from metagpt.utils.common import CodeParser, create_func_config, remove_comments class SummarizeAnalysis(Action): - PROMPT_TEMPLATE = """ + PROMPT_TEMPLATE: str = """ # Context {context} # Summary Output a 30-word summary on analysis tool and modeling algorithms you have used, and the corresponding result. Make sure to announce the complete path to your test prediction file. Your summary: """ - def __init__(self, name: str = "", context=None, llm=None) -> str: - super().__init__(name, context, llm) - async def run(self, conmpleted_plan: Plan) -> str: tasks = json.dumps( [task.dict() for task in conmpleted_plan.tasks], @@ -29,7 +26,7 @@ class SummarizeAnalysis(Action): class Reflect(Action): - PROMPT_TEMPLATE = """ + PROMPT_TEMPLATE: str = """ # Context __context__ # Latest User Requirement @@ -45,7 +42,7 @@ class Reflect(Action): } ``` """ - REWRITE_PLAN_INSTRUCTION = """Take this reflection for rewriting plan, modify the current plan in place, make reference to your specific instruction, think about you should + REWRITE_PLAN_INSTRUCTION: str = """Take this reflection for rewriting plan, modify the current plan in place, make reference to your specific instruction, think about you should change which task, add or delete what tasks in the plan. Only make necessary changes, keep reusable tasks unchanged, output the COMPLETE new plan starting from the first task. Your plan should have no more than 5 tasks.""" async def run(self, context: str, user_requirement: str = "") -> str: diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index b0c8dab3b..d1e108b54 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -28,7 +28,7 @@ from metagpt.utils.common import create_func_config, remove_comments class BaseWriteAnalysisCode(Action): - DEFAULT_SYSTEM_MSG = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt + DEFAULT_SYSTEM_MSG: str = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): @@ -76,9 +76,6 @@ class BaseWriteAnalysisCode(Action): class WriteCodeByGenerate(BaseWriteAnalysisCode): """Write code fully by generation""" - def __init__(self, name: str = "", context=None, llm=None) -> str: - super().__init__(name, context, llm) - async def run( self, context: [List[Message]], @@ -95,12 +92,14 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - def __init__(self, name: str = "", context=None, llm=None, schema_path=None): - super().__init__(name, context, llm) - self.schema_path = schema_path - self.available_tools = {} + schema_path: str = "" + available_tools: dict = {} - if self.schema_path is not None: + def __init__(self, schema_path="", **kwargs): + super().__init__(**kwargs) + self.schema_path = schema_path + + if schema_path: self._load_tools(schema_path) def _load_tools(self, schema_path, schema_module=None): @@ -223,7 +222,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): class MakeTools(WriteCodeByGenerate): - DEFAULT_SYSTEM_MSG = """Convert any codes provied for you to a very General Function Code startswith `def`.\n + DEFAULT_SYSTEM_MSG: str = """Convert any codes provied for you to a very General Function Code startswith `def`.\n **Notice: 1. Your code must contain a general function start with `def`. 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index d2553e609..16680e395 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -16,7 +16,7 @@ from metagpt.utils.common import CodeParser, create_func_config class WritePlan(Action): - PROMPT_TEMPLATE = """ + PROMPT_TEMPLATE: str = """ # Context: __context__ # Task: diff --git a/metagpt/plan/planner.py b/metagpt/plan/planner.py index dadc2e563..87492e455 100644 --- a/metagpt/plan/planner.py +++ b/metagpt/plan/planner.py @@ -1,5 +1,7 @@ import json +from pydantic import BaseModel, Field + from metagpt.actions.ask_review import AskReview, ReviewConst from metagpt.actions.write_plan import ( WritePlan, @@ -22,14 +24,17 @@ STRUCTURAL_CONTEXT = """ """ -class Planner: - def __init__(self, goal: str, working_memory: Memory, auto_run: bool = False, use_tools: bool = False): - self.plan = Plan(goal=goal) - self.auto_run = auto_run - self.use_tools = use_tools +class Planner(BaseModel): + plan: Plan + working_memory: Memory = Field( + default_factory=Memory + ) # memory for working on each task, discarded each time a task is done + auto_run: bool = False + use_tools: bool = False - # memory for working on each task, discarded each time a task is done - self.working_memory = working_memory + def __init__(self, goal: str, **kwargs): + plan = Plan(goal=goal) + super().__init__(plan=plan, **kwargs) @property def current_task(self): diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 25890bc93..390666fd5 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -1,5 +1,7 @@ from datetime import datetime +from pydantic import Field + from metagpt.actions.ask_review import ReviewConst from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.write_analysis_code import WriteCodeByGenerate @@ -10,6 +12,8 @@ from metagpt.utils.save_code import save_code_file class CodeInterpreter(Role): + execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) + def __init__( self, name="Charlie", @@ -20,11 +24,10 @@ class CodeInterpreter(Role): ): super().__init__(name=name, profile=profile, goal=goal) self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run, use_tools=use_tools) - self.execute_code = ExecutePyCode() @property def working_memory(self): - return self._rc.working_memory + return self.rc.working_memory async def _plan_and_act(self): rsp = await super()._plan_and_act() diff --git a/metagpt/roles/kaggle_manager.py b/metagpt/roles/kaggle_manager.py index e12f47051..3ef573a8c 100644 --- a/metagpt/roles/kaggle_manager.py +++ b/metagpt/roles/kaggle_manager.py @@ -5,10 +5,9 @@ import subprocess import fire import pandas as pd -from metagpt.actions import Action, BossRequirement +from metagpt.actions import Action, UserRequirement from metagpt.actions.ml_da_action import SummarizeAnalysis from metagpt.config import CONFIG -from metagpt.const import WORKSPACE_ROOT from metagpt.logs import logger from metagpt.roles import Role from metagpt.schema import Message @@ -31,7 +30,7 @@ def run_command(cmd): class DownloadData(Action): async def run(self, competition, data_desc="") -> str: - data_path = WORKSPACE_ROOT / competition + data_path = CONFIG.workspace_path / competition output = run_command(f"kaggle competitions list --search {competition}") assert output != "No competitions found", "You must provide the correct competition name" @@ -41,7 +40,7 @@ class DownloadData(Action): if not os.path.exists(data_path): # if True: # run_command(f"rm -r {data_path / '*'}") - run_command(f"unzip -o {WORKSPACE_ROOT / '*.zip'} -d {data_path}") # FIXME: not safe + run_command(f"unzip -o {CONFIG.workspace_path / '*.zip'} -d {data_path}") # FIXME: not safe file_list = run_command(f"ls {data_path}") @@ -55,7 +54,7 @@ class DownloadData(Action): class SubmitResult(Action): - PROMPT_TEMPLATE = """ + PROMPT_TEMPLATE: str = """ # Summary __summary__ # Your task @@ -78,7 +77,7 @@ class SubmitResult(Action): async def run(self, competition, submit_message="") -> str: submit_file_path = await self._parse_submit_file_path(submit_message) - data_path = WORKSPACE_ROOT / competition + data_path = CONFIG.workspace_path / competition submit_message = submit_message.replace("'", "") run_command(f"kaggle competitions submit {competition} -f {submit_file_path} -m '{submit_message}'") @@ -108,20 +107,20 @@ class KaggleManager(Role): def __init__(self, name="ABC", profile="KaggleManager", goal="", competition="titanic", data_desc=""): super().__init__(name=name, profile=profile, goal=goal) self._init_actions([DownloadData, SubmitResult]) - self._watch([BossRequirement, SummarizeAnalysis]) + self._watch([UserRequirement, SummarizeAnalysis]) self.competition = competition self.data_desc = data_desc # currently passed in, later can be scrapped down from web by another Role async def _think(self): observed = self.get_memories()[-1].cause_by - if observed == BossRequirement: + if observed == UserRequirement: self._set_state(0) # DownloadData, get competition of interest from human, download datasets elif observed == SummarizeAnalysis: self._set_state(1) # SubmitResult, get prediction from MLEngineer and submit it to Kaggle async def _act(self): - todo = self._rc.todo - logger.info(f"{self._setting}: ready to {self._rc.todo}") + todo = self.rc.todo + logger.info(f"{self._setting}: ready to {self.rc.todo}") if isinstance(todo, DownloadData): rsp = await todo.run(self.competition, self.data_desc) @@ -148,7 +147,7 @@ if __name__ == "__main__": async def main(requirement: str = requirement): role = KaggleManager(competition=competition, data_desc=data_desc) - # await role.run(Message(content="", cause_by=BossRequirement)) + # await role.run(Message(content="", cause_by=UserRequirement)) await role.run(Message(content=summary, cause_by=SummarizeAnalysis)) fire.Fire(main) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index a631daa47..a230b2e2d 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -10,7 +10,7 @@ from metagpt.actions.write_analysis_code import ( WriteCodeWithTools, ) from metagpt.actions.write_code_steps import WriteCodeSteps -from metagpt.const import PROJECT_ROOT +from metagpt.const import METAGPT_ROOT from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter from metagpt.roles.kaggle_manager import DownloadData, SubmitResult @@ -20,6 +20,13 @@ from metagpt.utils.common import remove_comments class MLEngineer(CodeInterpreter): + auto_run: bool = False + use_tools: bool = False + use_code_steps: bool = False + make_udfs: bool = False # whether to save user-defined functions + use_udfs: bool = False + data_desc: dict = {} + def __init__( self, name="Mark", @@ -32,13 +39,12 @@ class MLEngineer(CodeInterpreter): use_udfs=False, ): super().__init__(name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools) - self._watch([DownloadData, SubmitResult]) - + self.auto_run = auto_run self.use_tools = use_tools self.use_code_steps = use_code_steps - self.make_udfs = make_udfs # user-defined functions + self.make_udfs = make_udfs self.use_udfs = use_udfs - self.data_desc = {} + # self._watch([DownloadData, SubmitResult]) # in multi-agent settings async def _plan_and_act(self): ### Actions in a multi-agent multi-turn setting, a new attempt on the data ### @@ -60,7 +66,7 @@ class MLEngineer(CodeInterpreter): ### summarize analysis ### summary = await SummarizeAnalysis().run(self.planner.plan) rsp = Message(content=summary, cause_by=SummarizeAnalysis) - self._rc.memory.add(rsp) + self.rc.memory.add(rsp) return rsp @@ -108,7 +114,7 @@ class MLEngineer(CodeInterpreter): self.planner.current_task.task_type = "udf" schema_path = UDFS_YAML else: - schema_path = PROJECT_ROOT / "metagpt/tools/functions/schemas" + schema_path = METAGPT_ROOT / "metagpt/tools/functions/schemas" tool_context, code = await WriteCodeWithTools(schema_path=schema_path).run( context=context, plan=self.planner.plan, diff --git a/metagpt/roles/ml_engineer_simple.py b/metagpt/roles/ml_engineer_simple.py index 1006a4262..3f10af8d0 100644 --- a/metagpt/roles/ml_engineer_simple.py +++ b/metagpt/roles/ml_engineer_simple.py @@ -95,7 +95,7 @@ class MLEngineerSimple(Role): counter = 0 # redo the task again with help of human suggestions completed_plan_memory = self.get_useful_memories() # completed plan as a outcome - self._rc.memory.add(completed_plan_memory[0]) # add to persistent memory + self.rc.memory.add(completed_plan_memory[0]) # add to persistent memory prompt = JUDGE_PROMPT_TEMPLATE.format(user_requirement=self.goal, context=completed_plan_memory) rsp = await self._llm.aask(prompt) self.working_memory.add(Message(content=rsp, role="system")) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 0ea6d6ee6..a2f2f2e9d 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -146,6 +146,7 @@ class Role(SerializationMixin, is_polymorphic_base=True): actions: list[SerializeAsAny[Action]] = Field(default=[], validate_default=True) rc: RoleContext = Field(default_factory=RoleContext) subscription: set[str] = set() + planner: Planner = None # builtin variables recovered: bool = False # to tag if a recovered role @@ -173,7 +174,6 @@ class Role(SerializationMixin, is_polymorphic_base=True): self.llm.system_prompt = self._get_prefix() self._watch(data.get("watch") or [UserRequirement]) - self.planner = None def _reset(self): self.states = [] @@ -270,7 +270,7 @@ class Role(SerializationMixin, is_polymorphic_base=True): self.rc.max_react_loop = max_react_loop elif react_mode == RoleReactMode.PLAN_AND_ACT: self.planner = Planner( - goal=self._setting.goal, working_memory=self.rc.working_memory, auto_run=auto_run, use_tools=use_tools + goal=self.goal, working_memory=self.rc.working_memory, auto_run=auto_run, use_tools=use_tools ) def _watch(self, actions: Iterable[Type[Action]] | Iterable[Action]): diff --git a/metagpt/schema.py b/metagpt/schema.py index 31a83e5dd..e69f432db 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -337,7 +337,7 @@ class Plan(BaseModel): context: str = "" tasks: list[Task] = [] task_map: dict[str, Task] = {} - current_task_id = "" + current_task_id: str = "" def _topological_sort(self, tasks: list[Task]): task_map = {task.task_id: task for task in tasks} diff --git a/tests/metagpt/actions/test_write_plan.py b/tests/metagpt/actions/test_write_plan.py index 6f2e7d430..e1c93e8b2 100644 --- a/tests/metagpt/actions/test_write_plan.py +++ b/tests/metagpt/actions/test_write_plan.py @@ -1,8 +1,4 @@ -from metagpt.actions.write_plan import ( - Plan, - Task, - precheck_update_plan_from_rsp, -) +from metagpt.actions.write_plan import Plan, Task, precheck_update_plan_from_rsp def test_precheck_update_plan_from_rsp(): diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py index 418270e25..7c5c1939b 100644 --- a/tests/metagpt/roles/run_code_interpreter.py +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -1,6 +1,7 @@ import fire from metagpt.actions.execute_code import ExecutePyCode +from metagpt.const import DATA_PATH from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter from metagpt.roles.ml_engineer import MLEngineer @@ -53,10 +54,10 @@ async def run_code_interpreter( if __name__ == "__main__": - requirement = "Run data analysis on sklearn Iris dataset, include a plot" + # requirement = "Run data analysis on sklearn Iris dataset, include a plot" # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - # data_path = f"{DATA_PATH}/titanic" - # requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + data_path = f"{DATA_PATH}/titanic" + requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" From 0788080e205a5437f1974f77b4a203f0b57d1f07 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Wed, 10 Jan 2024 19:51:38 +0800 Subject: [PATCH 281/637] fixbug: fix todo_description --- metagpt/roles/role.py | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 3bcd600fc..3d5e55057 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -504,7 +504,13 @@ class Role(SerializationMixin, is_polymorphic_base=True): @property def todo(self) -> str: - """AgentStore uses this attribute to display to the user what actions the current role should take.""" + """ + AgentStore uses this attribute to display to the user what actions the current role should take. + """ + if self.rc.todo: + if self.rc.todo.desc: + return self.rc.todo.desc + return any_to_name(self.rc.todo) if self.actions: return any_to_name(self.actions[0]) return "" From cf2366b72ce2f5fcc8f5a283733eb48f35cf1c35 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Wed, 10 Jan 2024 21:13:36 +0800 Subject: [PATCH 282/637] feat: +ver --- setup.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/setup.py b/setup.py index d997b5f62..ea84fe299 100644 --- a/setup.py +++ b/setup.py @@ -57,7 +57,7 @@ extras_require["dev"] = (["pylint~=3.0.3", "black~=23.3.0", "isort~=5.12.0", "pr setup( name="metagpt", - version="0.6.3", + version="0.6.4", description="The Multi-Agent Framework", long_description=long_description, long_description_content_type="text/markdown", From e12ab25b7c51475c15eeaeba0eb9dfec472b889f Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 11 Jan 2024 00:23:26 +0800 Subject: [PATCH 283/637] generalize write code with tools, simplify ml_engineer --- metagpt/actions/write_analysis_code.py | 72 +++++-- metagpt/prompts/ml_engineer.py | 22 ++- metagpt/roles/code_interpreter.py | 42 ++++- metagpt/roles/ml_engineer.py | 199 +++++++------------- metagpt/roles/tool_maker.py | 46 +++++ tests/metagpt/roles/run_code_interpreter.py | 2 +- 6 files changed, 221 insertions(+), 162 deletions(-) create mode 100644 metagpt/roles/tool_maker.py diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index d1e108b54..aef86122b 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -12,14 +12,16 @@ import yaml from tenacity import retry, stop_after_attempt, wait_fixed from metagpt.actions import Action +from metagpt.const import METAGPT_ROOT from metagpt.llm import LLM from metagpt.logs import logger from metagpt.prompts.ml_engineer import ( CODE_GENERATOR_WITH_TOOLS, GENERATE_CODE_PROMPT, - ML_MODULE_MAP, - ML_SPECIFIC_PROMPT, + ML_TOOL_USAGE_PROMPT, SELECT_FUNCTION_TOOLS, + TASK_MODULE_MAP, + TASK_SPECIFIC_PROMPT, TOOL_RECOMMENDATION_PROMPT, TOOL_USAGE_PROMPT, ) @@ -60,13 +62,12 @@ class BaseWriteAnalysisCode(Action): } return messages - async def run(self, context: List[Message], plan: Plan = None, code_steps: str = "") -> str: + async def run(self, context: List[Message], plan: Plan = None) -> str: """Run of a code writing action, used in data analysis or modeling Args: context (List[Message]): Action output history, source action denoted by Message.cause_by plan (Plan, optional): Overall plan. Defaults to None. - code_steps (str, optional): suggested step breakdown for the current task. Defaults to "". Returns: str: The code string. @@ -92,15 +93,12 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - schema_path: str = "" + schema_path: Union[Path, str] = METAGPT_ROOT / "metagpt/tools/functions/schemas" available_tools: dict = {} - def __init__(self, schema_path="", **kwargs): + def __init__(self, **kwargs): super().__init__(**kwargs) - self.schema_path = schema_path - - if schema_path: - self._load_tools(schema_path) + self._load_tools(self.schema_path) def _load_tools(self, schema_path, schema_module=None): """Load tools from yaml file""" @@ -171,12 +169,11 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): self, context: List[Message], plan: Plan = None, - column_info: str = "", **kwargs, - ) -> Tuple[List[Message], str]: + ) -> str: task_type = plan.current_task.task_type available_tools = self.available_tools.get(task_type, {}) - special_prompt = ML_SPECIFIC_PROMPT.get(task_type, "") + special_prompt = TASK_SPECIFIC_PROMPT.get(task_type, "") code_steps = plan.current_task.code_steps finished_tasks = plan.get_finished_tasks() @@ -192,9 +189,54 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) logger.info(f"Recommended tools: \n{recommend_tools}") - module_name = ML_MODULE_MAP[task_type] + module_name = TASK_MODULE_MAP[task_type] - prompt = TOOL_USAGE_PROMPT.format( + else: + tool_catalog = {} + module_name = "" + + tools_instruction = TOOL_USAGE_PROMPT.format( + special_prompt=special_prompt, module_name=module_name, tool_catalog=tool_catalog + ) + + context.append(Message(content=tools_instruction, role="user")) + + prompt = self.process_msg(context) + + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + rsp = await self.llm.aask_code(prompt, **tool_config) + return rsp["code"] + + +class WriteCodeWithToolsML(WriteCodeWithTools): + async def run( + self, + context: List[Message], + plan: Plan = None, + column_info: str = "", + **kwargs, + ) -> Tuple[List[Message], str]: + task_type = plan.current_task.task_type + available_tools = self.available_tools.get(task_type, {}) + special_prompt = TASK_SPECIFIC_PROMPT.get(task_type, "") + code_steps = plan.current_task.code_steps + + finished_tasks = plan.get_finished_tasks() + code_context = [remove_comments(task.code) for task in finished_tasks] + code_context = "\n\n".join(code_context) + + if len(available_tools) > 0: + available_tools = {k: v["description"] for k, v in available_tools.items()} + + recommend_tools = await self._tool_recommendation( + plan.current_task.instruction, code_steps, available_tools + ) + tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) + logger.info(f"Recommended tools: \n{recommend_tools}") + + module_name = TASK_MODULE_MAP[task_type] + + prompt = ML_TOOL_USAGE_PROMPT.format( user_requirement=plan.goal, history_code=code_context, current_task=plan.current_task.instruction, diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 9b873d39f..13ee4db42 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -198,6 +198,24 @@ model.fit(train, y_train) """ TOOL_USAGE_PROMPT = """ +# Instruction +Write complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc. +Specifically, {special_prompt} + +# Capabilities +- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class. +- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. + +# Available Tools (can be empty): +Each Class tool is described in JSON format. When you call a tool, import the tool from `{module_name}` first. +{tool_catalog} + +# Constraints: +- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. +- Always prioritize using pre-defined tools for the same functionality. +""" + +ML_TOOL_USAGE_PROMPT = """ # Background As a data scientist, you need to help user to achieve their goal [{user_requirement}] step-by-step in an continuous Jupyter notebook. @@ -297,14 +315,14 @@ The current task is about evaluating a model, please note the following: - Use trained model from previous task result directly, do not mock or reload model yourself. """ -ML_SPECIFIC_PROMPT = { +TASK_SPECIFIC_PROMPT = { "data_preprocess": DATA_PREPROCESS_PROMPT, "feature_engineering": FEATURE_ENGINEERING_PROMPT, "model_train": MODEL_TRAIN_PROMPT, "model_evaluate": MODEL_EVALUATE_PROMPT, } -ML_MODULE_MAP = { +TASK_MODULE_MAP = { "data_preprocess": "metagpt.tools.functions.libs.data_preprocess", "feature_engineering": "metagpt.tools.functions.libs.feature_engineering", "udf": "metagpt.tools.functions.libs.udf", diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 390666fd5..9bb543d99 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -4,14 +4,20 @@ from pydantic import Field from metagpt.actions.ask_review import ReviewConst from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_analysis_code import WriteCodeByGenerate +from metagpt.actions.write_analysis_code import ( + WriteCodeByGenerate, + WriteCodeWithTools, +) from metagpt.logs import logger from metagpt.roles import Role +from metagpt.roles.tool_maker import ToolMaker from metagpt.schema import Message, Task, TaskResult from metagpt.utils.save_code import save_code_file class CodeInterpreter(Role): + use_tools: bool = False + make_udfs: bool = False # whether to save user-defined functions execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) def __init__( @@ -21,8 +27,10 @@ class CodeInterpreter(Role): goal="", auto_run=False, use_tools=False, + make_udfs=False, + **kwargs, ): - super().__init__(name=name, profile=profile, goal=goal) + super().__init__(name=name, profile=profile, goal=goal, use_tools=use_tools, make_udfs=make_udfs, **kwargs) self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run, use_tools=use_tools) @property @@ -36,6 +44,10 @@ class CodeInterpreter(Role): project_record = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") + # make tools out of workable codes for future use + if self.make_udfs: + await self.make_tools() + return rsp async def _act_on_task(self, current_task: Task) -> TaskResult: @@ -48,20 +60,18 @@ class CodeInterpreter(Role): success = False while not success and counter < max_retry: - context = self.planner.get_useful_memories() - - logger.info("Write code with pure generation") - - code = await WriteCodeByGenerate().run(context=context, plan=self.planner.plan, temperature=0.0) - cause_by = WriteCodeByGenerate + ### write code ### + code, cause_by = await self._write_code() self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) + ### execute code ### result, success = await self.execute_code.run(code) print(result) self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) + ### process execution result ### if "!pip" in code: success = False @@ -74,3 +84,19 @@ class CodeInterpreter(Role): counter = 0 # redo the task again with help of human suggestions return code, result, success + + async def _write_code(self): + todo = WriteCodeByGenerate() if not self.use_tools else WriteCodeWithTools() + logger.info(f"ready to {todo.name}") + + context = self.planner.get_useful_memories() + code = await todo.run(context=context, plan=self.planner.plan, temperature=0.0) + + return code, todo + + async def make_tools(self): + """Make user-defined functions(udfs, aka tools) for pure generation code.""" + logger.info("Plan completed. Now start to make tools ...") + tool_maker = ToolMaker() + for task in self.planner.plan.get_finished_tasks(): + await tool_maker.make_tool(task.code, task.instruction, task.task_id) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index a230b2e2d..b6d660137 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,31 +1,23 @@ -import json - from metagpt.actions.ask_review import ReviewConst from metagpt.actions.debug_code import DebugCode from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.ml_da_action import Reflect, SummarizeAnalysis, UpdateDataColumns -from metagpt.actions.write_analysis_code import ( - MakeTools, - WriteCodeByGenerate, - WriteCodeWithTools, -) +from metagpt.actions.write_analysis_code import WriteCodeWithToolsML from metagpt.actions.write_code_steps import WriteCodeSteps -from metagpt.const import METAGPT_ROOT from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter from metagpt.roles.kaggle_manager import DownloadData, SubmitResult from metagpt.schema import Message -from metagpt.tools.functions.libs.udf import UDFS_YAML -from metagpt.utils.common import remove_comments +from metagpt.utils.common import any_to_str class MLEngineer(CodeInterpreter): auto_run: bool = False - use_tools: bool = False use_code_steps: bool = False - make_udfs: bool = False # whether to save user-defined functions use_udfs: bool = False data_desc: dict = {} + debug_context: list = [] + latest_code: str = "" def __init__( self, @@ -38,27 +30,21 @@ class MLEngineer(CodeInterpreter): make_udfs=False, use_udfs=False, ): - super().__init__(name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools) - self.auto_run = auto_run - self.use_tools = use_tools - self.use_code_steps = use_code_steps - self.make_udfs = make_udfs - self.use_udfs = use_udfs + super().__init__( + name=name, + profile=profile, + goal=goal, + auto_run=auto_run, + use_tools=use_tools, + use_code_steps=use_code_steps, + make_udfs=make_udfs, + use_udfs=use_udfs, + ) # self._watch([DownloadData, SubmitResult]) # in multi-agent settings async def _plan_and_act(self): - ### Actions in a multi-agent multi-turn setting, a new attempt on the data ### - memories = self.get_memories() - if memories: - latest_event = memories[-1].cause_by - if latest_event == DownloadData: - self.planner.plan.context = memories[-1].content - elif latest_event == SubmitResult: - # self reflect on previous plan outcomes and think about how to improve the plan, add to working memory - await self._reflect() - - # get feedback for improvement from human, add to working memory - await self.planner.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) + ### a new attempt on the data, relevant in a multi-agent multi-turn setting ### + await self._prepare_data_context() ### general plan process ### await super()._plan_and_act() @@ -75,85 +61,48 @@ class MLEngineer(CodeInterpreter): await WriteCodeSteps().run(self.planner.plan) if self.use_code_steps else "" ) - counter = 0 - success = False - debug_context = [] - - while not success and counter < max_retry: - context = self.planner.get_useful_memories() - - if counter > 0 and (self.use_tools or self.use_udfs): - logger.warning("We got a bug code, now start to debug...") - code = await DebugCode().run( - plan=self.planner.current_task.instruction, - code=code, - runtime_result=self.working_memory.get(), - context=debug_context, - ) - logger.info(f"new code \n{code}") - cause_by = DebugCode - - elif (not self.use_tools and not self.use_udfs) or ( - self.planner.current_task.task_type == "other" and not self.use_udfs - ): - logger.info("Write code with pure generation") - code = await WriteCodeByGenerate().run(context=context, plan=self.planner.plan, temperature=0.0) - debug_context = [self.planner.get_useful_memories(task_exclude_field={"result", "code_steps"})[0]] - cause_by = WriteCodeByGenerate - - else: - logger.info("Write code with tools") - if self.use_udfs: - # use user-defined function tools. - logger.warning("Writing code with user-defined function tools by WriteCodeWithTools.") - logger.info( - f"Local user defined function as following:\ - \n{json.dumps(list(UDFS_YAML.keys()), indent=2, ensure_ascii=False)}" - ) - # set task_type to `udf` - self.planner.current_task.task_type = "udf" - schema_path = UDFS_YAML - else: - schema_path = METAGPT_ROOT / "metagpt/tools/functions/schemas" - tool_context, code = await WriteCodeWithTools(schema_path=schema_path).run( - context=context, - plan=self.planner.plan, - column_info=self.data_desc.get("column_info", ""), - ) - debug_context = tool_context - cause_by = WriteCodeWithTools - - self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) - - result, success = await self.execute_code.run(code) - print(result) - # make tools for successful code and long code. - if success and self.make_udfs and len(remove_comments(code).split("\n")) > 4: - logger.info("Execute code successfully. Now start to make tools ...") - await self.make_tools(code=code) - self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) - - if "!pip" in code: - success = False - - counter += 1 - - if not success and counter >= max_retry: - logger.info("coding failed!") - review, _ = await self.planner.ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) - if ReviewConst.CHANGE_WORD[0] in review: - counter = 0 # redo the task again with help of human suggestions + code, result, success = await super()._write_and_exec_code(max_retry=max_retry) if success: - if ( - self.use_tools and self.planner.current_task.task_type not in ["model_train", "model_evaluate"] - ) or self.use_udfs: + if self.use_tools and self.planner.current_task.task_type in ["data_preprocess", "feature_engineering"]: update_success, new_code = await self._update_data_columns() if update_success: code = code + "\n\n" + new_code return code, result, success + async def _write_code(self): + if not self.use_tools: + return await super()._write_code() + + code_execution_count = sum([msg.cause_by == any_to_str(ExecutePyCode) for msg in self.working_memory.get()]) + print("*" * 10, code_execution_count) + + if code_execution_count > 0: + logger.warning("We got a bug code, now start to debug...") + code = await DebugCode().run( + plan=self.planner.current_task.instruction, + code=self.latest_code, + runtime_result=self.working_memory.get(), + context=self.debug_context, + ) + logger.info(f"new code \n{code}") + cause_by = DebugCode + + else: + logger.info("Write code with tools") + tool_context, code = await WriteCodeWithToolsML().run( + context=[], # context assembled inside the Action + plan=self.planner.plan, + column_info=self.data_desc.get("column_info", ""), + ) + self.debug_context = tool_context + cause_by = WriteCodeWithToolsML + + self.latest_code = code + + return code, cause_by + async def _update_data_columns(self): logger.info("Check columns in updated data") rsp = await UpdateDataColumns().run(self.planner.plan) @@ -166,6 +115,19 @@ class MLEngineer(CodeInterpreter): self.data_desc["column_info"] = result return success, code + async def _prepare_data_context(self): + memories = self.get_memories() + if memories: + latest_event = memories[-1].cause_by + if latest_event == DownloadData: + self.planner.plan.context = memories[-1].content + elif latest_event == SubmitResult: + # self reflect on previous plan outcomes and think about how to improve the plan, add to working memory + await self._reflect() + + # get feedback for improvement from human, add to working memory + await self.planner.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) + async def _reflect(self): context = self.get_memories() context = "\n".join([str(msg) for msg in context]) @@ -173,38 +135,3 @@ class MLEngineer(CodeInterpreter): reflection = await Reflect().run(context=context) self.working_memory.add(Message(content=reflection, role="assistant")) self.working_memory.add(Message(content=Reflect.REWRITE_PLAN_INSTRUCTION, role="user")) - - async def make_tools(self, code: str): - """Make user-defined functions(udfs, aka tools) for pure generation code. - - Args: - code (str): pure generation code by class WriteCodeByGenerate. - """ - logger.warning( - f"Making tools for task_id {self.planner.current_task_id}: \ - `{self.planner.current_task.instruction}` \n code: \n {code}" - ) - make_tools = MakeTools() - make_tool_retries, make_tool_current_retry = 3, 0 - while True: - # start make tools - tool_code = await make_tools.run(code, self.planner.current_task.instruction) - make_tool_current_retry += 1 - - # check tool_code by execute_code - logger.info(f"Checking task_id {self.planner.current_task_id} tool code by executor...") - execute_result, execute_success = await self.execute_code.run(tool_code) - if not execute_success: - logger.error(f"Tool code faild to execute, \n{execute_result}\n.We will try to fix it ...") - # end make tools - if execute_success or make_tool_current_retry >= make_tool_retries: - if make_tool_current_retry >= make_tool_retries: - logger.error( - f"We have tried the maximum number of attempts {make_tool_retries}\ - and still have not created tools for task_id {self.planner.current_task_id} successfully,\ - we will skip it." - ) - break - # save successful tool code in udf - if execute_success: - make_tools.save(tool_code) diff --git a/metagpt/roles/tool_maker.py b/metagpt/roles/tool_maker.py new file mode 100644 index 000000000..a2f854adb --- /dev/null +++ b/metagpt/roles/tool_maker.py @@ -0,0 +1,46 @@ +from pydantic import Field + +from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.write_analysis_code import ( + MakeTools, +) +from metagpt.logs import logger +from metagpt.roles import Role +from metagpt.utils.common import remove_comments + + +class ToolMaker(Role): + execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) + + async def make_tool(self, code: str, instruction: str, task_id: str = ""): + if len(remove_comments(code).split("\n")) < 5: # no need to consider trivial codes with fewer than 5 lines + return + + logger.warning( + f"Making tools for task_id {task_id}: \ + `{instruction}` \n code: \n {code}" + ) + make_tools = MakeTools() + make_tool_retries, make_tool_current_retry = 3, 0 + while True: + # start make tools + tool_code = await make_tools.run(code, instruction) + make_tool_current_retry += 1 + + # check tool_code by execute_code + logger.info(f"Checking task_id {task_id} tool code by executor...") + execute_result, execute_success = await self.execute_code.run(tool_code) + if not execute_success: + logger.error(f"Tool code faild to execute, \n{execute_result}\n.We will try to fix it ...") + # end make tools + if execute_success or make_tool_current_retry >= make_tool_retries: + if make_tool_current_retry >= make_tool_retries: + logger.error( + f"We have tried the maximum number of attempts {make_tool_retries}\ + and still have not created tools for task_id {task_id} successfully,\ + we will skip it." + ) + break + # save successful tool code in udf + if execute_success: + make_tools.save(tool_code) diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py index 7c5c1939b..539b20286 100644 --- a/tests/metagpt/roles/run_code_interpreter.py +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -25,7 +25,7 @@ async def run_code_interpreter( """ if role_class == "ci": - role = CodeInterpreter(goal=requirement, auto_run=auto_run, use_tools=use_tools) + role = CodeInterpreter(goal=requirement, auto_run=auto_run, use_tools=use_tools, make_udfs=make_udfs) else: role = MLEngineer( goal=requirement, From 4ecd427bea9c42af1595af9fdee4785a7d0a6934 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 11 Jan 2024 00:47:28 +0800 Subject: [PATCH 284/637] formatting --- metagpt/roles/code_interpreter.py | 5 +---- metagpt/roles/ml_engineer.py | 24 ++---------------------- metagpt/roles/tool_maker.py | 4 +--- 3 files changed, 4 insertions(+), 29 deletions(-) diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 9bb543d99..6bbd923e6 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -4,10 +4,7 @@ from pydantic import Field from metagpt.actions.ask_review import ReviewConst from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_analysis_code import ( - WriteCodeByGenerate, - WriteCodeWithTools, -) +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.logs import logger from metagpt.roles import Role from metagpt.roles.tool_maker import ToolMaker diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index b6d660137..639a517d6 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -19,27 +19,8 @@ class MLEngineer(CodeInterpreter): debug_context: list = [] latest_code: str = "" - def __init__( - self, - name="Mark", - profile="MLEngineer", - goal="", - auto_run=False, - use_tools=False, - use_code_steps=False, - make_udfs=False, - use_udfs=False, - ): - super().__init__( - name=name, - profile=profile, - goal=goal, - auto_run=auto_run, - use_tools=use_tools, - use_code_steps=use_code_steps, - make_udfs=make_udfs, - use_udfs=use_udfs, - ) + def __init__(self, name="Mark", profile="MLEngineer", **kwargs): + super().__init__(name=name, profile=profile, **kwargs) # self._watch([DownloadData, SubmitResult]) # in multi-agent settings async def _plan_and_act(self): @@ -76,7 +57,6 @@ class MLEngineer(CodeInterpreter): return await super()._write_code() code_execution_count = sum([msg.cause_by == any_to_str(ExecutePyCode) for msg in self.working_memory.get()]) - print("*" * 10, code_execution_count) if code_execution_count > 0: logger.warning("We got a bug code, now start to debug...") diff --git a/metagpt/roles/tool_maker.py b/metagpt/roles/tool_maker.py index a2f854adb..7fec7b739 100644 --- a/metagpt/roles/tool_maker.py +++ b/metagpt/roles/tool_maker.py @@ -1,9 +1,7 @@ from pydantic import Field from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_analysis_code import ( - MakeTools, -) +from metagpt.actions.write_analysis_code import MakeTools from metagpt.logs import logger from metagpt.roles import Role from metagpt.utils.common import remove_comments From 3848596b0fa8990ab74002e44e426283df41c190 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 11 Jan 2024 14:04:44 +0800 Subject: [PATCH 285/637] update prompt --- metagpt/actions/write_code_guideline_an.py | 154 ++++++++++++--------- metagpt/roles/engineer.py | 11 +- 2 files changed, 98 insertions(+), 67 deletions(-) diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index 43645e80c..c08340cb7 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -10,70 +10,97 @@ import asyncio from metagpt.actions.action import Action from metagpt.actions.action_node import ActionNode -GUIDELINE = ActionNode( - key="Guideline", - expected_type=list[str], - instruction="Developing comprehensive and incremental development plans while providing detailed code guideline.", - example=[ - "Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Implement robust error handling for the division operation to mitigate potential issues related to division by zero.", - "Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Ensure seamless integration with the overall application architecture and maintain consistency with coding standards.", - ], -) - -INCREMENTAL_CHANGE = ActionNode( - key="Incremental Change", +GUIDELINES_AND_INCREMENTAL_CHANGE = ActionNode( + key="Guidelines and Incremental Change", expected_type=str, - instruction="Write Incremental Change by making a code draft that how to implement incremental development " - "including detailed steps based on the context.", - example="""- calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Implement robust error handling for the division operation to mitigate potential issues related to division by zero. + instruction="Developing comprehensive and step-by-step incremental development guideline, and Write Incremental " + "Change by making a code draft that how to implement incremental development including detailed steps based on the " + "context. Note: Track incremental changes using mark of '+' or '-' for add/modify/delete code", + example=""" +1. Guideline for calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Additionally, implement robust error handling for the division operation to mitigate potential issues related to division by zero. ```python -## calculator.py class Calculator: - ... - def subtract_numbers(self, num1: int, num2: int) -> int: - return num1 - num2 - def multiply_numbers(self, num1: int, num2: int) -> int: - return num1 * num2 - def divide_numbers(self, num1: int, num2: int) -> float: - if num2 == 0: - raise ValueError('Cannot divide by zero') - return num1 / num2 + self.result = number1 + number2 + return self.result + +- def sub(self, number1, number2) -> float: ++ def subtract(self, number1: float, number2: float) -> float: ++ ''' ++ Subtracts the second number from the first and returns the result. ++ ++ Args: ++ number1 (float): The number to be subtracted from. ++ number2 (float): The number to subtract. ++ ++ Returns: ++ float: The difference of number1 and number2. ++ ''' ++ self.result = number1 - number2 ++ return self.result ++ + def multiply(self, number1: float, number2: float) -> float: +- pass ++ ''' ++ Multiplies two numbers and returns the result. ++ ++ Args: ++ number1 (float): The first number to multiply. ++ number2 (float): The second number to multiply. ++ ++ Returns: ++ float: The product of number1 and number2. ++ ''' ++ self.result = number1 * number2 ++ return self.result ++ + def divide(self, number1: float, number2: float) -> float: +- pass ++ ''' ++ ValueError: If the second number is zero. ++ ''' ++ if number2 == 0: ++ raise ValueError('Cannot divide by zero') ++ self.result = number1 / number2 ++ return self.result ++ +- def reset_result(self): ++ def clear(self): ++ if self.result != 0.0: ++ print("Result is not zero, clearing...") ++ else: ++ print("Result is already zero, no need to clear.") ++ + self.result = 0.0 ``` -- main.py: Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Ensure seamless integration with the overall application architecture and maintain consistency with coding standards. +2. Guideline for main.py: Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Then, ensure seamless integration with the overall application architecture and maintain consistency with coding standards. ```python -## main.py -from flask import Flask, request, jsonify -from calculator import Calculator -app = Flask(__name__) -calculator = Calculator() -... -@app.route('/subtract_numbers', methods=['POST']) -def subtract_numbers(): - data = request.get_json() - num1 = data.get('num1', 0) - num2 = data.get('num2', 0) - result = calculator.subtract_numbers(num1, num2) - return jsonify({'result': result}), 200 -@app.route('/multiply_numbers', methods=['POST']) -def multiply_numbers(): - data = request.get_json() - num1 = data.get('num1', 0) - num2 = data.get('num2', 0) - result = calculator.multiply_numbers(num1, num2) - return jsonify({'result': result}), 200 -@app.route('/divide_numbers', methods=['POST']) -def divide_numbers(): - data = request.get_json() - num1 = data.get('num1', 1) - num2 = data.get('num2', 1) - try: - result = calculator.divide_numbers(num1, num2) - except ValueError as e: - return jsonify({'error': str(e)}), 400 - return jsonify({'result': result}), 200 -if __name__ == '__main__': - app.run() +def add_numbers(): + result = calculator.add_numbers(num1, num2) + return jsonify({'result': result}), 200 + +-# TODO: Implement subtraction, multiplication, and division operations ++@app.route('/subtract_numbers', methods=['POST']) ++def subtract_numbers(): ++ data = request.get_json() ++ num1 = data.get('num1', 0) ++ num2 = data.get('num2', 0) ++ result = calculator.subtract_numbers(num1, num2) ++ return jsonify({'result': result}), 200 ++ ++@app.route('/multiply_numbers', methods=['POST']) ++def multiply_numbers(): ++ data = request.get_json() ++ num1 = data.get('num1', 0) ++ num2 = data.get('num2', 0) ++ try: ++ result = calculator.divide_numbers(num1, num2) ++ except ValueError as e: ++ return jsonify({'error': str(e)}), 400 ++ return jsonify({'result': result}), 200 ++ + if __name__ == '__main__': + app.run() ```""", ) @@ -81,6 +108,9 @@ CODE_GUIDELINE_CONTEXT = """ ## New Requirements {requirement} +## PRD +{prd} + ## Design {design} @@ -403,22 +433,20 @@ Role: You are a professional engineer; The main goal is to complete incremental 2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets. 3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import. 4. Follow design: YOU MUST FOLLOW "Data structures and interfaces". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design. -5. Merge Incremental Change: If there is any Incremental Change, you must merge it into the code file. +5. Follow Guidelines and Incremental Change: If there is any Incremental Change, you must merge it into the code file according to the guidelines. 6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 7. Before using a external variable/module, make sure you import it first. 8. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. 9. Attention: If Legacy Code files contain "{filename} to be rewritten", you are required to merge the Incremental Change into the {filename} file when rewriting "{filename} to be rewritten". """ -GUIDE_NODES = [INCREMENTAL_CHANGE] - -WRITE_CODE_GUIDELINE_NODE = ActionNode.from_children("WriteCodeGuideline", GUIDE_NODES) +WRITE_CODE_GUIDELINE_NODE = ActionNode.from_children("WriteCodeGuideline", [GUIDELINES_AND_INCREMENTAL_CHANGE]) class WriteCodeGuideline(Action): async def run(self, context): self.llm.system_prompt = "You are a professional software engineer, your primary responsibility is to " - "meticulously craft comprehensive incremental development plans and deliver detailed Incremental Change" + "meticulously craft comprehensive incremental development guidelines and deliver detailed Incremental Change" return await WRITE_CODE_GUIDELINE_NODE.fill(context=context, llm=self.llm, schema="json") diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index a1e93be40..e0b22ea5b 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -36,6 +36,7 @@ from metagpt.config import CONFIG from metagpt.const import ( CODE_SUMMARIES_FILE_REPO, CODE_SUMMARIES_PDF_FILE_REPO, + PRDS_FILE_REPO, SYSTEM_DESIGN_FILE_REPO, TASK_FILE_REPO, ) @@ -346,18 +347,20 @@ class Engineer(Role): logger.info("Writing code guideline..") requirement = str(self.rc.memory.get_by_role("Human")[0]) - # prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) + prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) design_file_repo = CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO) task_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) - # prd = await prd_file_repo.get_all() - # prd = "\n".join([doc.content for doc in prd]) + prd = await prd_file_repo.get_all() + prd = "\n".join([doc.content for doc in prd]) design = await design_file_repo.get_all() design = "\n".join([doc.content for doc in design]) tasks = await task_file_repo.get_all() tasks = "\n".join([doc.content for doc in tasks]) old_codes = await self.get_old_codes() - context = CODE_GUIDELINE_CONTEXT.format(requirement=requirement, tasks=tasks, design=design, code=old_codes) + context = CODE_GUIDELINE_CONTEXT.format( + requirement=requirement, prd=prd, tasks=tasks, design=design, code=old_codes + ) node = await WriteCodeGuideline().run(context=context) guideline = node.instruct_content.model_dump_json() From 437bbca466397b7e639e879e9e2cae0e735bc76c Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 11 Jan 2024 14:10:52 +0800 Subject: [PATCH 286/637] make tool ask review --- metagpt/actions/ask_review.py | 13 +++++++------ metagpt/actions/write_analysis_code.py | 4 ++-- metagpt/const.py | 1 + metagpt/roles/code_interpreter.py | 7 ++++--- metagpt/roles/ml_engineer.py | 1 - metagpt/roles/tool_maker.py | 13 +++++++++++-- 6 files changed, 25 insertions(+), 14 deletions(-) diff --git a/metagpt/actions/ask_review.py b/metagpt/actions/ask_review.py index 7eb553b7e..0d671648b 100644 --- a/metagpt/actions/ask_review.py +++ b/metagpt/actions/ask_review.py @@ -23,14 +23,15 @@ class ReviewConst: class AskReview(Action): - async def run(self, context: List[Message], plan: Plan = None, trigger: str = "task"): - logger.info("Current overall plan:") - logger.info( - "\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks]) - ) + async def run(self, context: List[Message] = [], plan: Plan = None, trigger: str = "task"): + if plan: + logger.info("Current overall plan:") + logger.info( + "\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks]) + ) logger.info("most recent context:") - latest_action = context[-1].cause_by if context[-1].cause_by else "" + latest_action = context[-1].cause_by if context and context[-1].cause_by else "" review_instruction = ( ReviewConst.TASK_REVIEW_INSTRUCTION if trigger == ReviewConst.TASK_REVIEW_TRIGGER diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index aef86122b..c5f9c9166 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -12,7 +12,7 @@ import yaml from tenacity import retry, stop_after_attempt, wait_fixed from metagpt.actions import Action -from metagpt.const import METAGPT_ROOT +from metagpt.const import METAGPT_ROOT, TOOL_SCHEMA_PATH from metagpt.llm import LLM from metagpt.logs import logger from metagpt.prompts.ml_engineer import ( @@ -93,7 +93,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - schema_path: Union[Path, str] = METAGPT_ROOT / "metagpt/tools/functions/schemas" + schema_path: Union[Path, str] = TOOL_SCHEMA_PATH available_tools: dict = {} def __init__(self, **kwargs): diff --git a/metagpt/const.py b/metagpt/const.py index 811ff9516..b1666e092 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -70,6 +70,7 @@ TMP = METAGPT_ROOT / "tmp" SOURCE_ROOT = METAGPT_ROOT / "metagpt" PROMPT_PATH = SOURCE_ROOT / "prompts" SKILL_DIRECTORY = SOURCE_ROOT / "skills" +TOOL_SCHEMA_PATH = METAGPT_ROOT / "metagpt/tools/functions/schemas" # REAL CONSTS diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 6bbd923e6..9b13d8dcb 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -13,6 +13,7 @@ from metagpt.utils.save_code import save_code_file class CodeInterpreter(Role): + auto_run: bool = True use_tools: bool = False make_udfs: bool = False # whether to save user-defined functions execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) @@ -22,12 +23,12 @@ class CodeInterpreter(Role): name="Charlie", profile="CodeInterpreter", goal="", - auto_run=False, + auto_run=True, use_tools=False, make_udfs=False, **kwargs, ): - super().__init__(name=name, profile=profile, goal=goal, use_tools=use_tools, make_udfs=make_udfs, **kwargs) + super().__init__(name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools, make_udfs=make_udfs, **kwargs) self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run, use_tools=use_tools) @property @@ -96,4 +97,4 @@ class CodeInterpreter(Role): logger.info("Plan completed. Now start to make tools ...") tool_maker = ToolMaker() for task in self.planner.plan.get_finished_tasks(): - await tool_maker.make_tool(task.code, task.instruction, task.task_id) + await tool_maker.make_tool(code=task.code, instruction=task.instruction, task_id=task.task_id, auto_run=self.auto_run) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 639a517d6..cf903347d 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -12,7 +12,6 @@ from metagpt.utils.common import any_to_str class MLEngineer(CodeInterpreter): - auto_run: bool = False use_code_steps: bool = False use_udfs: bool = False data_desc: dict = {} diff --git a/metagpt/roles/tool_maker.py b/metagpt/roles/tool_maker.py index 7fec7b739..5453fd807 100644 --- a/metagpt/roles/tool_maker.py +++ b/metagpt/roles/tool_maker.py @@ -1,5 +1,6 @@ from pydantic import Field +from metagpt.actions.ask_review import AskReview from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.write_analysis_code import MakeTools from metagpt.logs import logger @@ -10,7 +11,7 @@ from metagpt.utils.common import remove_comments class ToolMaker(Role): execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) - async def make_tool(self, code: str, instruction: str, task_id: str = ""): + async def make_tool(self, code: str, instruction: str, task_id: str = "", auto_run=True): if len(remove_comments(code).split("\n")) < 5: # no need to consider trivial codes with fewer than 5 lines return @@ -41,4 +42,12 @@ class ToolMaker(Role): break # save successful tool code in udf if execute_success: - make_tools.save(tool_code) + _, confirmed = await self.ask_review(auto_run=auto_run) + if confirmed: + make_tools.save(tool_code) + + async def ask_review(self, auto_run: bool = True): + if not auto_run: + review, confirmed = await AskReview().run() + return review, confirmed + return "", True From 17aeb9f82591cf56d7e0b6fa71001b6a03470b3a Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 11 Jan 2024 14:15:33 +0800 Subject: [PATCH 287/637] formatting --- metagpt/actions/ask_review.py | 4 +++- metagpt/actions/write_analysis_code.py | 2 +- metagpt/roles/code_interpreter.py | 8 ++++++-- metagpt/roles/tool_maker.py | 2 +- 4 files changed, 11 insertions(+), 5 deletions(-) diff --git a/metagpt/actions/ask_review.py b/metagpt/actions/ask_review.py index 0d671648b..a20395104 100644 --- a/metagpt/actions/ask_review.py +++ b/metagpt/actions/ask_review.py @@ -27,7 +27,9 @@ class AskReview(Action): if plan: logger.info("Current overall plan:") logger.info( - "\n".join([f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks]) + "\n".join( + [f"{task.task_id}: {task.instruction}, is_finished: {task.is_finished}" for task in plan.tasks] + ) ) logger.info("most recent context:") diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index c5f9c9166..7d4597cf0 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -12,7 +12,7 @@ import yaml from tenacity import retry, stop_after_attempt, wait_fixed from metagpt.actions import Action -from metagpt.const import METAGPT_ROOT, TOOL_SCHEMA_PATH +from metagpt.const import TOOL_SCHEMA_PATH from metagpt.llm import LLM from metagpt.logs import logger from metagpt.prompts.ml_engineer import ( diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 9b13d8dcb..164c7cb12 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -28,7 +28,9 @@ class CodeInterpreter(Role): make_udfs=False, **kwargs, ): - super().__init__(name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools, make_udfs=make_udfs, **kwargs) + super().__init__( + name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools, make_udfs=make_udfs, **kwargs + ) self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run, use_tools=use_tools) @property @@ -97,4 +99,6 @@ class CodeInterpreter(Role): logger.info("Plan completed. Now start to make tools ...") tool_maker = ToolMaker() for task in self.planner.plan.get_finished_tasks(): - await tool_maker.make_tool(code=task.code, instruction=task.instruction, task_id=task.task_id, auto_run=self.auto_run) + await tool_maker.make_tool( + code=task.code, instruction=task.instruction, task_id=task.task_id, auto_run=self.auto_run + ) diff --git a/metagpt/roles/tool_maker.py b/metagpt/roles/tool_maker.py index 5453fd807..68d84b1e6 100644 --- a/metagpt/roles/tool_maker.py +++ b/metagpt/roles/tool_maker.py @@ -45,7 +45,7 @@ class ToolMaker(Role): _, confirmed = await self.ask_review(auto_run=auto_run) if confirmed: make_tools.save(tool_code) - + async def ask_review(self, auto_run: bool = True): if not auto_run: review, confirmed = await AskReview().run() From 9e0b9745beddd28188896b63ba35412563e83bb6 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 11 Jan 2024 14:22:23 +0800 Subject: [PATCH 288/637] default tool_config and module_name --- metagpt/actions/write_analysis_code.py | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 7d4597cf0..186a12063 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -180,6 +180,9 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): code_context = [remove_comments(task.code) for task in finished_tasks] code_context = "\n\n".join(code_context) + tool_catalog = {} + module_name = "" + if len(available_tools) > 0: available_tools = {k: v["description"] for k, v in available_tools.items()} @@ -191,10 +194,6 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): module_name = TASK_MODULE_MAP[task_type] - else: - tool_catalog = {} - module_name = "" - tools_instruction = TOOL_USAGE_PROMPT.format( special_prompt=special_prompt, module_name=module_name, tool_catalog=tool_catalog ) From e56caa6f5e842b4d0e33a13ff003720f412fb1be Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 11 Jan 2024 19:29:49 +0800 Subject: [PATCH 289/637] update --- examples/sd_tool_usage.py | 40 ++++++++++++++++++++++++++++++++++ metagpt/prompts/ml_engineer.py | 2 ++ 2 files changed, 42 insertions(+) create mode 100644 examples/sd_tool_usage.py diff --git a/examples/sd_tool_usage.py b/examples/sd_tool_usage.py new file mode 100644 index 000000000..59fddb85d --- /dev/null +++ b/examples/sd_tool_usage.py @@ -0,0 +1,40 @@ +# -*- coding: utf-8 -*- +# @Date : 1/11/2024 7:06 PM +# @Author : stellahong (stellahong@fuzhi.ai) +# @Desc : +import asyncio +from metagpt.const import METAGPT_ROOT +from metagpt.actions.write_analysis_code import WriteCodeWithTools +from metagpt.plan.planner import Planner +from metagpt.actions.execute_code import ExecutePyCode +from metagpt.roles.code_interpreter import CodeInterpreter + +sd_url = 'http://106.75.10.65:19094/sdapi/v1/txt2img' +requirement = f"i have a text2image tool, generate a girl image use it, sd_url={sd_url}" + +if __name__ == "__main__": + code_interpreter = CodeInterpreter(use_tools=True, goal=requirement) + asyncio.run(code_interpreter.run(requirement)) + # planner = Planner( + # goal="i have a sdt2i tool, generate a girl image use it, sd_url='http://106.75.10.65:19094/sdapi/v1/txt2img'", + # auto_run=True) + # asyncio.run(planner.update_plan()) + +# schema_path = METAGPT_ROOT / "metagpt/tools/functions/schemas" +# # +# prompt = "1girl, beautiful" +# planner = Planner( +# goal="i have a sdt2i tool, generate a girl image use it, sd_url='http://106.75.10.65:19094/sdapi/v1/txt2img'", +# auto_run=True) +# asyncio.run(planner.update_plan()) +# planner.plan.current_task.task_type = "sd" +# planner.plan.current_task.instruction = "Use the sdt2i tool with the provided API endpoint to generate the girl image." +# executor = ExecutePyCode() +# +# tool_context, code = asyncio.run(WriteCodeWithTools(schema_path=schema_path).run( +# context=f"task prompt: {prompt}", +# plan=planner.plan, +# column_info="", +# )) +# print(code) +# asyncio.run(executor.run(code)) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 13ee4db42..a5bb2af73 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -58,6 +58,7 @@ Please assign a task type to each task in the list below from the given categori - **data_preprocess**: Only for changing value inplace. - **model_train**: Only for training model. - **model_evaluate**: Only for evaluating model. +- **stable_diffusion**: Related to text2image, image2image using stable diffusion model. - **other**: Any tasks that do not fit into the previous categories, such as visualization, summarizing findings, etc. """ @@ -326,4 +327,5 @@ TASK_MODULE_MAP = { "data_preprocess": "metagpt.tools.functions.libs.data_preprocess", "feature_engineering": "metagpt.tools.functions.libs.feature_engineering", "udf": "metagpt.tools.functions.libs.udf", + "stable_diffusion": "metagpt.tools.sd_engine", } From a98edada1aaeb34528418d516709780d14bad122 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 11 Jan 2024 20:48:27 +0800 Subject: [PATCH 290/637] =?UTF-8?q?=E5=A2=9E=E5=8A=A0=E9=9D=9E=E5=BC=82?= =?UTF-8?q?=E6=AD=A5=E6=8E=A5=E5=8F=A3=20sd=E5=B7=A5=E5=85=B7yaml?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../functions/schemas/stable_diffusion.yml | 49 ++++++++++++++ metagpt/tools/sd_engine.py | 65 +++++++++++-------- 2 files changed, 88 insertions(+), 26 deletions(-) create mode 100644 metagpt/tools/functions/schemas/stable_diffusion.yml diff --git a/metagpt/tools/functions/schemas/stable_diffusion.yml b/metagpt/tools/functions/schemas/stable_diffusion.yml new file mode 100644 index 000000000..119449caa --- /dev/null +++ b/metagpt/tools/functions/schemas/stable_diffusion.yml @@ -0,0 +1,49 @@ +SDEngine: + type: class + description: "Generate image using stable diffusion model" + methods: + __init__: + description: "Initialize the SDEngine instance." + parameters: + properties: + sd_url: + type: str + description: "URL of the stable diffusion service." + + simple_run_t2i: + description: "Run the stable diffusion API for multiple prompts, calling the stable diffusion API to generate images." + parameters: + properties: + payload: + type: dict + description: "Dictionary of input parameters for the stable diffusion API." + auto_save: + type: bool + description: "Save generated images automatically." + required: + - prompts + construct_payload: + description: "Modify and set the API parameters for image generation." + parameters: + properties: + prompt: + type: str + description: "Text input for image generation." + required: + - prompt + returns: + payload: + type: dict + description: "Updated parameters for the stable diffusion API." + save: + description: "Save generated images to the output directory." + parameters: + properties: + imgs: + type: str + description: "Generated images." + save_name: + type: str + description: "Output image name. Default is empty." + required: + - imgs diff --git a/metagpt/tools/sd_engine.py b/metagpt/tools/sd_engine.py index c4d9d2df4..de2988d2a 100644 --- a/metagpt/tools/sd_engine.py +++ b/metagpt/tools/sd_engine.py @@ -2,13 +2,14 @@ # @Date : 2023/7/19 16:28 # @Author : stellahong (stellahong@deepwisdom.ai) # @Desc : -import asyncio import base64 import io import json from os.path import join from typing import List +import hashlib +import requests from aiohttp import ClientSession from PIL import Image, PngImagePlugin @@ -51,59 +52,70 @@ default_negative_prompt = "(easynegative:0.8),black, dark,Low resolution" class SDEngine: - def __init__(self): + def __init__(self, sd_url=""): # Initialize the SDEngine with configuration - self.sd_url = CONFIG.get("SD_URL") + self.sd_url = sd_url if sd_url else CONFIG.get("SD_URL") self.sd_t2i_url = f"{self.sd_url}{CONFIG.get('SD_T2I_API')}" # Define default payload settings for SD API self.payload = payload logger.info(self.sd_t2i_url) - + def construct_payload( - self, - prompt, - negtive_prompt=default_negative_prompt, - width=512, - height=512, - sd_model="galaxytimemachinesGTM_photoV20", + self, + prompt, + negtive_prompt=default_negative_prompt, + width=512, + height=512, + sd_model="galaxytimemachinesGTM_photoV20", ): # Configure the payload with provided inputs self.payload["prompt"] = prompt - self.payload["negtive_prompt"] = negtive_prompt + self.payload["negative_prompt"] = negtive_prompt self.payload["width"] = width self.payload["height"] = height self.payload["override_settings"]["sd_model_checkpoint"] = sd_model logger.info(f"call sd payload is {self.payload}") return self.payload - - def _save(self, imgs, save_name=""): + + def save(self, imgs, save_name=""): save_dir = CONFIG.workspace_path / SD_OUTPUT_FILE_REPO if not save_dir.exists(): save_dir.mkdir(parents=True, exist_ok=True) batch_decode_base64_to_image(imgs, str(save_dir), save_name=save_name) - - async def run_t2i(self, prompts: List): + + def simple_run_t2i(self, payload: dict, auto_save: bool = True): + with requests.Session() as session: + logger.debug(self.sd_t2i_url) + rsp = session.post(self.sd_t2i_url, json=payload, timeout=600) + + results = rsp.json()["images"] + if auto_save: + save_name = hashlib.sha256(payload["prompt"][:10].encode()).hexdigest()[:6] + self.save(results, save_name=f"output_{save_name}") + return results + + async def run_t2i(self, payloads: List): # Asynchronously run the SD API for multiple prompts session = ClientSession() - for payload_idx, payload in enumerate(prompts): + for payload_idx, payload in enumerate(payloads): results = await self.run(url=self.sd_t2i_url, payload=payload, session=session) - self._save(results, save_name=f"output_{payload_idx}") + self.save(results, save_name=f"output_{payload_idx}") await session.close() - + async def run(self, url, payload, session): # Perform the HTTP POST request to the SD API async with session.post(url, json=payload, timeout=600) as rsp: data = await rsp.read() - + rsp_json = json.loads(data) imgs = rsp_json["images"] logger.info(f"callback rsp json is {rsp_json.keys()}") return imgs - + async def run_i2i(self): # todo: 添加图生图接口调用 raise NotImplementedError - + async def run_sam(self): # todo:添加SAM接口调用 raise NotImplementedError @@ -125,9 +137,10 @@ def batch_decode_base64_to_image(imgs, save_dir="", save_name=""): if __name__ == "__main__": engine = SDEngine() - prompt = "pixel style, game design, a game interface should be minimalistic and intuitive with the score and high score displayed at the top. The snake and its food should be easily distinguishable. The game should have a simple color scheme, with a contrasting color for the snake and its food. Complete interface boundary" - + prompt = "1girl, beautiful" + prompt = "1boy, hansom" engine.construct_payload(prompt) - - event_loop = asyncio.get_event_loop() - event_loop.run_until_complete(engine.run_t2i(prompt)) + + engine.simple_run_t2i(engine.payload) + # event_loop = asyncio.get_event_loop() + # event_loop.run_until_complete(engine.run_t2i([engine.payload])) From af26fe06cf15ac39ee5c4dc8ebd404c79cde1a07 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 11 Jan 2024 21:50:35 +0800 Subject: [PATCH 291/637] update debug_code ut update sd_tool_usage example --- examples/sd_tool_usage.py | 41 +++++------------- metagpt/actions/debug_code.py | 29 ------------- metagpt/roles/ml_engineer.py | 1 - tests/conftest.py | 8 ++-- tests/metagpt/actions/test_debug_code.py | 54 ++++++++++++++++++++++++ 5 files changed, 68 insertions(+), 65 deletions(-) create mode 100644 tests/metagpt/actions/test_debug_code.py diff --git a/examples/sd_tool_usage.py b/examples/sd_tool_usage.py index 59fddb85d..82ee6a709 100644 --- a/examples/sd_tool_usage.py +++ b/examples/sd_tool_usage.py @@ -3,38 +3,17 @@ # @Author : stellahong (stellahong@fuzhi.ai) # @Desc : import asyncio -from metagpt.const import METAGPT_ROOT -from metagpt.actions.write_analysis_code import WriteCodeWithTools -from metagpt.plan.planner import Planner -from metagpt.actions.execute_code import ExecutePyCode + from metagpt.roles.code_interpreter import CodeInterpreter -sd_url = 'http://106.75.10.65:19094/sdapi/v1/txt2img' -requirement = f"i have a text2image tool, generate a girl image use it, sd_url={sd_url}" + +async def main(requirement: str = ""): + code_interpreter = CodeInterpreter(use_tools=True, goal=requirement) + await code_interpreter.run(requirement) + if __name__ == "__main__": - code_interpreter = CodeInterpreter(use_tools=True, goal=requirement) - asyncio.run(code_interpreter.run(requirement)) - # planner = Planner( - # goal="i have a sdt2i tool, generate a girl image use it, sd_url='http://106.75.10.65:19094/sdapi/v1/txt2img'", - # auto_run=True) - # asyncio.run(planner.update_plan()) - -# schema_path = METAGPT_ROOT / "metagpt/tools/functions/schemas" -# # -# prompt = "1girl, beautiful" -# planner = Planner( -# goal="i have a sdt2i tool, generate a girl image use it, sd_url='http://106.75.10.65:19094/sdapi/v1/txt2img'", -# auto_run=True) -# asyncio.run(planner.update_plan()) -# planner.plan.current_task.task_type = "sd" -# planner.plan.current_task.instruction = "Use the sdt2i tool with the provided API endpoint to generate the girl image." -# executor = ExecutePyCode() -# -# tool_context, code = asyncio.run(WriteCodeWithTools(schema_path=schema_path).run( -# context=f"task prompt: {prompt}", -# plan=planner.plan, -# column_info="", -# )) -# print(code) -# asyncio.run(executor.run(code)) + sd_url = 'http://106.75.10.65:19094' + requirement = f"I want to generate an image of a beautiful girl using the stable diffusion text2image tool, sd_url={sd_url}" + + asyncio.run(main(requirement)) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 26a84bcf2..74a188e9f 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -85,20 +85,14 @@ class DebugCode(BaseWriteAnalysisCode): async def run_reflection( self, - # goal, - # finished_code, - # finished_code_result, context: List[Message], code, runtime_result, ) -> dict: info = [] - # finished_code_and_result = finished_code + "\n [finished results]\n\n" + finished_code_result reflection_prompt = REFLECTION_PROMPT.format( debug_example=DEBUG_REFLECTION_EXAMPLE, context=context, - # goal=goal, - # finished_code=finished_code_and_result, code=code, runtime_result=runtime_result, ) @@ -106,33 +100,14 @@ class DebugCode(BaseWriteAnalysisCode): info.append(Message(role="system", content=system_prompt)) info.append(Message(role="user", content=reflection_prompt)) - # msg = messages_to_str(info) - # resp = await self.llm.aask(msg=msg) resp = await self.llm.aask_code(messages=info, **create_func_config(CODE_REFLECTION)) logger.info(f"reflection is {resp}") return resp - # async def rewrite_code(self, reflection: str = "", context: List[Message] = None) -> str: - # """ - # 根据reflection重写代码 - # """ - # info = context - # # info.append(Message(role="assistant", content=f"[code context]:{code_context}" - # # f"finished code are executable, and you should based on the code to continue your current code debug and improvement" - # # f"[reflection]: \n {reflection}")) - # info.append(Message(role="assistant", content=f"[reflection]: \n {reflection}")) - # info.append(Message(role="user", content=f"[improved impl]:\n Return in Python block")) - # msg = messages_to_str(info) - # resp = await self.llm.aask(msg=msg) - # improv_code = CodeParser.parse_code(block=None, text=resp) - # return improv_code async def run( self, context: List[Message] = None, - plan: str = "", - # finished_code: str = "", - # finished_code_result: str = "", code: str = "", runtime_result: str = "", ) -> str: @@ -140,14 +115,10 @@ class DebugCode(BaseWriteAnalysisCode): 根据当前运行代码和报错信息进行reflection和纠错 """ reflection = await self.run_reflection( - # plan, - # finished_code=finished_code, - # finished_code_result=finished_code_result, code=code, context=context, runtime_result=runtime_result, ) # 根据reflection结果重写代码 - # improv_code = await self.rewrite_code(reflection, context=context) improv_code = reflection["improved_impl"] return improv_code diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index cf903347d..a60642bff 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -60,7 +60,6 @@ class MLEngineer(CodeInterpreter): if code_execution_count > 0: logger.warning("We got a bug code, now start to debug...") code = await DebugCode().run( - plan=self.planner.current_task.instruction, code=self.latest_code, runtime_result=self.working_memory.get(), context=self.debug_context, diff --git a/tests/conftest.py b/tests/conftest.py index 6f5c04f06..dc89e897f 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -34,14 +34,14 @@ def rsp_cache(): rsp_cache_file_path = TEST_DATA_PATH / "rsp_cache.json" # read repo-provided new_rsp_cache_file_path = TEST_DATA_PATH / "rsp_cache_new.json" # exporting a new copy if os.path.exists(rsp_cache_file_path): - with open(rsp_cache_file_path, "r") as f1: + with open(rsp_cache_file_path, "r", encoding="utf-8") as f1: rsp_cache_json = json.load(f1) else: rsp_cache_json = {} yield rsp_cache_json - with open(rsp_cache_file_path, "w") as f2: + with open(rsp_cache_file_path, "w", encoding="utf-8") as f2: json.dump(rsp_cache_json, f2, indent=4, ensure_ascii=False) - with open(new_rsp_cache_file_path, "w") as f2: + with open(new_rsp_cache_file_path, "w", encoding="utf-8") as f2: json.dump(RSP_CACHE_NEW, f2, indent=4, ensure_ascii=False) @@ -139,7 +139,7 @@ def loguru_caplog(caplog): # init & dispose git repo -@pytest.fixture(scope="function", autouse=True) +@pytest.fixture(scope="function", autouse=False) def setup_and_teardown_git_repo(request): CONFIG.git_repo = GitRepository(local_path=DEFAULT_WORKSPACE_ROOT / f"unittest/{uuid.uuid4().hex}") CONFIG.git_reinit = True diff --git a/tests/metagpt/actions/test_debug_code.py b/tests/metagpt/actions/test_debug_code.py new file mode 100644 index 000000000..675c07f78 --- /dev/null +++ b/tests/metagpt/actions/test_debug_code.py @@ -0,0 +1,54 @@ +# -*- coding: utf-8 -*- +# @Date : 1/11/2024 8:51 PM +# @Author : stellahong (stellahong@fuzhi.ai) +# @Desc : + +import pytest + +from metagpt.actions.debug_code import DebugCode, messages_to_str +from metagpt.schema import Message + +ErrorStr = '''Tested passed: + +Tests failed: +assert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5] +''' + +CODE = ''' +def sort_array(arr): + # Helper function to count the number of ones in the binary representation + def count_ones(n): + return bin(n).count('1') + + # Sort the array using a custom key function + # The key function returns a tuple (number of ones, value) for each element + # This ensures that if two elements have the same number of ones, they are sorted by their value + sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x)) + + return sorted_arr +``` +''' + +DebugContext = '''Solve the problem in Python: +def sort_array(arr): + """ + In this Kata, you have to sort an array of non-negative integers according to + number of ones in their binary representation in ascending order. + For similar number of ones, sort based on decimal value. + + It must be implemented like this: + >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] + >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2] + >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4] + """ +''' +@pytest.mark.asyncio +async def test_debug_code(): + debug_context = Message(content=DebugContext) + new_code = await DebugCode().run(context=debug_context, code=CODE, runtime_result=ErrorStr) + assert "def sort_array(arr)" in new_code + +def test_messages_to_str(): + debug_context = Message(content=DebugContext) + msg_str = messages_to_str([debug_context]) + assert "user: Solve the problem in Python" in msg_str \ No newline at end of file From 3be26cf94f5b4827ee97f47c3904ebd35158cde1 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 11 Jan 2024 21:56:38 +0800 Subject: [PATCH 292/637] add sd ut --- .gitignore | 9 +++++++++ tests/metagpt/tools/functions/test_sd.py | 17 +++++++++++++++++ 2 files changed, 26 insertions(+) create mode 100644 tests/metagpt/tools/functions/test_sd.py diff --git a/.gitignore b/.gitignore index b5dafc3fc..0a78c3d58 100644 --- a/.gitignore +++ b/.gitignore @@ -178,3 +178,12 @@ htmlcov.* *-structure.csv *-structure.json +/Titanic/2023_12_07_11_44_319a116fff/LLM_inout_pair/*.json +/ICR/2023_12_06_14_14_26e593d09f/LLM_inout_pair/*.json +/ICR/5cd9acb669c443fabe763e8f1ade5e86/workspace/*.txt +/ICR/5cd9acb669c443fabe763e8f1ade5e86/workspace/*.csv +/Titanic/9530b3c5550a4366ae92e5af6a74e6c3/workspace/*.csv +/Titanic/9530b3c5550a4366ae92e5af6a74e6c3/workspace/*.txt +/metagpt/roles/catboost_info/*.tsv +/metagpt/roles/catboost_info/*.json +/Titanic/9530b3c5550a4366ae92e5af6a74e6c3/workspace/*.md diff --git a/tests/metagpt/tools/functions/test_sd.py b/tests/metagpt/tools/functions/test_sd.py new file mode 100644 index 000000000..405ac9a32 --- /dev/null +++ b/tests/metagpt/tools/functions/test_sd.py @@ -0,0 +1,17 @@ +# -*- coding: utf-8 -*- +# @Date : 1/10/2024 10:07 PM +# @Author : stellahong (stellahong@fuzhi.ai) +# @Desc : +from metagpt.tools.sd_engine import SDEngine + +def test_sd_tools(): + engine = SDEngine() + prompt = "1boy, hansom" + engine.construct_payload(prompt) + engine.simple_run_t2i(engine.payload) + +def test_sd_construct_payload(): + engine = SDEngine() + prompt = "1boy, hansom" + engine.construct_payload(prompt) + assert "negative_prompt" in engine.payload \ No newline at end of file From 12bc0104b6c4030eeea51dd8be01297087395b87 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Thu, 11 Jan 2024 22:43:02 +0800 Subject: [PATCH 293/637] add asyn sd ut --- metagpt/tools/sd_engine.py | 43 +++++++++--------------- tests/metagpt/tools/functions/test_sd.py | 17 ++++++++-- 2 files changed, 31 insertions(+), 29 deletions(-) diff --git a/metagpt/tools/sd_engine.py b/metagpt/tools/sd_engine.py index de2988d2a..ba61fd496 100644 --- a/metagpt/tools/sd_engine.py +++ b/metagpt/tools/sd_engine.py @@ -3,11 +3,11 @@ # @Author : stellahong (stellahong@deepwisdom.ai) # @Desc : import base64 +import hashlib import io import json from os.path import join from typing import List -import hashlib import requests from aiohttp import ClientSession @@ -59,14 +59,14 @@ class SDEngine: # Define default payload settings for SD API self.payload = payload logger.info(self.sd_t2i_url) - + def construct_payload( - self, - prompt, - negtive_prompt=default_negative_prompt, - width=512, - height=512, - sd_model="galaxytimemachinesGTM_photoV20", + self, + prompt, + negtive_prompt=default_negative_prompt, + width=512, + height=512, + sd_model="galaxytimemachinesGTM_photoV20", ): # Configure the payload with provided inputs self.payload["prompt"] = prompt @@ -76,24 +76,24 @@ class SDEngine: self.payload["override_settings"]["sd_model_checkpoint"] = sd_model logger.info(f"call sd payload is {self.payload}") return self.payload - + def save(self, imgs, save_name=""): save_dir = CONFIG.workspace_path / SD_OUTPUT_FILE_REPO if not save_dir.exists(): save_dir.mkdir(parents=True, exist_ok=True) batch_decode_base64_to_image(imgs, str(save_dir), save_name=save_name) - + def simple_run_t2i(self, payload: dict, auto_save: bool = True): with requests.Session() as session: logger.debug(self.sd_t2i_url) rsp = session.post(self.sd_t2i_url, json=payload, timeout=600) - + results = rsp.json()["images"] if auto_save: save_name = hashlib.sha256(payload["prompt"][:10].encode()).hexdigest()[:6] self.save(results, save_name=f"output_{save_name}") return results - + async def run_t2i(self, payloads: List): # Asynchronously run the SD API for multiple prompts session = ClientSession() @@ -101,21 +101,21 @@ class SDEngine: results = await self.run(url=self.sd_t2i_url, payload=payload, session=session) self.save(results, save_name=f"output_{payload_idx}") await session.close() - + async def run(self, url, payload, session): # Perform the HTTP POST request to the SD API async with session.post(url, json=payload, timeout=600) as rsp: data = await rsp.read() - + rsp_json = json.loads(data) imgs = rsp_json["images"] logger.info(f"callback rsp json is {rsp_json.keys()}") return imgs - + async def run_i2i(self): # todo: 添加图生图接口调用 raise NotImplementedError - + async def run_sam(self): # todo:添加SAM接口调用 raise NotImplementedError @@ -133,14 +133,3 @@ def batch_decode_base64_to_image(imgs, save_dir="", save_name=""): for idx, _img in enumerate(imgs): save_name = join(save_dir, save_name) decode_base64_to_image(_img, save_name=save_name) - - -if __name__ == "__main__": - engine = SDEngine() - prompt = "1girl, beautiful" - prompt = "1boy, hansom" - engine.construct_payload(prompt) - - engine.simple_run_t2i(engine.payload) - # event_loop = asyncio.get_event_loop() - # event_loop.run_until_complete(engine.run_t2i([engine.payload])) diff --git a/tests/metagpt/tools/functions/test_sd.py b/tests/metagpt/tools/functions/test_sd.py index 405ac9a32..142101cad 100644 --- a/tests/metagpt/tools/functions/test_sd.py +++ b/tests/metagpt/tools/functions/test_sd.py @@ -2,16 +2,29 @@ # @Date : 1/10/2024 10:07 PM # @Author : stellahong (stellahong@fuzhi.ai) # @Desc : +import pytest + from metagpt.tools.sd_engine import SDEngine + def test_sd_tools(): engine = SDEngine() prompt = "1boy, hansom" engine.construct_payload(prompt) engine.simple_run_t2i(engine.payload) - + + def test_sd_construct_payload(): engine = SDEngine() prompt = "1boy, hansom" engine.construct_payload(prompt) - assert "negative_prompt" in engine.payload \ No newline at end of file + assert "negative_prompt" in engine.payload + + +@pytest.mark.asyncio +async def test_sd_asyn_t2i(): + engine = SDEngine() + prompt = "1boy, hansom" + engine.construct_payload(prompt) + await engine.run_t2i([engine.payload]) + assert "negative_prompt" in engine.payload From e99c5f29f4e8c108355cc4d19cbc531789fcba12 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 11 Jan 2024 22:55:31 +0800 Subject: [PATCH 294/637] tool management at one place, add aask_code mock, azure mock --- metagpt/actions/write_analysis_code.py | 28 +++++----- metagpt/actions/write_plan.py | 6 ++- metagpt/prompts/ml_engineer.py | 55 +------------------- metagpt/prompts/tool_type.py | 35 +++++++++++++ metagpt/tools/__init__.py | 51 ++++++++++++++++++ tests/conftest.py | 9 ++-- tests/metagpt/actions/test_write_plan.py | 19 ++++++- tests/metagpt/roles/test_code_interpreter.py | 13 +++++ tests/mock/mock_llm.py | 25 ++++++++- 9 files changed, 167 insertions(+), 74 deletions(-) create mode 100644 metagpt/prompts/tool_type.py create mode 100644 tests/metagpt/roles/test_code_interpreter.py diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 186a12063..04cad34a5 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -20,14 +20,16 @@ from metagpt.prompts.ml_engineer import ( GENERATE_CODE_PROMPT, ML_TOOL_USAGE_PROMPT, SELECT_FUNCTION_TOOLS, - TASK_MODULE_MAP, - TASK_SPECIFIC_PROMPT, TOOL_RECOMMENDATION_PROMPT, TOOL_USAGE_PROMPT, ) from metagpt.schema import Message, Plan +from metagpt.tools import TOOL_TYPE_MAPPINGS from metagpt.utils.common import create_func_config, remove_comments +TOOL_TYPE_MODULE = {k: v.module for k, v in TOOL_TYPE_MAPPINGS.items()} +TOOL_TYPE_USAGE_PROMPT = {k: v.usage_prompt for k, v in TOOL_TYPE_MAPPINGS.items()} + class BaseWriteAnalysisCode(Action): DEFAULT_SYSTEM_MSG: str = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt @@ -171,9 +173,11 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): plan: Plan = None, **kwargs, ) -> str: - task_type = plan.current_task.task_type - available_tools = self.available_tools.get(task_type, {}) - special_prompt = TASK_SPECIFIC_PROMPT.get(task_type, "") + tool_type = ( + plan.current_task.task_type + ) # find tool type from task type through exact match, can extend to retrieval in the future + available_tools = self.available_tools.get(tool_type, {}) + special_prompt = TOOL_TYPE_USAGE_PROMPT.get(tool_type, "") code_steps = plan.current_task.code_steps finished_tasks = plan.get_finished_tasks() @@ -189,10 +193,10 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): recommend_tools = await self._tool_recommendation( plan.current_task.instruction, code_steps, available_tools ) - tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) + tool_catalog = self._parse_recommend_tools(tool_type, recommend_tools) logger.info(f"Recommended tools: \n{recommend_tools}") - module_name = TASK_MODULE_MAP[task_type] + module_name = TOOL_TYPE_MODULE[tool_type] tools_instruction = TOOL_USAGE_PROMPT.format( special_prompt=special_prompt, module_name=module_name, tool_catalog=tool_catalog @@ -215,9 +219,9 @@ class WriteCodeWithToolsML(WriteCodeWithTools): column_info: str = "", **kwargs, ) -> Tuple[List[Message], str]: - task_type = plan.current_task.task_type - available_tools = self.available_tools.get(task_type, {}) - special_prompt = TASK_SPECIFIC_PROMPT.get(task_type, "") + tool_type = plan.current_task.task_type + available_tools = self.available_tools.get(tool_type, {}) + special_prompt = TOOL_TYPE_USAGE_PROMPT.get(tool_type, "") code_steps = plan.current_task.code_steps finished_tasks = plan.get_finished_tasks() @@ -230,10 +234,10 @@ class WriteCodeWithToolsML(WriteCodeWithTools): recommend_tools = await self._tool_recommendation( plan.current_task.instruction, code_steps, available_tools ) - tool_catalog = self._parse_recommend_tools(task_type, recommend_tools) + tool_catalog = self._parse_recommend_tools(tool_type, recommend_tools) logger.info(f"Recommended tools: \n{recommend_tools}") - module_name = TASK_MODULE_MAP[task_type] + module_name = TOOL_TYPE_MODULE[tool_type] prompt = ML_TOOL_USAGE_PROMPT.format( user_requirement=plan.goal, diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index 16680e395..c7ef541b9 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -12,6 +12,7 @@ from metagpt.actions import Action from metagpt.logs import logger from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_CONFIG, ASSIGN_TASK_TYPE_PROMPT from metagpt.schema import Message, Plan, Task +from metagpt.tools import TOOL_TYPE_MAPPINGS from metagpt.utils.common import CodeParser, create_func_config @@ -46,7 +47,10 @@ class WritePlan(Action): List[Dict]: tasks with task type assigned """ task_list = "\n".join([f"Task {task['task_id']}: {task['instruction']}" for task in tasks]) - prompt = ASSIGN_TASK_TYPE_PROMPT.format(task_list=task_list) + task_type_desc = "\n".join([f"- **{item.name}**: {item.desc}" for item in TOOL_TYPE_MAPPINGS.values()]) + prompt = ASSIGN_TASK_TYPE_PROMPT.format( + task_list=task_list, task_type_desc=task_type_desc + ) # task types are set to be the same as tool types, for now tool_config = create_func_config(ASSIGN_TASK_TYPE_CONFIG) rsp = await self.llm.aask_code(prompt, **tool_config) task_type_list = rsp["task_type"] diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 13ee4db42..3baf79843 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -54,11 +54,7 @@ Please assign a task type to each task in the list below from the given categori {task_list} ## All Task Type: -- **feature_engineering**: Only for creating new columns for input data. -- **data_preprocess**: Only for changing value inplace. -- **model_train**: Only for training model. -- **model_evaluate**: Only for evaluating model. -- **other**: Any tasks that do not fit into the previous categories, such as visualization, summarizing findings, etc. +{task_type_desc} """ ASSIGN_TASK_TYPE_CONFIG = { @@ -278,52 +274,3 @@ for col in num_cols: - The output code should contain all steps implemented correctly in 'Code Steps'. """ # - If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. - -DATA_PREPROCESS_PROMPT = """ -The current task is about data preprocessing, please note the following: -- Monitor data types per column, applying appropriate methods. -- Ensure operations are on existing dataset columns. -- Avoid writing processed data to files. -- Avoid any change to label column, such as standardization, etc. -- Prefer alternatives to one-hot encoding for categorical data. -- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later. -- Each step do data preprocessing to train, must do same for test separately at the same time. -""" - -FEATURE_ENGINEERING_PROMPT = """ -The current task is about feature engineering. when performing it, please adhere to the following principles: -- Generate as diverse features as possible to improve the model's performance step-by-step. -- If potential impactful features are not included in 'Code Steps', add new steps to generate them. -- Avoid creating redundant or excessively numerous features in one step. -- Exclude ID columns from feature generation and remove them. -- Each step do feature engineering to train, must do same for test separately at the same time. -- Avoid using the label column to create features, except for cat encoding. -- Use the data from previous task result if exist, do not mock or reload data yourself. -""" - -MODEL_TRAIN_PROMPT = """ -The current task is about training a model, please ensure high performance: -- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as lightGBM, XGBoost, CatBoost, etc. -- If non-numeric columns exist, perform label encode together with all steps. -- Use the data from previous task result directly, do not mock or reload data yourself. -- Set suitable hyperparameters for the model, make metrics as high as possible. -""" - -MODEL_EVALUATE_PROMPT = """ -The current task is about evaluating a model, please note the following: -- Ensure that the evaluated data is same processed as the training data. If not, remember use object in 'Done Tasks' to transform the data. -- Use trained model from previous task result directly, do not mock or reload model yourself. -""" - -TASK_SPECIFIC_PROMPT = { - "data_preprocess": DATA_PREPROCESS_PROMPT, - "feature_engineering": FEATURE_ENGINEERING_PROMPT, - "model_train": MODEL_TRAIN_PROMPT, - "model_evaluate": MODEL_EVALUATE_PROMPT, -} - -TASK_MODULE_MAP = { - "data_preprocess": "metagpt.tools.functions.libs.data_preprocess", - "feature_engineering": "metagpt.tools.functions.libs.feature_engineering", - "udf": "metagpt.tools.functions.libs.udf", -} diff --git a/metagpt/prompts/tool_type.py b/metagpt/prompts/tool_type.py new file mode 100644 index 000000000..25cb1431e --- /dev/null +++ b/metagpt/prompts/tool_type.py @@ -0,0 +1,35 @@ +DATA_PREPROCESS_PROMPT = """ +The current task is about data preprocessing, please note the following: +- Monitor data types per column, applying appropriate methods. +- Ensure operations are on existing dataset columns. +- Avoid writing processed data to files. +- Avoid any change to label column, such as standardization, etc. +- Prefer alternatives to one-hot encoding for categorical data. +- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later. +- Each step do data preprocessing to train, must do same for test separately at the same time. +""" + +FEATURE_ENGINEERING_PROMPT = """ +The current task is about feature engineering. when performing it, please adhere to the following principles: +- Generate as diverse features as possible to improve the model's performance step-by-step. +- If potential impactful features are not included in 'Code Steps', add new steps to generate them. +- Avoid creating redundant or excessively numerous features in one step. +- Exclude ID columns from feature generation and remove them. +- Each step do feature engineering to train, must do same for test separately at the same time. +- Avoid using the label column to create features, except for cat encoding. +- Use the data from previous task result if exist, do not mock or reload data yourself. +""" + +MODEL_TRAIN_PROMPT = """ +The current task is about training a model, please ensure high performance: +- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as lightGBM, XGBoost, CatBoost, etc. +- If non-numeric columns exist, perform label encode together with all steps. +- Use the data from previous task result directly, do not mock or reload data yourself. +- Set suitable hyperparameters for the model, make metrics as high as possible. +""" + +MODEL_EVALUATE_PROMPT = """ +The current task is about evaluating a model, please note the following: +- Ensure that the evaluated data is same processed as the training data. If not, remember use object in 'Done Tasks' to transform the data. +- Use trained model from previous task result directly, do not mock or reload model yourself. +""" diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index aab8c990c..543a2b8bb 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -9,6 +9,16 @@ from enum import Enum +from pydantic import BaseModel + +from metagpt.const import TOOL_SCHEMA_PATH +from metagpt.prompts.tool_type import ( + DATA_PREPROCESS_PROMPT, + FEATURE_ENGINEERING_PROMPT, + MODEL_TRAIN_PROMPT, + MODEL_EVALUATE_PROMPT, +) + class SearchEngineType(Enum): SERPAPI_GOOGLE = "serpapi" @@ -27,3 +37,44 @@ class WebBrowserEngineType(Enum): def __missing__(cls, key): """Default type conversion""" return cls.CUSTOM + + +class ToolType(BaseModel): + name: str + module: str = "" + desc: str + usage_prompt: str = "" + + +TOOL_TYPE_MAPPINGS = { + "data_preprocess": ToolType( + name="data_preprocess", + module=str(TOOL_SCHEMA_PATH / "data_preprocess"), + desc="Only for changing value inplace.", + usage_prompt=DATA_PREPROCESS_PROMPT, + ), + "feature_engineering": ToolType( + name="feature_engineering", + module=str(TOOL_SCHEMA_PATH / "feature_engineering"), + desc="Only for creating new columns for input data.", + usage_prompt=FEATURE_ENGINEERING_PROMPT, + ), + "model_train": ToolType( + name="model_train", + module="", + desc="Only for training model.", + usage_prompt=MODEL_TRAIN_PROMPT, + ), + "model_evaluate": ToolType( + name="model_evaluate", + module="", + desc="Only for evaluating model.", + usage_prompt=MODEL_EVALUATE_PROMPT, + ), + "other": ToolType( + name="other", + module="", + desc="Any tasks that do not fit into the previous categories", + usage_prompt="", + ), +} diff --git a/tests/conftest.py b/tests/conftest.py index 6f5c04f06..7dec506bb 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -34,14 +34,14 @@ def rsp_cache(): rsp_cache_file_path = TEST_DATA_PATH / "rsp_cache.json" # read repo-provided new_rsp_cache_file_path = TEST_DATA_PATH / "rsp_cache_new.json" # exporting a new copy if os.path.exists(rsp_cache_file_path): - with open(rsp_cache_file_path, "r") as f1: + with open(rsp_cache_file_path, "r", encoding="utf-8") as f1: rsp_cache_json = json.load(f1) else: rsp_cache_json = {} yield rsp_cache_json - with open(rsp_cache_file_path, "w") as f2: + with open(rsp_cache_file_path, "w", encoding="utf-8") as f2: json.dump(rsp_cache_json, f2, indent=4, ensure_ascii=False) - with open(new_rsp_cache_file_path, "w") as f2: + with open(new_rsp_cache_file_path, "w", encoding="utf-8") as f2: json.dump(RSP_CACHE_NEW, f2, indent=4, ensure_ascii=False) @@ -60,6 +60,7 @@ def llm_mock(rsp_cache, mocker, request): llm.rsp_cache = rsp_cache mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", llm.aask) mocker.patch("metagpt.provider.base_llm.BaseLLM.aask_batch", llm.aask_batch) + mocker.patch("metagpt.provider.openai_api.OpenAILLM.aask_code", llm.aask_code) yield mocker if hasattr(request.node, "test_outcome") and request.node.test_outcome.passed: if llm.rsp_candidates: @@ -67,7 +68,7 @@ def llm_mock(rsp_cache, mocker, request): cand_key = list(rsp_candidate.keys())[0] cand_value = list(rsp_candidate.values())[0] if cand_key not in llm.rsp_cache: - logger.info(f"Added '{cand_key[:100]} ... -> {cand_value[:20]} ...' to response cache") + logger.info(f"Added '{cand_key[:100]} ... -> {str(cand_value)[:20]} ...' to response cache") llm.rsp_cache.update(rsp_candidate) RSP_CACHE_NEW.update(rsp_candidate) diff --git a/tests/metagpt/actions/test_write_plan.py b/tests/metagpt/actions/test_write_plan.py index e1c93e8b2..9abc6c798 100644 --- a/tests/metagpt/actions/test_write_plan.py +++ b/tests/metagpt/actions/test_write_plan.py @@ -1,4 +1,12 @@ -from metagpt.actions.write_plan import Plan, Task, precheck_update_plan_from_rsp +import pytest + +from metagpt.actions.write_plan import ( + Plan, + Task, + WritePlan, + precheck_update_plan_from_rsp, +) +from metagpt.schema import Message def test_precheck_update_plan_from_rsp(): @@ -12,3 +20,12 @@ def test_precheck_update_plan_from_rsp(): invalid_rsp = "wrong" success, _ = precheck_update_plan_from_rsp(invalid_rsp, plan) assert not success + + +@pytest.mark.asyncio +async def test_write_plan(): + rsp = await WritePlan().run(context=[Message("run analysis on sklearn iris dataset", role="user")]) + + assert "task_id" in rsp + assert "instruction" in rsp + assert "json" not in rsp # the output should be the content inside ```json ``` diff --git a/tests/metagpt/roles/test_code_interpreter.py b/tests/metagpt/roles/test_code_interpreter.py new file mode 100644 index 000000000..8595b9b15 --- /dev/null +++ b/tests/metagpt/roles/test_code_interpreter.py @@ -0,0 +1,13 @@ +import pytest + +from metagpt.logs import logger +from metagpt.roles.code_interpreter import CodeInterpreter + + +@pytest.mark.asyncio +async def test_code_interpreter(): + requirement = "Run data analysis on sklearn Iris dataset, include a plot" + ci = CodeInterpreter(goal=requirement, auto_run=True, use_tools=False) + rsp = await ci.run(requirement) + logger.info(rsp) + assert len(rsp.content) > 0 diff --git a/tests/mock/mock_llm.py b/tests/mock/mock_llm.py index 6e7a1cdd5..45b28c63b 100644 --- a/tests/mock/mock_llm.py +++ b/tests/mock/mock_llm.py @@ -1,10 +1,16 @@ -from typing import Optional +import json +from typing import Optional, Union +from metagpt.config import CONFIG from metagpt.logs import log_llm_stream, logger +from metagpt.provider.azure_openai_api import AzureOpenAILLM from metagpt.provider.openai_api import OpenAILLM +from metagpt.schema import Message + +OriginalLLM = OpenAILLM if not CONFIG.openai_api_type else AzureOpenAILLM -class MockLLM(OpenAILLM): +class MockLLM(OriginalLLM): def __init__(self, allow_open_api_call): super().__init__() self.allow_open_api_call = allow_open_api_call @@ -58,6 +64,15 @@ class MockLLM(OpenAILLM): context.append(self._assistant_msg(rsp_text)) return self._extract_assistant_rsp(context) + async def original_aask_code(self, messages: Union[str, Message, list[dict]], **kwargs) -> dict: + """ + A copy of metagpt.provider.openai_api.OpenAILLM.aask_code, we can't use super().aask because it will be mocked. + Since openai_api.OpenAILLM.aask_code is different from base_llm.BaseLLM.aask_code, we use the former. + """ + messages = self._process_message(messages) + rsp = await self._achat_completion_function(messages, **kwargs) + return self.get_choice_function_arguments(rsp) + async def aask( self, msg: str, @@ -78,6 +93,12 @@ class MockLLM(OpenAILLM): rsp = await self._mock_rsp(msg_key, self.original_aask_batch, msgs, timeout) return rsp + async def aask_code(self, messages: Union[str, Message, list[dict]], **kwargs) -> dict: + messages = self._process_message(messages) + msg_key = json.dumps(messages, ensure_ascii=False) + rsp = await self._mock_rsp(msg_key, self.original_aask_code, messages, **kwargs) + return rsp + async def _mock_rsp(self, msg_key, ask_func, *args, **kwargs): if msg_key not in self.rsp_cache: if not self.allow_open_api_call: From 39915ec2bac1280c375745f044266e06d4b211ee Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 12 Jan 2024 10:50:35 +0800 Subject: [PATCH 295/637] add comments to clarify tool_type prompts --- metagpt/prompts/tool_type.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/metagpt/prompts/tool_type.py b/metagpt/prompts/tool_type.py index 25cb1431e..ec848bbe4 100644 --- a/metagpt/prompts/tool_type.py +++ b/metagpt/prompts/tool_type.py @@ -1,3 +1,4 @@ +# Prompt for using tools of "data_preprocess" type DATA_PREPROCESS_PROMPT = """ The current task is about data preprocessing, please note the following: - Monitor data types per column, applying appropriate methods. @@ -9,6 +10,7 @@ The current task is about data preprocessing, please note the following: - Each step do data preprocessing to train, must do same for test separately at the same time. """ +# Prompt for using tools of "feature_engineering" type FEATURE_ENGINEERING_PROMPT = """ The current task is about feature engineering. when performing it, please adhere to the following principles: - Generate as diverse features as possible to improve the model's performance step-by-step. @@ -20,6 +22,7 @@ The current task is about feature engineering. when performing it, please adhere - Use the data from previous task result if exist, do not mock or reload data yourself. """ +# Prompt for using tools of "model_train" type MODEL_TRAIN_PROMPT = """ The current task is about training a model, please ensure high performance: - Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as lightGBM, XGBoost, CatBoost, etc. @@ -28,6 +31,7 @@ The current task is about training a model, please ensure high performance: - Set suitable hyperparameters for the model, make metrics as high as possible. """ +# Prompt for using tools of "model_evaluate" type MODEL_EVALUATE_PROMPT = """ The current task is about evaluating a model, please note the following: - Ensure that the evaluated data is same processed as the training data. If not, remember use object in 'Done Tasks' to transform the data. From 9946280c9e96e6efc0435fcb3fb4b08c8b17c277 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 12 Jan 2024 14:56:31 +0800 Subject: [PATCH 296/637] update locally --- tests/data/rsp_cache.json | 145 -------------------------------------- 1 file changed, 145 deletions(-) delete mode 100644 tests/data/rsp_cache.json diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json deleted file mode 100644 index db452f676..000000000 --- a/tests/data/rsp_cache.json +++ /dev/null @@ -1,145 +0,0 @@ -{ - "\n## context\n\n### Project Name\n\n\n### Original Requirements\n['需要一个基于LLM做总结的搜索引擎']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"zh_cn\",\n \"Programming Language\": \"\",\n \"Original Requirements\": \"需要一个基于LLM做总结的搜索引擎\",\n \"Project Name\": \"search_engine_llm\",\n \"Product Goals\": [\n \"提供基于LLM的搜索功能\",\n \"提高搜索结果的准确性和相关性\",\n \"提供用户友好的搜索界面\"\n ],\n \"User Stories\": [\n \"作为用户,我希望能够通过关键词搜索到相关的结果\",\n \"作为用户,我希望搜索结果能够按照相关性排序\",\n \"作为用户,我希望搜索界面简洁明了,易于使用\"\n ],\n \"Competitive Analysis\": [\n \"百度搜索引擎:提供全面的搜索功能,但结果可能不够准确\",\n \"谷歌搜索引擎:提供准确的搜索结果,但在中国访问速度较慢\",\n \"搜狗搜索引擎:提供快速的搜索结果,但广告较多\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"搜索引擎的准确性和速度\\\"\\n x-axis \\\"准确性低\\\" --> \\\"准确性高\\\"\\n y-axis \\\"速度慢\\\" --> \\\"速度快\\\"\\n quadrant-1 \\\"需要改进\\\"\\n quadrant-2 \\\"需要提高速度\\\"\\n quadrant-3 \\\"需要提高准确性\\\"\\n quadrant-4 \\\"目标产品\\\"\\n \\\"百度搜索引擎\\\": [0.3, 0.6]\\n \\\"谷歌搜索引擎\\\": [0.45, 0.23]\\n \\\"搜狗搜索引擎\\\": [0.57, 0.69]\\n \\\"目标产品\\\": [0.8, 0.8]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"基于LLM算法实现搜索功能\"\n ],\n [\n \"P0\",\n \"提高搜索结果的准确性和相关性\"\n ]\n ],\n \"UI Design draft\": \"搜索界面设计简洁明了,提供关键词搜索框和搜索结果展示区域。\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", - "hello chatgpt": "Hello! How can I assist you today?", - "hello world": "Hello! How can I assist you today?", - "\n## context\n```\nclass UIDesign(Action):\n #Class representing the UI Design action.\n def __init__(self, name, context=None, llm=None):\n super().__init__(name, context, llm) # 需要调用LLM进一步丰富UI设计的prompt\n @parse\n def parse_requirement(self, context: str):\n #Parse UI Design draft from the context using regex.\n pattern = r\"## UI Design draft.*?\n(.*?)## Anything UNCLEAR\"\n return context, pattern\n @parse\n def parse_ui_elements(self, context: str):\n #Parse Selected Elements from the context using regex.\n pattern = r\"## Selected Elements.*?\n(.*?)## HTML Layout\"\n return context, pattern\n @parse\n def parse_css_code(self, context: str):\n pattern = r\"```css.*?\n(.*?)## Anything UNCLEAR\"\n return context, pattern\n @parse\n def parse_html_code(self, context: str):\n pattern = r\"```html.*?\n(.*?)```\"\n return context, pattern\n async def draw_icons(self, context, *args, **kwargs):\n #Draw icons using SDEngine.\n engine = SDEngine()\n icon_prompts = self.parse_ui_elements(context)\n icons = icon_prompts.split(\"\n\")\n icons = [s for s in icons if len(s.strip()) > 0]\n prompts_batch = []\n for icon_prompt in icons:\n # fixme: 添加icon lora\n prompt = engine.construct_payload(icon_prompt + \".\")\n prompts_batch.append(prompt)\n await engine.run_t2i(prompts_batch)\n logger.info(\"Finish icon design using StableDiffusion API\")\n async def _save(self, css_content, html_content):\n save_dir = CONFIG.workspace_path / \"resources\" / \"codes\"\n if not os.path.exists(save_dir):\n os.makedirs(save_dir, exist_ok=True)\n # Save CSS and HTML content to files\n css_file_path = save_dir / \"ui_design.css\"\n html_file_path = save_dir / \"ui_design.html\"\n with open(css_file_path, \"w\") as css_file:\n css_file.write(css_content)\n with open(html_file_path, \"w\") as html_file:\n html_file.write(html_content)\n async def run(self, requirements: list[Message], *args, **kwargs) -> ActionOutput:\n #Run the UI Design action.\n # fixme: update prompt (根据需求细化prompt)\n context = requirements[-1].content\n ui_design_draft = self.parse_requirement(context=context)\n # todo: parse requirements str\n prompt = PROMPT_TEMPLATE.format(context=ui_design_draft, format_example=FORMAT_EXAMPLE)\n logger.info(prompt)\n ui_describe = await self._aask_v1(prompt, \"ui_design\", OUTPUT_MAPPING)\n logger.info(ui_describe.content)\n logger.info(ui_describe.instruct_content)\n css = self.parse_css_code(context=ui_describe.content)\n html = self.parse_html_code(context=ui_describe.content)\n await self._save(css_content=css, html_content=html)\n await self.draw_icons(ui_describe.content)\n return ui_describe\n```\n-----\n## format example\n[CONTENT]\n{\n \"ClassView\": \"classDiagram\n class A {\n -int x\n +int y\n -int speed\n -int direction\n +__init__(x: int, y: int, speed: int, direction: int)\n +change_direction(new_direction: int) None\n +move() None\n }\n \"\n}\n[/CONTENT]\n## nodes: \": # \"\n- ClassView: # Generate the mermaid class diagram corresponding to source code in \"context.\"\n## constraint\n- Language: Please use the same language as the user input.\n- Format: output wrapped inside [CONTENT][/CONTENT] as format example, nothing else.\n## action\nFill in the above nodes(ClassView) based on the format example.\n": "ClassView: str # Generate the mermaid class diagram corresponding to source code in \"context.\"", - "\n## context\n\n### Project Name\n\n\n### Original Requirements\n['Make a cli snake game']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Make a cli snake game\",\n \"Project Name\": \"cli_snake_game\",\n \"Product Goals\": [\n \"Create an engaging and enjoyable snake game experience\",\n \"Implement smooth and responsive controls\",\n \"Include different difficulty levels\"\n ],\n \"User Stories\": [\n \"As a player, I want to control the snake using arrow keys\",\n \"As a player, I want to see my score increase as I eat food\",\n \"As a player, I want the game to end if the snake collides with itself or the boundaries\",\n \"As a player, I want to be able to choose between different difficulty levels\",\n \"As a player, I want to see a game over message when the game ends\"\n ],\n \"Competitive Analysis\": [\n \"Snake Game A: Simple interface, lacks difficulty levels\",\n \"Snake Game B: Responsive controls, but limited features\",\n \"Snake Game C: Multiple difficulty levels, but outdated UI\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Engagement and Features of Snake Games\\\"\\n x-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n y-axis \\\"Low Features\\\" --> \\\"High Features\\\"\\n quadrant-1 \\\"Improve Engagement & Features\\\"\\n quadrant-2 \\\"Improve Engagement\\\"\\n quadrant-3 \\\"Improve Features\\\"\\n quadrant-4 \\\"Satisfactory\\\"\\n \\\"Snake Game A\\\": [0.4, 0.2]\\n \\\"Snake Game B\\\": [0.6, 0.4]\\n \\\"Snake Game C\\\": [0.7, 0.6]\\n \\\"Our Snake Game\\\": [0.8, 0.8]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"Implement snake movement and collision detection\"\n ],\n [\n \"P0\",\n \"Generate food at random positions\"\n ],\n [\n \"P0\",\n \"Increase score when snake eats food\"\n ],\n [\n \"P1\",\n \"Implement game over condition\"\n ],\n [\n \"P1\",\n \"Allow player to choose difficulty level\"\n ]\n ],\n \"UI Design draft\": \"The game will be displayed in the command line interface (CLI). The snake and food will be represented by characters. The score and game over message will be displayed at the bottom of the screen.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", - "\n## context\n{\"Language\":\"en_us\",\"Programming Language\":\"Python\",\"Original Requirements\":\"Make a cli snake game\",\"Project Name\":\"cli_snake_game\",\"Product Goals\":[\"Create an engaging and enjoyable snake game experience\",\"Implement smooth and responsive controls\",\"Include different difficulty levels\"],\"User Stories\":[\"As a player, I want to control the snake using arrow keys\",\"As a player, I want to see my score increase as I eat food\",\"As a player, I want the game to end if the snake collides with itself or the boundaries\",\"As a player, I want to be able to choose between different difficulty levels\",\"As a player, I want to see a game over message when the game ends\"],\"Competitive Analysis\":[\"Snake Game A: Simple interface, lacks difficulty levels\",\"Snake Game B: Responsive controls, but limited features\",\"Snake Game C: Multiple difficulty levels, but outdated UI\"],\"Competitive Quadrant Chart\":\"quadrantChart\\n title \\\"Engagement and Features of Snake Games\\\"\\n x-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n y-axis \\\"Low Features\\\" --> \\\"High Features\\\"\\n quadrant-1 \\\"Improve Engagement & Features\\\"\\n quadrant-2 \\\"Improve Engagement\\\"\\n quadrant-3 \\\"Improve Features\\\"\\n quadrant-4 \\\"Satisfactory\\\"\\n \\\"Snake Game A\\\": [0.4, 0.2]\\n \\\"Snake Game B\\\": [0.6, 0.4]\\n \\\"Snake Game C\\\": [0.7, 0.6]\\n \\\"Our Snake Game\\\": [0.8, 0.8]\",\"Requirement Analysis\":\"\",\"Requirement Pool\":[[\"P0\",\"Implement snake movement and collision detection\"],[\"P0\",\"Generate food at random positions\"],[\"P0\",\"Increase score when snake eats food\"],[\"P1\",\"Implement game over condition\"],[\"P1\",\"Allow player to choose difficulty level\"]],\"UI Design draft\":\"The game will be displayed in the command line interface (CLI). The snake and food will be represented by characters. The score and game over message will be displayed at the bottom of the screen.\",\"Anything UNCLEAR\":\"\"}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", - "\n## context\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Required Python packages\": [\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"openapi: 3.0.0 ...\",\n \"Shared Knowledge\": \"'game.py' contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on how to start and initialize third-party libraries.\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Required Python packages: typing.List[str] # Provide required Python packages in requirements.txt format.\n- Required Other language third-party packages: typing.List[str] # List down the required packages for languages other than Python.\n- Logic Analysis: typing.List[typing.List[str]] # Provide a list of files with the classes/methods/functions to be implemented, including dependency analysis and imports.\n- Task list: typing.List[str] # Break down the tasks into a list of filenames, prioritized by dependency order.\n- Full API spec: # Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end and back-end communication is not required, leave it blank.\n- Shared Knowledge: # Detail any shared knowledge, like common utility functions or configuration variables.\n- Anything UNCLEAR: # Mention any unclear aspects in the project management context and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Required Python packages\": [\n \"python-dotenv==0.17.1\",\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"main.py\",\n \"Contains the main function to start the game\"\n ],\n [\n \"game.py\",\n \"Contains the Game class and related functions\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"\",\n \"Shared Knowledge\": \"\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n```", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n```\n\n\n\n# Format example 1\n## Code Review: game.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: game.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - No, the code is not implemented as per the requirements. The logic for moving the snake's body, changing the direction of the snake, checking collision, generating food, starting the game, updating the game state, ending the game, and changing the difficulty of the game is missing. To achieve the requirements, the logic for each of these functions needs to be implemented step by step.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - No, the code logic is not correct as the functions are not implemented. To correct the logic, each function needs to be implemented with the appropriate logic for the game.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, all functions are not implemented. The following steps can be followed to implement each function:\n - Snake.move(): Implement the logic to move the snake's body based on the current direction.\n - Snake.change_direction(): Implement the logic to change the direction of the snake.\n - Snake.check_collision(): Implement the logic to check if the snake has collided with itself or the boundaries of the game.\n - Food.generate_food(): Implement the logic to generate a new position for the food.\n - SnakeGame.start_game(): Implement the logic to start the game.\n - SnakeGame.update_game(): Implement the logic to update the game state.\n - SnakeGame.end_game(): Implement the logic to end the game.\n - SnakeGame.change_difficulty(): Implement the logic to change the difficulty of the game.\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, there are no pre-dependencies imported in the code.\n\n6. Are methods from other files being reused correctly?\n - No, there are no methods from other files being reused in the code.\n\n## Actions: Implement the logic for each function step by step.\n\n## Code Review Result: LBTM", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n```\n\n## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - No, the code is not implemented as per the requirements. The logic for moving the snake's body, changing the direction of the snake, checking collision, generating food, starting the game, updating the game state, ending the game, and changing the difficulty of the game is missing. To achieve the requirements, you need to implement these logic steps in their respective methods.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - The code logic is not complete, as mentioned in the previous point. You need to implement the missing logic steps in their respective methods to ensure the code's correctness.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, not all functions are implemented. The following methods need to be implemented:\n - Snake.move(): Implement the logic to move the snake's body based on the current direction.\n - Snake.change_direction(direction: Position): Implement the logic to change the direction of the snake.\n - Snake.check_collision(): Implement the logic to check if the snake has collided with itself or the boundaries of the game.\n - Food.generate_food(): Implement the logic to generate a new position for the food.\n - SnakeGame.start_game(): Implement the logic to start the game.\n - SnakeGame.update_game(): Implement the logic to update the game state.\n - SnakeGame.end_game(): Implement the logic to end the game.\n - SnakeGame.change_difficulty(difficulty: int): Implement the logic to change the difficulty of the game.\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, there are no pre-dependencies required for this code.\n\n6. Are methods from other files being reused correctly?\n - No, there are no methods from other files being reused in this code.\n\n## Actions: Implement the missing logic steps in their respective methods.\n\n## Code Review Result: LBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite game.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## game.py\n...\n```\n": "```python\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # Implement the logic to move the snake's body based on the current direction\n head = self.body[0]\n new_head = Position(head.x + self.direction.x, head.y + self.direction.y)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction: Position):\n # Implement the logic to change the direction of the snake\n self.direction = direction\n\n def check_collision(self):\n # Implement the logic to check if the snake has collided with itself or the boundaries of the game\n head = self.body[0]\n if head.x < 0 or head.x >= GAME_WIDTH or head.y < 0 or head.y >= GAME_HEIGHT:\n return True\n for body_part in self.body[1:]:\n if head.x == body_part.x and head.y == body_part.y:\n return True\n return False\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # Implement the logic to generate a new position for the food\n self.position = Position(random.randint(0, GAME_WIDTH - 1), random.randint(0, GAME_HEIGHT - 1))\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # Implement the logic to start the game\n self.snake.body = [Position(0, 0)]\n self.snake.direction = Position(1, 0)\n self.food.generate_food()\n\n def update_game(self):\n # Implement the logic to update the game state\n self.snake.move()\n if self.snake.check_collision():\n self.end_game()\n elif self.snake.body[0].x == self.food.position.x and self.snake.body[0].y == self.food.position.y:\n self.score += 1\n self.snake.body.append(Position(0, 0))\n self.food.generate_food()\n\n def end_game(self):\n # Implement the logic to end the game\n print(\"Game Over\")\n print(\"Score:\", self.score)\n\n def change_difficulty(self, difficulty: int):\n # Implement the logic to change the difficulty of the game\n self.difficulty = difficulty\n```\n", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n```\n\n## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - No, the code is not implemented as per the requirements. The logic for moving the snake's body, changing the direction of the snake, checking collision, generating food, starting the game, updating the game state, ending the game, and changing the difficulty of the game is missing. To achieve the requirements, the logic for each of these functions needs to be implemented step by step.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - No, the code logic is not correct as the functions are not implemented. To correct the logic, each function needs to be implemented with the appropriate logic for the game.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, all functions are not implemented. The following steps can be followed to implement each function:\n - Snake.move(): Implement the logic to move the snake's body based on the current direction.\n - Snake.change_direction(): Implement the logic to change the direction of the snake.\n - Snake.check_collision(): Implement the logic to check if the snake has collided with itself or the boundaries of the game.\n - Food.generate_food(): Implement the logic to generate a new position for the food.\n - SnakeGame.start_game(): Implement the logic to start the game.\n - SnakeGame.update_game(): Implement the logic to update the game state.\n - SnakeGame.end_game(): Implement the logic to end the game.\n - SnakeGame.change_difficulty(): Implement the logic to change the difficulty of the game.\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, there are no pre-dependencies imported in the code.\n\n6. Are methods from other files being reused correctly?\n - No, there are no methods from other files being reused in the code.\n\n## Actions: Implement the logic for each function step by step.\n\n## Code Review Result: LBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite game.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## game.py\n...\n```\n": "```python\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n pass\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n pass\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n pass\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n pass\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n pass\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n pass\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n pass\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n pass\n```\n```", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Legacy Code\n```Code\n----- game.py\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: main.py\n```python\n## main.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: main.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: main.py\n\n```python\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n```\n\nIn the `main.py` file, we import the `SnakeGame` class from the `game` module. We then create an instance of the `SnakeGame` class and call the `start_game` method to start the game. We then enter a loop where we continuously call the `update_game` method until the `end_game` method returns `True`, indicating that the game is over.", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n----- game.py\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n\n\n# Format example 1\n## Code Review: main.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: main.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - The code in `game.py` is not fully implemented as per the requirements. There are several TODO comments indicating where the logic needs to be implemented. To achieve the requirements, the following steps can be taken:\n - Implement the logic to move the snake's body based on the current direction in the `move` method of the `Snake` class.\n - Implement the logic to change the direction of the snake in the `change_direction` method of the `Snake` class.\n - Implement the logic to check if the snake has collided with itself or the boundaries of the game in the `check_collision` method of the `Snake` class.\n - Implement the logic to generate a new position for the food in the `generate_food` method of the `Food` class.\n - Implement the logic to start the game in the `start_game` method of the `SnakeGame` class.\n - Implement the logic to update the game state in the `update_game` method of the `SnakeGame` class.\n - Implement the logic to end the game in the `end_game` method of the `SnakeGame` class.\n - Implement the logic to change the difficulty of the game in the `change_difficulty` method of the `SnakeGame` class.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - Since the code is not fully implemented, it is not possible to determine if the logic is completely correct. The missing logic needs to be implemented to ensure the correctness of the code.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces. The classes `SnakeGame`, `Snake`, `Food`, and `Position` are defined as per the class diagram.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, not all functions are implemented. The following functions need to be implemented:\n - `move` method in the `Snake` class\n - `change_direction` method in the `Snake` class\n - `check_collision` method in the `Snake` class\n - `generate_food` method in the `Food` class\n - `start_game` method in the `SnakeGame` class\n - `update_game` method in the `SnakeGame` class\n - `end_game` method in the `SnakeGame` class\n - `change_difficulty` method in the `SnakeGame` class\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, the necessary pre-dependencies have not been imported in the code. The following packages need to be imported:\n - `from typing import List` in `game.py`\n\n6. Are methods from other files being reused correctly?\n - Yes, the methods from other files are being reused correctly. The `SnakeGame` class in `game.py` is imported and used correctly in `main.py`.\n\n## Actions: Implement the missing logic in the code.\n\n## Code Review Result: LBTM", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n----- game.py\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n## Code Review: game.py\n\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n - No, the code is not implemented as per the requirements. The logic for moving the snake's body, changing the direction of the snake, and checking collision is missing. To achieve the requirements, you need to implement the missing logic in the respective methods.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - The code logic is not complete as the implementation is missing for the methods `move`, `change_direction`, and `check_collision`. You need to add the correct logic to these methods to ensure the snake moves, changes direction, and checks for collisions correctly.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, not all functions are implemented. The methods `move`, `change_direction`, `check_collision`, and `generate_food` are missing their respective logic. To implement them, you need to add the necessary code to perform the desired functionality.\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, there are no pre-dependencies required for the code in `game.py`.\n\n6. Are methods from other files being reused correctly?\n - No, there are no methods from other files being reused in `game.py`.\n\n## Actions: Implement the missing logic in the respective methods.\n\n## Code Review Result: LBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite main.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## main.py\n...\n```\n": "```python\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n```\n\n```python\n## game.py\n\nfrom typing import List\nimport random\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n new_head = Position(self.body[0].x + self.direction.x, self.body[0].y + self.direction.y)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction: Position):\n self.direction = direction\n\n def check_collision(self):\n head = self.body[0]\n if head.x < 0 or head.x >= 10 or head.y < 0 or head.y >= 10:\n return True\n for segment in self.body[1:]:\n if segment.x == head.x and segment.y == head.y:\n return True\n return False\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n self.position = Position(random.randint(0, 9), random.randint(0, 9))\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n self.snake.body = [Position(0, 0)]\n self.food.generate_food()\n\n def update_game(self):\n self.snake.move()\n if self.snake.check_collision():\n self.end_game()\n\n def end_game(self):\n print(\"Game Over\")\n print(\"Score:\", self.score)\n exit()\n\n def change_difficulty(self, difficulty: int):\n self.difficulty = difficulty\n```\n\nThe missing logic has been implemented in the respective methods. The snake can now move, change direction, and check for collisions correctly. The food is also generated at random positions.", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n----- game.py\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - The code in `game.py` is not fully implemented as per the requirements. There are several TODO comments indicating where the logic needs to be implemented. To achieve the requirements, the following steps can be taken:\n - Implement the logic to move the snake's body based on the current direction in the `move` method of the `Snake` class.\n - Implement the logic to change the direction of the snake in the `change_direction` method of the `Snake` class.\n - Implement the logic to check if the snake has collided with itself or the boundaries of the game in the `check_collision` method of the `Snake` class.\n - Implement the logic to generate a new position for the food in the `generate_food` method of the `Food` class.\n - Implement the logic to start the game in the `start_game` method of the `SnakeGame` class.\n - Implement the logic to update the game state in the `update_game` method of the `SnakeGame` class.\n - Implement the logic to end the game in the `end_game` method of the `SnakeGame` class.\n - Implement the logic to change the difficulty of the game in the `change_difficulty` method of the `SnakeGame` class.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - Since the code is not fully implemented, it is not possible to determine if the logic is completely correct. The missing logic needs to be implemented to ensure the correctness of the code.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces. The classes `SnakeGame`, `Snake`, `Food`, and `Position` are defined as per the class diagram.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, not all functions are implemented. The following functions need to be implemented:\n - `move` method in the `Snake` class\n - `change_direction` method in the `Snake` class\n - `check_collision` method in the `Snake` class\n - `generate_food` method in the `Food` class\n - `start_game` method in the `SnakeGame` class\n - `update_game` method in the `SnakeGame` class\n - `end_game` method in the `SnakeGame` class\n - `change_difficulty` method in the `SnakeGame` class\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, the necessary pre-dependencies have not been imported in the code. The following packages need to be imported:\n - `from typing import List` in `game.py`\n\n6. Are methods from other files being reused correctly?\n - Yes, the methods from other files are being reused correctly. The `SnakeGame` class in `game.py` is imported and used correctly in `main.py`.\n\n## Actions: Implement the missing logic in the code.\n\n## Code Review Result: LBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite main.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## main.py\n...\n```\n": "```python\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n```\n\n```python\n## game.py\n\nfrom typing import List\nimport random\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # Update the position of the snake's body based on the current direction\n head = self.body[0]\n new_head = Position(head.x + self.direction.x, head.y + self.direction.y)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction: Position):\n # Change the direction of the snake\n self.direction = direction\n\n def check_collision(self):\n # Check if the snake has collided with itself or the boundaries of the game\n head = self.body[0]\n if head.x < 0 or head.x >= 10 or head.y < 0 or head.y >= 10:\n return True\n for i in range(1, len(self.body)):\n if self.body[i].x == head.x and self.body[i].y == head.y:\n return True\n return False\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # Generate a new position for the food\n self.position = Position(random.randint(0, 9), random.randint(0, 9))\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # Start the game\n self.snake.body.append(Position(0, 0))\n self.food.generate_food()\n\n def update_game(self):\n # Update the game state\n self.snake.move()\n if self.snake.check_collision():\n self.end_game()\n elif self.snake.body[0].x == self.food.position.x and self.snake.body[0].y == self.food.position.y:\n self.score += 1\n self.snake.body.append(Position(0, 0))\n self.food.generate_food()\n\n def end_game(self):\n # End the game\n print(\"Game Over\")\n print(\"Score:\", self.score)\n\n def change_difficulty(self, difficulty: int):\n # Change the difficulty of the game\n self.difficulty = difficulty\n```\n```", - "## History Messages\n0: Human: Topic: climate change. Under 80 words per message.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.", - "## History Messages\n0: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.", - "## History Messages\n0: Bob(Republican candidate): I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.\n1: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n2: Human: Topic: climate change. Under 80 words per message.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "Bob: Climate change is a matter of utmost importance! We cannot ignore the urgency it demands. The potential consequences are truly alarming, and we must act now to protect our planet. Let's unite as a global community and take bold steps towards a sustainable future. Our children and future generations deserve nothing less!", - "## History Messages\n0: Alex(Democratic candidate): Bob: Climate change is a matter of utmost importance! We cannot ignore the urgency it demands. The potential consequences are truly alarming, and we must act now to protect our planet. Let's unite as a global community and take bold steps towards a sustainable future. Our children and future generations deserve nothing less!\n1: Bob(Republican candidate): I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.\n2: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "Alex(Democratic candidate): Bob, I am truly passionate about the urgency of addressing climate change. The potential consequences are alarming, and we cannot ignore them any longer. Our planet's well-being is at stake, and it is our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!", - "## History Messages\n0: Bob(Republican candidate): Alex(Democratic candidate): Bob, I am truly passionate about the urgency of addressing climate change. The potential consequences are alarming, and we cannot ignore them any longer. Our planet's well-being is at stake, and it is our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!\n1: Alex(Democratic candidate): Bob: Climate change is a matter of utmost importance! We cannot ignore the urgency it demands. The potential consequences are truly alarming, and we must act now to protect our planet. Let's unite as a global community and take bold steps towards a sustainable future. Our children and future generations deserve nothing less!\n2: Bob(Republican candidate): I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.\n3: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n4: Human: Topic: climate change. Under 80 words per message.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "Bob: Alex, I am genuinely alarmed by the potential consequences of climate change. We cannot ignore this urgent issue any longer! Our planet's well-being is at stake, and it's our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!", - "## History Messages\n0: Alex(Democratic candidate): Bob: Alex, I am genuinely alarmed by the potential consequences of climate change. We cannot ignore this urgent issue any longer! Our planet's well-being is at stake, and it's our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!\n1: Bob(Republican candidate): Alex(Democratic candidate): Bob, I am truly passionate about the urgency of addressing climate change. The potential consequences are alarming, and we cannot ignore them any longer. Our planet's well-being is at stake, and it is our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!\n2: Alex(Democratic candidate): Bob: Climate change is a matter of utmost importance! We cannot ignore the urgency it demands. The potential consequences are truly alarming, and we must act now to protect our planet. Let's unite as a global community and take bold steps towards a sustainable future. Our children and future generations deserve nothing less!\n3: Bob(Republican candidate): I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.\n4: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "I share your deep concern about climate change. The potential consequences are truly alarming, and we cannot afford to ignore this urgent issue any longer. Our planet's well-being is at stake, and it is our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!", - "## History Messages\n0: user: Topic: climate change. Under 80 words per message.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "1: Climate change is a pressing issue that demands immediate action. The consequences of inaction are dire, and we cannot afford to ignore the warnings any longer. Our planet is at stake, and it's time to prioritize sustainability and reduce our carbon footprint. Let's come together and fight for a better future for ourselves and future generations. #ActNow #SaveOurPlanet 💚🌍\n\n2: It breaks my heart to see the devastating effects of climate change. The rising sea levels, extreme weather events, and loss of biodiversity are all clear signs that we need to take action now. We owe it to our planet and future generations to make a change. Let's be responsible stewards of the Earth and work towards a sustainable and greener future. #ClimateAction #ProtectOurHome 🌱🌎\n\n3: Climate change is not just an environmental issue; it's a matter of social justice. The most vulnerable communities are disproportionately affected by its impacts. We cannot turn a blind eye to the injustice and inequality that climate change exacerbates. It's time to stand up for climate justice and ensure that everyone has equal access to a clean and safe environment. Together, we can create a more equitable and sustainable world. #ClimateJustice #EqualityForAll 🌍✊\n\n4: The science is clear: climate change is real and caused by human activities. It's frustrating to see the denial and skepticism surrounding this issue. We need to listen to the experts, trust the evidence, and take decisive action. Let's not allow ignorance and short-term interests to hinder our progress. We have the power to make a difference, so let's use it and combat climate change head-on. #ListenToScience #TakeAction 🌏🔬\n\n5: Climate change is not just a problem for future generations; it's affecting us right now. The heatwaves, droughts, and wildfires we're experiencing are all linked to climate change. We need to adapt to these changes and mitigate their impacts. It's time to prioritize renewable energy, sustainable agriculture, and conservation efforts. Our actions today will determine the world we leave behind for our children. Let's make it a better one. #ActNow #ForOurChildren 🌱🌍", - "\n## context\nwhat's the answer to 123+456?\n\n-----\n\n## format example\n[CONTENT]\n{\n \"reasoning\": \"\",\n \"answer\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- reasoning: # reasoning step by step\n- answer: # the final answer\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"reasoning\": \"To find the sum of 123 and 456, we add the corresponding digits together starting from the rightmost digit. \\n\\n3 + 6 = 9\\n2 + 5 = 7\\n1 + 4 = 5\\n\\nTherefore, the sum of 123 and 456 is 579.\",\n \"answer\": \"579\"\n}\n[/CONTENT]", - "\n## context\nwhat's the answer to 123+456?\n\n-----\n\n## format example\n[CONTENT]\n{\n \"reasoning\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- reasoning: # reasoning step by step\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"reasoning\": \"To find the sum of 123 and 456, we add the corresponding digits together. Starting from the rightmost digit, we have 3 + 6 = 9. Moving to the next digit, we have 2 + 5 = 7. Finally, adding the leftmost digits, we have 1 + 4 = 5. Therefore, the sum of 123 and 456 is 579.\"\n}\n[/CONTENT]", - "\n## context\nwhat's the answer to 123+456?\n\n-----\n\n## format example\n[CONTENT]\n{\n \"answer\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- answer: # the final answer\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"answer\": \"579\"\n}\n[/CONTENT]", - "\n## context\n我们需要一个音乐播放器,它应该有播放、暂停、上一曲、下一曲等功能。\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"We will use a Python open-source framework, such as Pygame or tkinter, to develop the music player. These frameworks provide built-in functions and classes for handling audio playback and user interface. We will analyze the difficult points of the requirements and select the framework that best meets our needs.\",\n \"File list\": [\n \"main.py\",\n \"music_player.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class MusicPlayer {\\n -current_song: Song\\n -playlist: List[Song]\\n +play()\\n +pause()\\n +next_song()\\n +previous_song()\\n }\\n class Song {\\n -title: str\\n -artist: str\\n -duration: int\\n +get_title() str\\n +get_artist() str\\n +get_duration() int\\n }\\n MusicPlayer --> Song\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant MP as MusicPlayer\\n participant S as Song\\n MP->>S: play()\\n S-->>MP: return\\n MP->>S: pause()\\n S-->>MP: return\\n MP->>S: next_song()\\n S-->>MP: return\\n MP->>S: previous_song()\\n S-->>MP: return\\n\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", - "\n## context\n\n### Legacy Content\n{\"Implementation approach\":\"We will use a Python open-source framework, such as Pygame or tkinter, to develop the music player. These frameworks provide built-in functions and classes for handling audio playback and user interface. We will analyze the difficult points of the requirements and select the framework that best meets our needs.\",\"File list\":[\"main.py\",\"music_player.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class MusicPlayer {\\n -current_song: Song\\n -playlist: List[Song]\\n +play()\\n +pause()\\n +next_song()\\n +previous_song()\\n }\\n class Song {\\n -title: str\\n -artist: str\\n -duration: int\\n +get_title() str\\n +get_artist() str\\n +get_duration() int\\n }\\n MusicPlayer --> Song\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant MP as MusicPlayer\\n participant S as Song\\n MP->>S: play()\\n S-->>MP: return\\n MP->>S: pause()\\n S-->>MP: return\\n MP->>S: next_song()\\n S-->>MP: return\\n MP->>S: previous_song()\\n S-->>MP: return\\n\",\"Anything UNCLEAR\":\"\"}\n\n### New Requirements\n## Original Requirements\nThe original requirement is to create a game similar to the classic text-based adventure game, Zork.\n\n## Product Goals\n```python\nproduct_goals = [\n \"Create an engaging text-based adventure game\",\n \"Ensure the game is easy to navigate and user-friendly\",\n \"Incorporate compelling storytelling and puzzles\"\n]\n```\n\n## User Stories\n```python\nuser_stories = [\n \"As a player, I want to be able to easily input commands so that I can interact with the game world\",\n \"As a player, I want to explore various rooms and locations to uncover the game's story\",\n \"As a player, I want to solve puzzles to progress in the game\",\n \"As a player, I want to interact with various in-game objects to enhance my gameplay experience\",\n \"As a player, I want a game that challenges my problem-solving skills and keeps me engaged\"\n]\n```\n\n## Competitive Analysis\n```python\ncompetitive_analysis = [\n \"Zork: The original text-based adventure game with complex puzzles and engaging storytelling\",\n \"The Hitchhiker's Guide to the Galaxy: A text-based game with a unique sense of humor and challenging gameplay\",\n \"Colossal Cave Adventure: The first text adventure game which set the standard for the genre\",\n \"Quest: A platform that lets users create their own text adventure games\",\n \"ChatGPT: An AI that can generate text-based adventure games\",\n \"The Forest of Doom: A text-based game with a fantasy setting and multiple endings\",\n \"Wizards Choice: A text-based game with RPG elements and a focus on player choice\"\n]\n```\n\n## Competitive Quadrant Chart\n```mermaid\nquadrantChart\n title Reach and engagement of text-based adventure games\n x-axis Low Reach --> High Reach\n y-axis Low Engagement --> High Engagement\n quadrant-1 High potential games\n quadrant-2 Popular but less engaging games\n quadrant-3 Less popular and less engaging games\n quadrant-4 Popular and engaging games\n \"Zork\": [0.9, 0.8]\n \"Hitchhiker's Guide\": [0.7, 0.7]\n \"Colossal Cave Adventure\": [0.8, 0.6]\n \"Quest\": [0.4, 0.5]\n \"ChatGPT\": [0.3, 0.6]\n \"Forest of Doom\": [0.5, 0.4]\n \"Wizards Choice\": [0.6, 0.5]\n \"Our Target Product\": [0.5, 0.6]\n```\n\n## Requirement Analysis\nThe goal is to create a text-based adventure game similar to Zork. The game should be engaging, user-friendly, and feature compelling storytelling and puzzles. It should allow players to explore various rooms and locations, interact with in-game objects, and solve puzzles to progress. The game should also challenge players' problem-solving skills and keep them engaged.\n\n## Requirement Pool\n```python\nrequirement_pool = [\n (\"Design an intuitive command input system for player interactions\", \"P0\"),\n (\"Create a variety of rooms and locations for players to explore\", \"P0\"),\n (\"Develop engaging puzzles that players need to solve to progress\", \"P0\"),\n (\"Incorporate a compelling story that unfolds as players explore the game world\", \"P1\"),\n (\"Ensure the game is user-friendly and easy to navigate\", \"P1\")\n]\n```\n\n## Anything UNCLEAR\nThe original requirement did not specify the platform for the game (web, mobile, desktop) or any specific features or themes for the game's story and puzzles. More information on these aspects could help in further refining the product requirements and design.\n\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"We will use a Python open-source framework, such as Pygame or tkinter, to develop the text-based adventure game. These frameworks provide built-in functions and classes for handling user input, managing game state, and displaying text-based interfaces. We will analyze the difficult points of the requirements and select the framework that best meets our needs.\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Game {\\n -current_room: Room\\n -player: Player\\n +start()\\n +process_input(input: str)\\n +move(direction: str)\\n +interact(object: str)\\n }\\n class Room {\\n -name: str\\n -description: str\\n -exits: Dict[str, Room]\\n -objects: List[str]\\n +get_name() str\\n +get_description() str\\n +get_exits() Dict[str, Room]\\n +get_objects() List[str]\\n }\\n class Player {\\n -inventory: List[str]\\n +get_inventory() List[str]\\n +add_to_inventory(item: str)\\n +remove_from_inventory(item: str)\\n }\\n Game --> Room\\n Game --> Player\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant G as Game\\n participant R as Room\\n participant P as Player\\n G->>G: start()\\n G->>G: process_input(input)\\n G->>G: move(direction)\\n G->>R: get_exits()\\n R-->>G: return exits\\n G->>R: get_objects()\\n R-->>G: return objects\\n G->>P: get_inventory()\\n P-->>G: return inventory\\n G->>P: add_to_inventory(item)\\n P->>G: return\\n G->>P: remove_from_inventory(item)\\n P->>G: return\\n G-->>G: continue\\n\",\n \"Anything UNCLEAR\": \"The original requirement did not specify the platform for the game (web, mobile, desktop) or any specific features or themes for the game's story and puzzles. More information on these aspects could help in further refining the product requirements and design.\"\n}\n[/CONTENT]", - "Here is the Product Requirement Document (PRD):\n\n我们需要一个音乐播放器,它应该有播放、暂停、上一曲、下一曲等功能。\n\nHere is the list of APIs designed based on this PRD:\n\n\n数据结构:\n1. Song: 包含歌曲信息,如标题、艺术家等。\n2. Playlist: 包含一系列歌曲。\n\nAPI列表:\n1. play(song: Song): 开始播放指定的歌曲。\n2. pause(): 暂停当前播放的歌曲。\n3. next(): 跳到播放列表的下一首歌曲。\n4. previous(): 跳到播放列表的上一首歌曲。\n\n\nPlease review whether this API design meets the requirements of the PRD, and whether it complies with good design practices.": "Based on the provided PRD, the API design seems to meet the requirements. It includes the necessary functions such as play, pause, next, and previous, which are all mentioned in the PRD.\n\nIn terms of compliance with good design practices, the API design could be improved by considering the following suggestions:\n\n1. Use consistent naming conventions: It is recommended to use consistent naming conventions for API functions. For example, instead of using \"next\" and \"previous\", it would be better to use \"playNext\" and \"playPrevious\" to make the naming more intuitive and consistent.\n\n2. Error handling: It is important to consider error handling in the API design. For example, if the play function is called with an invalid song, the API should return an appropriate error response.\n\n3. Return values: It would be helpful to define the expected return values for each API function. For example, the play function could return a boolean value indicating whether the song started playing successfully.\n\n4. Additional functionalities: Depending on the requirements, it might be beneficial to include additional functionalities in the API design. For example, adding a function to create or modify playlists could enhance the overall user experience.\n\nOverall, the provided API design meets the requirements of the PRD, but there are some areas where it could be further improved to align with good design practices.", - "\n## context\n\n## topic\n如何做一个生日蛋糕\n\n## record\n我认为应该先准备好材料,然后再开始做蛋糕。\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Questions\": [\n \"1. What ...\",\n \"2. How ...\",\n \"3. ...\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Questions: list[str] # Task: Refer to the context to further inquire about the details that interest you, within a word limit of 150 words. Please provide the specific details you would like to inquire about here\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Questions\": [\n \"1. 做生日蛋糕需要准备哪些材料?\",\n \"2. 做生日蛋糕的步骤是什么?\",\n \"3. 你有什么建议或技巧可以分享吗?\"\n ]\n}\n[/CONTENT]", - "Now I will provide you with the OCR text recognition results for the invoice.\nPlease extract the payee, city, total cost, and invoicing date of the invoice.\n\nThe OCR data of the invoice are as follows:\n[[[[[391.0, 43.0], [801.0, 43.0], [801.0, 81.0], [391.0, 81.0]], ('某地增值税电子普通发票', 1.0)], [[[844.0, 45.0], [1028.0, 45.0], [1028.0, 62.0], [844.0, 62.0]], ('发票代码:00100210001', 1.0)], [[[842.0, 73.0], [917.0, 73.0], [917.0, 94.0], [842.0, 94.0]], ('发票号码:', 1.0)], [[[924.0, 76.0], [1004.0, 76.0], [1004.0, 93.0], [924.0, 93.0]], ('07099363', 1.0)], [[[842.0, 107.0], [919.0, 107.0], [919.0, 124.0], [842.0, 124.0]], ('开票日期:', 1.0)], [[[930.0, 107.0], [1056.0, 107.0], [1056.0, 124.0], [930.0, 124.0]], ('2023年02月03日', 1.0)], [[[30.0, 141.0], [104.0, 141.0], [104.0, 163.0], [30.0, 163.0]], ('机器编号:', 1.0)], [[[124.0, 143.0], [236.0, 143.0], [236.0, 160.0], [124.0, 160.0]], ('499090000000', 1.0)], [[[842.0, 138.0], [1139.0, 138.0], [1139.0, 155.0], [842.0, 155.0]], ('校验码:10014320023319800000', 1.0)], [[[38.0, 187.0], [61.0, 187.0], [61.0, 208.0], [38.0, 208.0]], ('购', 1.0)], [[[77.0, 187.0], [96.0, 187.0], [96.0, 206.0], [77.0, 206.0]], ('名', 1.0)], [[[164.0, 186.0], [192.0, 186.0], [192.0, 206.0], [164.0, 206.0]], ('称:', 1.0)], [[[210.0, 185.0], [373.0, 185.0], [373.0, 206.0], [210.0, 206.0]], ('北京A科技有限公司', 1.0)], [[[686.0, 191.0], [698.0, 191.0], [698.0, 205.0], [686.0, 205.0]], ('密', 0.55)], [[[717.0, 190.0], [1162.0, 190.0], [1162.0, 207.0], [717.0, 207.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.99)], [[[76.0, 213.0], [192.0, 213.0], [192.0, 236.0], [76.0, 236.0]], ('纳税人识别号:', 1.0)], [[[212.0, 216.0], [414.0, 216.0], [414.0, 233.0], [212.0, 233.0]], ('91011111AA2AAAAA00', 1.0)], [[[715.0, 212.0], [1146.0, 213.0], [1146.0, 235.0], [715.0, 233.0]], ('07-*123<><>8000087*<64>4<8*,', 0.96)], [[[38.0, 223.0], [60.0, 223.0], [60.0, 246.0], [38.0, 246.0]], ('买', 1.0)], [[[682.0, 222.0], [701.0, 222.0], [701.0, 241.0], [682.0, 241.0]], ('码', 1.0)], [[[74.0, 239.0], [195.0, 242.0], [194.0, 267.0], [73.0, 264.0]], ('地址电话:', 0.98)], [[[715.0, 239.0], [1150.0, 239.0], [1150.0, 261.0], [715.0, 261.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[38.0, 258.0], [60.0, 258.0], [60.0, 282.0], [38.0, 282.0]], ('方', 1.0)], [[[74.0, 272.0], [194.0, 272.0], [194.0, 294.0], [74.0, 294.0]], ('开户行及账号:', 1.0)], [[[713.0, 263.0], [1153.0, 266.0], [1152.0, 287.0], [713.0, 284.0]], ('24-004*96-012>9819<<>97>>000', 1.0)], [[[65.0, 303.0], [283.0, 303.0], [283.0, 328.0], [65.0, 328.0]], ('货物或应税劳务、服务名称', 1.0)], [[[360.0, 299.0], [435.0, 299.0], [435.0, 321.0], [360.0, 321.0]], ('规格型号', 1.0)], [[[483.0, 299.0], [525.0, 299.0], [525.0, 323.0], [483.0, 323.0]], ('单位', 1.0)], [[[561.0, 299.0], [620.0, 299.0], [620.0, 323.0], [561.0, 323.0]], ('数量', 1.0)], [[[682.0, 299.0], [734.0, 299.0], [734.0, 323.0], [682.0, 323.0]], ('单价', 1.0)], [[[855.0, 301.0], [880.0, 301.0], [880.0, 321.0], [855.0, 321.0]], ('额', 1.0)], [[[942.0, 299.0], [986.0, 299.0], [986.0, 323.0], [942.0, 323.0]], ('税率', 1.0)], [[[1058.0, 301.0], [1084.0, 301.0], [1084.0, 321.0], [1058.0, 321.0]], ('税', 1.0)], [[[1093.0, 301.0], [1119.0, 301.0], [1119.0, 321.0], [1093.0, 321.0]], ('额', 1.0)], [[[30.0, 330.0], [200.0, 330.0], [200.0, 351.0], [30.0, 351.0]], ('餐饮服务*餐饮服务', 1.0)], [[[627.0, 328.0], [643.0, 328.0], [643.0, 346.0], [627.0, 346.0]], ('1', 1.0)], [[[692.0, 330.0], [752.0, 330.0], [752.0, 349.0], [692.0, 349.0]], ('379.25', 1.0)], [[[861.0, 329.0], [922.0, 329.0], [922.0, 351.0], [861.0, 351.0]], ('379.25', 1.0)], [[[968.0, 325.0], [999.0, 325.0], [999.0, 346.0], [968.0, 346.0]], ('6%', 1.0)], [[[1104.0, 329.0], [1158.0, 329.0], [1158.0, 351.0], [1104.0, 351.0]], ('22.75', 1.0)], [[[27.0, 357.0], [221.0, 357.0], [221.0, 378.0], [27.0, 378.0]], ('*日用杂品*灵感保温袋', 1.0)], [[[627.0, 351.0], [643.0, 351.0], [643.0, 372.0], [627.0, 372.0]], ('1', 1.0)], [[[710.0, 355.0], [751.0, 355.0], [751.0, 373.0], [710.0, 373.0]], ('8.85', 1.0)], [[[880.0, 354.0], [923.0, 354.0], [923.0, 376.0], [880.0, 376.0]], ('8.85', 1.0)], [[[957.0, 354.0], [1000.0, 354.0], [1000.0, 376.0], [957.0, 376.0]], ('13%', 0.96)], [[[1117.0, 351.0], [1159.0, 351.0], [1159.0, 375.0], [1117.0, 375.0]], ('1.15', 1.0)], [[[853.0, 526.0], [926.0, 529.0], [925.0, 551.0], [852.0, 548.0]], ('¥388.10', 0.94)], [[[128.0, 536.0], [153.0, 536.0], [153.0, 557.0], [128.0, 557.0]], ('合', 1.0)], [[[184.0, 536.0], [213.0, 536.0], [213.0, 557.0], [184.0, 557.0]], ('计', 1.0)], [[[1097.0, 529.0], [1160.0, 529.0], [1160.0, 551.0], [1097.0, 551.0]], ('¥23.90', 0.93)], [[[97.0, 564.0], [223.0, 564.0], [223.0, 589.0], [97.0, 589.0]], ('价税合计 (大写)', 1.0)], [[[329.0, 562.0], [498.0, 566.0], [497.0, 591.0], [329.0, 587.0]], ('肆佰壹拾贰圆整', 1.0)], [[[869.0, 563.0], [1005.0, 566.0], [1005.0, 588.0], [868.0, 585.0]], ('(小写)¥412.00', 0.96)], [[[38.0, 610.0], [61.0, 610.0], [61.0, 634.0], [38.0, 634.0]], ('销', 1.0)], [[[77.0, 604.0], [94.0, 604.0], [94.0, 623.0], [77.0, 623.0]], ('名', 1.0)], [[[155.0, 603.0], [406.0, 604.0], [406.0, 625.0], [155.0, 624.0]], ('称:深圳蛋糕餐饮有限公司', 1.0)], [[[681.0, 617.0], [703.0, 617.0], [703.0, 641.0], [681.0, 641.0]], ('备', 1.0)], [[[78.0, 629.0], [365.0, 629.0], [365.0, 646.0], [78.0, 646.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[40.0, 649.0], [58.0, 649.0], [58.0, 667.0], [40.0, 667.0]], ('售', 1.0)], [[[74.0, 650.0], [438.0, 651.0], [438.0, 676.0], [74.0, 675.0]], ('地址、电话:深圳市南山区成功大厦B座', 1.0)], [[[76.0, 674.0], [360.0, 675.0], [360.0, 697.0], [76.0, 696.0]], ('开户行及账号:中国银行深圳支行', 1.0)], [[[681.0, 672.0], [703.0, 672.0], [703.0, 695.0], [681.0, 695.0]], ('注', 1.0)], [[[41.0, 685.0], [57.0, 685.0], [57.0, 702.0], [41.0, 702.0]], ('方', 1.0)], [[[38.0, 717.0], [174.0, 717.0], [174.0, 738.0], [38.0, 738.0]], ('收款人:小明', 1.0)], [[[361.0, 718.0], [484.0, 718.0], [484.0, 739.0], [361.0, 739.0]], ('复核:小蔡', 1.0)], [[[597.0, 718.0], [682.0, 718.0], [682.0, 739.0], [597.0, 739.0]], ('开票人:', 0.99)], [[[707.0, 717.0], [752.0, 717.0], [752.0, 741.0], [707.0, 741.0]], ('小红', 1.0)], [[[870.0, 712.0], [1000.0, 712.0], [1000.0, 733.0], [870.0, 733.0]], ('销售方: (章)', 0.99)]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. The total cost refers to the total price and tax. Do not include `¥`.\n2. The city must be the recipient's city.\n2. The returned JSON dictionary must be returned in ch\n3. Mandatory requirement to output in JSON format: {\"收款人\":\"x\",\"城市\":\"x\",\"总费用/元\":\"\",\"开票日期\":\"\"}.\n": "Based on the OCR data provided, the extracted information from the invoice is as follows:\n\n- Payee: 小明 (收款人)\n- City: 深圳市 (城市)\n- Total cost: 412.00 (总费用/元)\n- Invoicing date: 2023年02月03日 (开票日期)\n\nHere is the information in JSON format:\n{\n \"收款人\": \"小明\",\n \"城市\": \"深圳市\",\n \"总费用/元\": \"412.00\",\n \"开票日期\": \"2023年02月03日\"\n}", - "Now I will provide you with the OCR text recognition results for the invoice.\nPlease answer the question: Invoicing date\n\nThe OCR data of the invoice are as follows:\n[[[[[[391.0, 43.0], [801.0, 43.0], [801.0, 81.0], [391.0, 81.0]], ('某地增值税电子普通发票', 1.0)], [[[844.0, 45.0], [1028.0, 45.0], [1028.0, 62.0], [844.0, 62.0]], ('发票代码:00100210001', 1.0)], [[[842.0, 73.0], [917.0, 73.0], [917.0, 94.0], [842.0, 94.0]], ('发票号码:', 1.0)], [[[924.0, 76.0], [1004.0, 76.0], [1004.0, 93.0], [924.0, 93.0]], ('07099363', 1.0)], [[[842.0, 107.0], [919.0, 107.0], [919.0, 124.0], [842.0, 124.0]], ('开票日期:', 1.0)], [[[930.0, 107.0], [1056.0, 107.0], [1056.0, 124.0], [930.0, 124.0]], ('2023年02月03日', 1.0)], [[[30.0, 141.0], [104.0, 141.0], [104.0, 163.0], [30.0, 163.0]], ('机器编号:', 1.0)], [[[124.0, 143.0], [236.0, 143.0], [236.0, 160.0], [124.0, 160.0]], ('499090000000', 1.0)], [[[842.0, 138.0], [1139.0, 138.0], [1139.0, 155.0], [842.0, 155.0]], ('校验码:10014320023319800000', 1.0)], [[[38.0, 187.0], [61.0, 187.0], [61.0, 208.0], [38.0, 208.0]], ('购', 1.0)], [[[77.0, 187.0], [96.0, 187.0], [96.0, 206.0], [77.0, 206.0]], ('名', 1.0)], [[[164.0, 186.0], [192.0, 186.0], [192.0, 206.0], [164.0, 206.0]], ('称:', 1.0)], [[[210.0, 185.0], [373.0, 185.0], [373.0, 206.0], [210.0, 206.0]], ('北京A科技有限公司', 1.0)], [[[686.0, 191.0], [698.0, 191.0], [698.0, 205.0], [686.0, 205.0]], ('密', 0.55)], [[[717.0, 190.0], [1162.0, 190.0], [1162.0, 207.0], [717.0, 207.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.99)], [[[76.0, 213.0], [192.0, 213.0], [192.0, 236.0], [76.0, 236.0]], ('纳税人识别号:', 1.0)], [[[212.0, 216.0], [414.0, 216.0], [414.0, 233.0], [212.0, 233.0]], ('91011111AA2AAAAA00', 1.0)], [[[715.0, 212.0], [1146.0, 213.0], [1146.0, 235.0], [715.0, 233.0]], ('07-*123<><>8000087*<64>4<8*,', 0.96)], [[[38.0, 223.0], [60.0, 223.0], [60.0, 246.0], [38.0, 246.0]], ('买', 1.0)], [[[682.0, 222.0], [701.0, 222.0], [701.0, 241.0], [682.0, 241.0]], ('码', 1.0)], [[[74.0, 239.0], [195.0, 242.0], [194.0, 267.0], [73.0, 264.0]], ('地址电话:', 0.98)], [[[715.0, 239.0], [1150.0, 239.0], [1150.0, 261.0], [715.0, 261.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[38.0, 258.0], [60.0, 258.0], [60.0, 282.0], [38.0, 282.0]], ('方', 1.0)], [[[74.0, 272.0], [194.0, 272.0], [194.0, 294.0], [74.0, 294.0]], ('开户行及账号:', 1.0)], [[[713.0, 263.0], [1153.0, 266.0], [1152.0, 287.0], [713.0, 284.0]], ('24-004*96-012>9819<<>97>>000', 1.0)], [[[65.0, 303.0], [283.0, 303.0], [283.0, 328.0], [65.0, 328.0]], ('货物或应税劳务、服务名称', 1.0)], [[[360.0, 299.0], [435.0, 299.0], [435.0, 321.0], [360.0, 321.0]], ('规格型号', 1.0)], [[[483.0, 299.0], [525.0, 299.0], [525.0, 323.0], [483.0, 323.0]], ('单位', 1.0)], [[[561.0, 299.0], [620.0, 299.0], [620.0, 323.0], [561.0, 323.0]], ('数量', 1.0)], [[[682.0, 299.0], [734.0, 299.0], [734.0, 323.0], [682.0, 323.0]], ('单价', 1.0)], [[[855.0, 301.0], [880.0, 301.0], [880.0, 321.0], [855.0, 321.0]], ('额', 1.0)], [[[942.0, 299.0], [986.0, 299.0], [986.0, 323.0], [942.0, 323.0]], ('税率', 1.0)], [[[1058.0, 301.0], [1084.0, 301.0], [1084.0, 321.0], [1058.0, 321.0]], ('税', 1.0)], [[[1093.0, 301.0], [1119.0, 301.0], [1119.0, 321.0], [1093.0, 321.0]], ('额', 1.0)], [[[30.0, 330.0], [200.0, 330.0], [200.0, 351.0], [30.0, 351.0]], ('餐饮服务*餐饮服务', 1.0)], [[[627.0, 328.0], [643.0, 328.0], [643.0, 346.0], [627.0, 346.0]], ('1', 1.0)], [[[692.0, 330.0], [752.0, 330.0], [752.0, 349.0], [692.0, 349.0]], ('379.25', 1.0)], [[[861.0, 329.0], [922.0, 329.0], [922.0, 351.0], [861.0, 351.0]], ('379.25', 1.0)], [[[968.0, 325.0], [999.0, 325.0], [999.0, 346.0], [968.0, 346.0]], ('6%', 1.0)], [[[1104.0, 329.0], [1158.0, 329.0], [1158.0, 351.0], [1104.0, 351.0]], ('22.75', 1.0)], [[[27.0, 357.0], [221.0, 357.0], [221.0, 378.0], [27.0, 378.0]], ('*日用杂品*灵感保温袋', 1.0)], [[[627.0, 351.0], [643.0, 351.0], [643.0, 372.0], [627.0, 372.0]], ('1', 1.0)], [[[710.0, 355.0], [751.0, 355.0], [751.0, 373.0], [710.0, 373.0]], ('8.85', 1.0)], [[[880.0, 354.0], [923.0, 354.0], [923.0, 376.0], [880.0, 376.0]], ('8.85', 1.0)], [[[957.0, 354.0], [1000.0, 354.0], [1000.0, 376.0], [957.0, 376.0]], ('13%', 0.96)], [[[1117.0, 351.0], [1159.0, 351.0], [1159.0, 375.0], [1117.0, 375.0]], ('1.15', 1.0)], [[[853.0, 526.0], [926.0, 529.0], [925.0, 551.0], [852.0, 548.0]], ('¥388.10', 0.94)], [[[128.0, 536.0], [153.0, 536.0], [153.0, 557.0], [128.0, 557.0]], ('合', 1.0)], [[[184.0, 536.0], [213.0, 536.0], [213.0, 557.0], [184.0, 557.0]], ('计', 1.0)], [[[1097.0, 529.0], [1160.0, 529.0], [1160.0, 551.0], [1097.0, 551.0]], ('¥23.90', 0.93)], [[[97.0, 564.0], [223.0, 564.0], [223.0, 589.0], [97.0, 589.0]], ('价税合计 (大写)', 1.0)], [[[329.0, 562.0], [498.0, 566.0], [497.0, 591.0], [329.0, 587.0]], ('肆佰壹拾贰圆整', 1.0)], [[[869.0, 563.0], [1005.0, 566.0], [1005.0, 588.0], [868.0, 585.0]], ('(小写)¥412.00', 0.96)], [[[38.0, 610.0], [61.0, 610.0], [61.0, 634.0], [38.0, 634.0]], ('销', 1.0)], [[[77.0, 604.0], [94.0, 604.0], [94.0, 623.0], [77.0, 623.0]], ('名', 1.0)], [[[155.0, 603.0], [406.0, 604.0], [406.0, 625.0], [155.0, 624.0]], ('称:深圳蛋糕餐饮有限公司', 1.0)], [[[681.0, 617.0], [703.0, 617.0], [703.0, 641.0], [681.0, 641.0]], ('备', 1.0)], [[[78.0, 629.0], [365.0, 629.0], [365.0, 646.0], [78.0, 646.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[40.0, 649.0], [58.0, 649.0], [58.0, 667.0], [40.0, 667.0]], ('售', 1.0)], [[[74.0, 650.0], [438.0, 651.0], [438.0, 676.0], [74.0, 675.0]], ('地址、电话:深圳市南山区成功大厦B座', 1.0)], [[[76.0, 674.0], [360.0, 675.0], [360.0, 697.0], [76.0, 696.0]], ('开户行及账号:中国银行深圳支行', 1.0)], [[[681.0, 672.0], [703.0, 672.0], [703.0, 695.0], [681.0, 695.0]], ('注', 1.0)], [[[41.0, 685.0], [57.0, 685.0], [57.0, 702.0], [41.0, 702.0]], ('方', 1.0)], [[[38.0, 717.0], [174.0, 717.0], [174.0, 738.0], [38.0, 738.0]], ('收款人:小明', 1.0)], [[[361.0, 718.0], [484.0, 718.0], [484.0, 739.0], [361.0, 739.0]], ('复核:小蔡', 1.0)], [[[597.0, 718.0], [682.0, 718.0], [682.0, 739.0], [597.0, 739.0]], ('开票人:', 0.99)], [[[707.0, 717.0], [752.0, 717.0], [752.0, 741.0], [707.0, 741.0]], ('小红', 1.0)], [[[870.0, 712.0], [1000.0, 712.0], [1000.0, 733.0], [870.0, 733.0]], ('销售方: (章)', 0.99)]]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. Answer in ch language.\n2. Enforce restrictions on not returning OCR data sent to you.\n3. Return with markdown syntax layout.\n": "The invoicing date on the invoice is **2023年02月03日**.", - "\n## context\nI just graduated and hope to find a job as a Python engineer\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Questions\": [\n \"1. What ...\",\n \"2. How ...\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Questions: list[str] # Role: You are an interviewer of our company who is well-knonwn in frontend or backend develop;\nRequirement: Provide a list of questions for the interviewer to ask the interviewee, by reading the resume of the interviewee in the context.\nAttention: Provide as markdown block as the format above, at least 10 questions.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Questions\": [\n \"1. Can you tell me about your experience with Python programming?\",\n \"2. Have you worked on any projects using Python? If so, can you describe one of them?\",\n \"3. What frameworks or libraries are you familiar with in Python?\",\n \"4. How do you handle errors and exceptions in Python?\",\n \"5. Can you explain the concept of generators in Python?\",\n \"6. How do you manage dependencies in Python projects?\",\n \"7. Have you used any testing frameworks in Python? If yes, which ones?\",\n \"8. Can you explain the difference between list comprehension and generator expression in Python?\",\n \"9. How do you optimize the performance of Python code?\",\n \"10. Can you describe a situation where you had to debug a complex Python program? How did you approach it?\"\n ]\n}\n[/CONTENT]", - "\n## context\n{'Implementation approach': '我们将使用Python编程语言,并选择合适的开源框架来实现贪吃蛇游戏。我们将分析需求中的难点,并选择合适的开源框架来简化开发流程。', 'File list': ['main.py', 'game.py'], 'Data structures and interfaces': '\\nclassDiagram\\n class Game {\\n -int width\\n -int height\\n -int score\\n -int speed\\n -List snake\\n -Point food\\n +__init__(width: int, height: int, speed: int)\\n +start_game()\\n +change_direction(direction: str)\\n +game_over()\\n +update_snake()\\n +update_food()\\n +check_collision()\\n }\\n class Point {\\n -int x\\n -int y\\n +__init__(x: int, y: int)\\n }\\n Game --> Point\\n', 'Program call flow': '\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: start_game()\\n M->>G: change_direction(direction)\\n G->>G: update_snake()\\n G->>G: update_food()\\n G->>G: check_collision()\\n G-->>G: game_over()\\n', 'Anything UNCLEAR': ''}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Required Python packages\": [\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"openapi: 3.0.0 ...\",\n \"Shared Knowledge\": \"'game.py' contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on how to start and initialize third-party libraries.\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Required Python packages: typing.List[str] # Provide required Python packages in requirements.txt format.\n- Required Other language third-party packages: typing.List[str] # List down the required packages for languages other than Python.\n- Logic Analysis: typing.List[typing.List[str]] # Provide a list of files with the classes/methods/functions to be implemented, including dependency analysis and imports.\n- Task list: typing.List[str] # Break down the tasks into a list of filenames, prioritized by dependency order.\n- Full API spec: # Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end and back-end communication is not required, leave it blank.\n- Shared Knowledge: # Detail any shared knowledge, like common utility functions or configuration variables.\n- Anything UNCLEAR: # Mention any unclear aspects in the project management context and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Required Python packages\": [\n \"pygame==2.0.1\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and related functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, imports Game class from game.py\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"\",\n \"Shared Knowledge\": \"'game.py' contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\nimport asyncio\nfrom pathlib import Path\n\nimport typer\n\nfrom metagpt.config import CONFIG\n\napp = typer.Typer(add_completion=False)\n\n\n@app.command()\ndef startup(\n idea: str = typer.Argument(..., help=\"Your innovative idea, such as 'Create a 2048 game.'\"),\n investment: float = typer.Option(default=3.0, help=\"Dollar amount to invest in the AI company.\"),\n n_round: int = typer.Option(default=5, help=\"Number of rounds for the simulation.\"),\n code_review: bool = typer.Option(default=True, help=\"Whether to use code review.\"),\n run_tests: bool = typer.Option(default=False, help=\"Whether to enable QA for adding & running tests.\"),\n implement: bool = typer.Option(default=True, help=\"Enable or disable code implementation.\"),\n project_name: str = typer.Option(default=\"\", help=\"Unique project name, such as 'game_2048'.\"),\n inc: bool = typer.Option(default=False, help=\"Incremental mode. Use it to coop with existing repo.\"),\n project_path: str = typer.Option(\n default=\"\",\n help=\"Specify the directory path of the old version project to fulfill the incremental requirements.\",\n ),\n reqa_file: str = typer.Option(\n default=\"\", help=\"Specify the source file name for rewriting the quality assurance code.\"\n ),\n max_auto_summarize_code: int = typer.Option(\n default=0,\n help=\"The maximum number of times the 'SummarizeCode' action is automatically invoked, with -1 indicating \"\n \"unlimited. This parameter is used for debugging the workflow.\",\n ),\n recover_path: str = typer.Option(default=None, help=\"recover the project from existing serialized storage\"),\n):\n \"\"\"Run a startup. Be a boss.\"\"\"\n from metagpt.roles import (\n Architect,\n Engineer,\n ProductManager,\n ProjectManager,\n QaEngineer,\n )\n from metagpt.team import Team\n\n CONFIG.update_via_cli(project_path, project_name, inc, reqa_file, max_auto_summarize_code)\n\n if not recover_path:\n company = Team()\n company.hire(\n [\n ProductManager(),\n Architect(),\n ProjectManager(),\n ]\n )\n\n if implement or code_review:\n company.hire([Engineer(n_borg=5, use_code_review=code_review)])\n\n if run_tests:\n company.hire([QaEngineer()])\n else:\n # # stg_path = SERDESER_PATH.joinpath(\"team\")\n stg_path = Path(recover_path)\n if not stg_path.exists() or not str(stg_path).endswith(\"team\"):\n raise FileNotFoundError(f\"{recover_path} not exists or not endswith `team`\")\n\n company = Team.deserialize(stg_path=stg_path)\n idea = company.idea # use original idea\n\n company.invest(investment)\n company.run_project(idea)\n asyncio.run(company.run(n_round=n_round))\n\n\nif __name__ == \"__main__\":\n app()\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "The Mermaid Sequence Diagram for the given code is as follows:\n\n```mermaid\nsequenceDiagram\n participant User\n participant Typer\n participant Team\n participant ProductManager\n participant Architect\n participant ProjectManager\n participant Engineer\n participant QaEngineer\n\n User ->> Typer: Run startup command\n Typer ->> Team: Create Team instance\n Team ->> Team: Hire ProductManager, Architect, ProjectManager\n Team ->> Team: Hire Engineer (if implement or code_review is True)\n Team ->> Team: Hire QaEngineer (if run_tests is True)\n User ->> Team: Set project_path, project_name, inc, reqa_file, max_auto_summarize_code\n Team ->> Team: Update CONFIG with CLI arguments\n Team ->> Team: Invest in the company\n Team ->> Team: Run project with the given idea\n Team ->> Team: Run simulation for n_rounds\n\n```\n\nNote: The diagram represents the sequence of interactions between different participants (User, Typer, Team, ProductManager, Architect, ProjectManager, Engineer, QaEngineer) in the code.", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n\nfrom __future__ import annotations\n\nimport asyncio\nimport json\nfrom concurrent import futures\nfrom typing import Literal, overload\n\ntry:\n from duckduckgo_search import DDGS\nexcept ImportError:\n raise ImportError(\n \"To use this module, you should have the `duckduckgo_search` Python package installed. \"\n \"You can install it by running the command: `pip install -e.[search-ddg]`\"\n )\n\nfrom metagpt.config import CONFIG\n\n\nclass DDGAPIWrapper:\n \"\"\"Wrapper around duckduckgo_search API.\n\n To use this module, you should have the `duckduckgo_search` Python package installed.\n \"\"\"\n\n def __init__(\n self,\n *,\n loop: asyncio.AbstractEventLoop | None = None,\n executor: futures.Executor | None = None,\n ):\n kwargs = {}\n if CONFIG.global_proxy:\n kwargs[\"proxies\"] = CONFIG.global_proxy\n self.loop = loop\n self.executor = executor\n self.ddgs = DDGS(**kwargs)\n\n @overload\n def run(\n self,\n query: str,\n max_results: int = 8,\n as_string: Literal[True] = True,\n focus: list[str] | None = None,\n ) -> str:\n ...\n\n @overload\n def run(\n self,\n query: str,\n max_results: int = 8,\n as_string: Literal[False] = False,\n focus: list[str] | None = None,\n ) -> list[dict[str, str]]:\n ...\n\n async def run(\n self,\n query: str,\n max_results: int = 8,\n as_string: bool = True,\n ) -> str | list[dict]:\n \"\"\"Return the results of a Google search using the official Google API\n\n Args:\n query: The search query.\n max_results: The number of results to return.\n as_string: A boolean flag to determine the return type of the results. If True, the function will\n return a formatted string with the search results. If False, it will return a list of dictionaries\n containing detailed information about each search result.\n\n Returns:\n The results of the search.\n \"\"\"\n loop = self.loop or asyncio.get_event_loop()\n future = loop.run_in_executor(\n self.executor,\n self._search_from_ddgs,\n query,\n max_results,\n )\n search_results = await future\n\n # Return the list of search result URLs\n if as_string:\n return json.dumps(search_results, ensure_ascii=False)\n return search_results\n\n def _search_from_ddgs(self, query: str, max_results: int):\n return [\n {\"link\": i[\"href\"], \"snippet\": i[\"body\"], \"title\": i[\"title\"]}\n for (_, i) in zip(range(max_results), self.ddgs.text(query))\n ]\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(DDGAPIWrapper().run)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant User\n participant DDGAPIWrapper\n participant DDGS\n participant asyncio\n participant futures\n participant CONFIG\n participant fire\n\n User->>DDGAPIWrapper: Instantiate DDGAPIWrapper\n Note over DDGAPIWrapper: Wrapper around duckduckgo_search API\n \n alt Check if duckduckgo_search package is installed\n DDGAPIWrapper->>DDGAPIWrapper: Raise ImportError\n else\n DDGAPIWrapper->>DDGAPIWrapper: Set kwargs with global_proxy if available\n DDGAPIWrapper->>DDGAPIWrapper: Set loop and executor attributes\n DDGAPIWrapper->>DDGS: Instantiate DDGS with kwargs\n end\n\n User->>DDGAPIWrapper: Call run() method\n Note over DDGAPIWrapper: Overloaded method with different return types\n\n alt Return type is True\n DDGAPIWrapper->>asyncio: Get event loop\n DDGAPIWrapper->>loop: Run search_from_ddgs() in executor\n loop->>futures: Run search_from_ddgs() in executor\n futures->>DDGAPIWrapper: Return search results\n DDGAPIWrapper->>DDGAPIWrapper: Format search results as string\n DDGAPIWrapper->>User: Return search results as string\n else\n DDGAPIWrapper->>asyncio: Get event loop\n DDGAPIWrapper->>loop: Run search_from_ddgs() in executor\n loop->>futures: Run search_from_ddgs() in executor\n futures->>DDGAPIWrapper: Return search results\n DDGAPIWrapper->>User: Return search results as list of dictionaries\n end\n\n Note over DDGAPIWrapper: Private method _search_from_ddgs()\n\n DDGAPIWrapper->>DDGS: Call text() method with query\n DDGS->>DDGAPIWrapper: Return search results\n DDGAPIWrapper->>DDGAPIWrapper: Format search results as list of dictionaries\n DDGAPIWrapper->>User: Return search results as list of dictionaries\n\n User->>fire: Import fire module\n fire->>DDGAPIWrapper: Call run() method\n Note over DDGAPIWrapper: Run search() method with default parameters\n```", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/8/17\n@Author : mashenquan\n@File : metagpt_oas3_api_svc.py\n@Desc : MetaGPT OpenAPI Specification 3.0 REST API service\n\n curl -X 'POST' \\\n 'http://localhost:8080/openapi/greeting/dave' \\\n -H 'accept: text/plain' \\\n -H 'Content-Type: application/json' \\\n -d '{}'\n\"\"\"\n\nfrom pathlib import Path\n\nimport connexion\n\n\ndef oas_http_svc():\n \"\"\"Start the OAS 3.0 OpenAPI HTTP service\"\"\"\n print(\"http://localhost:8080/oas3/ui/\")\n specification_dir = Path(__file__).parent.parent.parent / \"docs/.well-known\"\n app = connexion.AsyncApp(__name__, specification_dir=str(specification_dir))\n app.add_api(\"metagpt_oas3_api.yaml\")\n app.add_api(\"openapi.yaml\")\n app.run(port=8080)\n\n\nif __name__ == \"__main__\":\n oas_http_svc()\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "To translate the given Python code into a Mermaid Sequence Diagram, we need to understand the flow of the code and identify the interactions between different components. Here's the translated code into a Mermaid Sequence Diagram:\n\n```mermaid\nsequenceDiagram\n participant User\n participant metagpt_oas3_api_svc.py\n participant connexion\n participant metagpt_oas3_api.yaml\n participant openapi.yaml\n\n User->>metagpt_oas3_api_svc.py: Start the OAS 3.0 OpenAPI HTTP service\n metagpt_oas3_api_svc.py->>connexion: Create an AsyncApp instance\n metagpt_oas3_api_svc.py->>connexion: Add the metagpt_oas3_api.yaml specification\n metagpt_oas3_api_svc.py->>connexion: Add the openapi.yaml specification\n metagpt_oas3_api_svc.py->>connexion: Run the HTTP service on port 8080\n connexion->>User: Display the URL for accessing the OAS 3.0 UI\n\n Note over metagpt_oas3_api_svc.py, connexion: The HTTP service is running on http://localhost:8080/oas3/ui/\n```\n\nIn the diagram, the User starts the OAS 3.0 OpenAPI HTTP service by executing the `oas_http_svc()` function in the `metagpt_oas3_api_svc.py` file. This function creates an instance of the `connexion.AsyncApp` class from the `connexion` library. The `metagpt_oas3_api.yaml` and `openapi.yaml` specifications are added to the app. Finally, the HTTP service is run on port 8080, and the URL for accessing the OAS 3.0 UI is displayed to the User.", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/5/23 18:27\n@Author : alexanderwu\n@File : search_engine_serpapi.py\n\"\"\"\nfrom typing import Any, Dict, Optional, Tuple\n\nimport aiohttp\nfrom pydantic import BaseModel, ConfigDict, Field, field_validator\n\nfrom metagpt.config import CONFIG\n\n\nclass SerpAPIWrapper(BaseModel):\n model_config = ConfigDict(arbitrary_types_allowed=True)\n\n search_engine: Any = None #: :meta private:\n params: dict = Field(\n default_factory=lambda: {\n \"engine\": \"google\",\n \"google_domain\": \"google.com\",\n \"gl\": \"us\",\n \"hl\": \"en\",\n }\n )\n # should add `validate_default=True` to check with default value\n serpapi_api_key: Optional[str] = Field(default=None, validate_default=True)\n aiosession: Optional[aiohttp.ClientSession] = None\n\n @field_validator(\"serpapi_api_key\", mode=\"before\")\n @classmethod\n def check_serpapi_api_key(cls, val: str):\n val = val or CONFIG.serpapi_api_key\n if not val:\n raise ValueError(\n \"To use, make sure you provide the serpapi_api_key when constructing an object. Alternatively, \"\n \"ensure that the environment variable SERPAPI_API_KEY is set with your API key. You can obtain \"\n \"an API key from https://serpapi.com/.\"\n )\n return val\n\n async def run(self, query, max_results: int = 8, as_string: bool = True, **kwargs: Any) -> str:\n \"\"\"Run query through SerpAPI and parse result async.\"\"\"\n result = await self.results(query, max_results)\n return self._process_response(result, as_string=as_string)\n\n async def results(self, query: str, max_results: int) -> dict:\n \"\"\"Use aiohttp to run query through SerpAPI and return the results async.\"\"\"\n\n def construct_url_and_params() -> Tuple[str, Dict[str, str]]:\n params = self.get_params(query)\n params[\"source\"] = \"python\"\n params[\"num\"] = max_results\n params[\"output\"] = \"json\"\n url = \"https://serpapi.com/search\"\n return url, params\n\n url, params = construct_url_and_params()\n if not self.aiosession:\n async with aiohttp.ClientSession() as session:\n async with session.get(url, params=params) as response:\n res = await response.json()\n else:\n async with self.aiosession.get(url, params=params) as response:\n res = await response.json()\n\n return res\n\n def get_params(self, query: str) -> Dict[str, str]:\n \"\"\"Get parameters for SerpAPI.\"\"\"\n _params = {\n \"api_key\": self.serpapi_api_key,\n \"q\": query,\n }\n params = {**self.params, **_params}\n return params\n\n @staticmethod\n def _process_response(res: dict, as_string: bool) -> str:\n \"\"\"Process response from SerpAPI.\"\"\"\n # logger.debug(res)\n focus = [\"title\", \"snippet\", \"link\"]\n get_focused = lambda x: {i: j for i, j in x.items() if i in focus}\n\n if \"error\" in res.keys():\n raise ValueError(f\"Got error from SerpAPI: {res['error']}\")\n if \"answer_box\" in res.keys() and \"answer\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"answer\"]\n elif \"answer_box\" in res.keys() and \"snippet\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"snippet\"]\n elif \"answer_box\" in res.keys() and \"snippet_highlighted_words\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"snippet_highlighted_words\"][0]\n elif \"sports_results\" in res.keys() and \"game_spotlight\" in res[\"sports_results\"].keys():\n toret = res[\"sports_results\"][\"game_spotlight\"]\n elif \"knowledge_graph\" in res.keys() and \"description\" in res[\"knowledge_graph\"].keys():\n toret = res[\"knowledge_graph\"][\"description\"]\n elif \"snippet\" in res[\"organic_results\"][0].keys():\n toret = res[\"organic_results\"][0][\"snippet\"]\n else:\n toret = \"No good search result found\"\n\n toret_l = []\n if \"answer_box\" in res.keys() and \"snippet\" in res[\"answer_box\"].keys():\n toret_l += [get_focused(res[\"answer_box\"])]\n if res.get(\"organic_results\"):\n toret_l += [get_focused(i) for i in res.get(\"organic_results\")]\n\n return str(toret) + \"\\n\" + str(toret_l) if as_string else toret_l\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(SerpAPIWrapper().run)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "The Mermaid Sequence Diagram for the given code is as follows:\n\n```mermaid\nsequenceDiagram\n participant User\n participant SerpAPIWrapper\n participant aiohttp.ClientSession\n participant SerpAPI\n\n User->>SerpAPIWrapper: Run query\n SerpAPIWrapper->>SerpAPIWrapper: Check serpapi_api_key\n alt serpapi_api_key is not provided\n SerpAPIWrapper-->>User: Raise ValueError\n else serpapi_api_key is provided\n SerpAPIWrapper->>SerpAPIWrapper: Get params\n SerpAPIWrapper->>SerpAPI: Send request\n SerpAPI-->>SerpAPIWrapper: Return response\n SerpAPIWrapper->>SerpAPIWrapper: Process response\n SerpAPIWrapper-->>User: Return result\n end\n```\n\nPlease note that the diagram is a simplified representation of the code logic and may not include all the details.", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/5/23 18:27\n@Author : alexanderwu\n@File : search_engine_serpapi.py\n\"\"\"\nimport json\nfrom typing import Any, Dict, Optional, Tuple\n\nimport aiohttp\nfrom pydantic import BaseModel, ConfigDict, Field, field_validator\n\nfrom metagpt.config import CONFIG\n\n\nclass SerperWrapper(BaseModel):\n model_config = ConfigDict(arbitrary_types_allowed=True)\n\n search_engine: Any = None #: :meta private:\n payload: dict = Field(default_factory=lambda: {\"page\": 1, \"num\": 10})\n serper_api_key: Optional[str] = Field(default=None, validate_default=True)\n aiosession: Optional[aiohttp.ClientSession] = None\n\n @field_validator(\"serper_api_key\", mode=\"before\")\n @classmethod\n def check_serper_api_key(cls, val: str):\n val = val or CONFIG.serper_api_key\n if not val:\n raise ValueError(\n \"To use, make sure you provide the serper_api_key when constructing an object. Alternatively, \"\n \"ensure that the environment variable SERPER_API_KEY is set with your API key. You can obtain \"\n \"an API key from https://serper.dev/.\"\n )\n return val\n\n async def run(self, query: str, max_results: int = 8, as_string: bool = True, **kwargs: Any) -> str:\n \"\"\"Run query through Serper and parse result async.\"\"\"\n if isinstance(query, str):\n return self._process_response((await self.results([query], max_results))[0], as_string=as_string)\n else:\n results = [self._process_response(res, as_string) for res in await self.results(query, max_results)]\n return \"\\n\".join(results) if as_string else results\n\n async def results(self, queries: list[str], max_results: int = 8) -> dict:\n \"\"\"Use aiohttp to run query through Serper and return the results async.\"\"\"\n\n def construct_url_and_payload_and_headers() -> Tuple[str, Dict[str, str]]:\n payloads = self.get_payloads(queries, max_results)\n url = \"https://google.serper.dev/search\"\n headers = self.get_headers()\n return url, payloads, headers\n\n url, payloads, headers = construct_url_and_payload_and_headers()\n if not self.aiosession:\n async with aiohttp.ClientSession() as session:\n async with session.post(url, data=payloads, headers=headers) as response:\n res = await response.json()\n else:\n async with self.aiosession.get.post(url, data=payloads, headers=headers) as response:\n res = await response.json()\n\n return res\n\n def get_payloads(self, queries: list[str], max_results: int) -> Dict[str, str]:\n \"\"\"Get payloads for Serper.\"\"\"\n payloads = []\n for query in queries:\n _payload = {\n \"q\": query,\n \"num\": max_results,\n }\n payloads.append({**self.payload, **_payload})\n return json.dumps(payloads, sort_keys=True)\n\n def get_headers(self) -> Dict[str, str]:\n headers = {\"X-API-KEY\": self.serper_api_key, \"Content-Type\": \"application/json\"}\n return headers\n\n @staticmethod\n def _process_response(res: dict, as_string: bool = False) -> str:\n \"\"\"Process response from SerpAPI.\"\"\"\n # logger.debug(res)\n focus = [\"title\", \"snippet\", \"link\"]\n\n def get_focused(x):\n return {i: j for i, j in x.items() if i in focus}\n\n if \"error\" in res.keys():\n raise ValueError(f\"Got error from SerpAPI: {res['error']}\")\n if \"answer_box\" in res.keys() and \"answer\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"answer\"]\n elif \"answer_box\" in res.keys() and \"snippet\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"snippet\"]\n elif \"answer_box\" in res.keys() and \"snippet_highlighted_words\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"snippet_highlighted_words\"][0]\n elif \"sports_results\" in res.keys() and \"game_spotlight\" in res[\"sports_results\"].keys():\n toret = res[\"sports_results\"][\"game_spotlight\"]\n elif \"knowledge_graph\" in res.keys() and \"description\" in res[\"knowledge_graph\"].keys():\n toret = res[\"knowledge_graph\"][\"description\"]\n elif \"snippet\" in res[\"organic\"][0].keys():\n toret = res[\"organic\"][0][\"snippet\"]\n else:\n toret = \"No good search result found\"\n\n toret_l = []\n if \"answer_box\" in res.keys() and \"snippet\" in res[\"answer_box\"].keys():\n toret_l += [get_focused(res[\"answer_box\"])]\n if res.get(\"organic\"):\n toret_l += [get_focused(i) for i in res.get(\"organic\")]\n\n return str(toret) + \"\\n\" + str(toret_l) if as_string else toret_l\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(SerperWrapper().run)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "The Mermaid Sequence Diagram for the given code can be represented as follows:\n\n```mermaid\nsequenceDiagram\n participant User\n participant SerperWrapper\n participant aiohttp.ClientSession\n participant SerpAPI\n\n User->>SerperWrapper: run(query, max_results, as_string, **kwargs)\n SerperWrapper->>SerperWrapper: _process_response()\n SerperWrapper->>SerperWrapper: get_payloads()\n SerperWrapper->>SerperWrapper: get_headers()\n SerperWrapper->>aiohttp.ClientSession: post(url, data, headers)\n aiohttp.ClientSession->>SerpAPI: POST /search\n SerpAPI-->>aiohttp.ClientSession: Response\n aiohttp.ClientSession-->>SerperWrapper: Response\n SerperWrapper->>SerperWrapper: _process_response()\n SerperWrapper->>User: Response\n```\n\nNote: This diagram represents the flow of execution for the `run()` method in the `SerperWrapper` class. It shows the interaction between the user, the `SerperWrapper` object, the `aiohttp.ClientSession`, and the SerpAPI.", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\nfrom __future__ import annotations\n\nimport asyncio\nimport json\nfrom concurrent import futures\nfrom typing import Optional\nfrom urllib.parse import urlparse\n\nimport httplib2\nfrom pydantic import BaseModel, ConfigDict, Field, field_validator\n\nfrom metagpt.config import CONFIG\nfrom metagpt.logs import logger\n\ntry:\n from googleapiclient.discovery import build\n from googleapiclient.errors import HttpError\nexcept ImportError:\n raise ImportError(\n \"To use this module, you should have the `google-api-python-client` Python package installed. \"\n \"You can install it by running the command: `pip install -e.[search-google]`\"\n )\n\n\nclass GoogleAPIWrapper(BaseModel):\n model_config = ConfigDict(arbitrary_types_allowed=True)\n\n google_api_key: Optional[str] = Field(default=None, validate_default=True)\n google_cse_id: Optional[str] = Field(default=None, validate_default=True)\n loop: Optional[asyncio.AbstractEventLoop] = None\n executor: Optional[futures.Executor] = None\n\n @field_validator(\"google_api_key\", mode=\"before\")\n @classmethod\n def check_google_api_key(cls, val: str):\n val = val or CONFIG.google_api_key\n if not val:\n raise ValueError(\n \"To use, make sure you provide the google_api_key when constructing an object. Alternatively, \"\n \"ensure that the environment variable GOOGLE_API_KEY is set with your API key. You can obtain \"\n \"an API key from https://console.cloud.google.com/apis/credentials.\"\n )\n return val\n\n @field_validator(\"google_cse_id\", mode=\"before\")\n @classmethod\n def check_google_cse_id(cls, val: str):\n val = val or CONFIG.google_cse_id\n if not val:\n raise ValueError(\n \"To use, make sure you provide the google_cse_id when constructing an object. Alternatively, \"\n \"ensure that the environment variable GOOGLE_CSE_ID is set with your API key. You can obtain \"\n \"an API key from https://programmablesearchengine.google.com/controlpanel/create.\"\n )\n return val\n\n @property\n def google_api_client(self):\n build_kwargs = {\"developerKey\": self.google_api_key}\n if CONFIG.global_proxy:\n parse_result = urlparse(CONFIG.global_proxy)\n proxy_type = parse_result.scheme\n if proxy_type == \"https\":\n proxy_type = \"http\"\n build_kwargs[\"http\"] = httplib2.Http(\n proxy_info=httplib2.ProxyInfo(\n getattr(httplib2.socks, f\"PROXY_TYPE_{proxy_type.upper()}\"),\n parse_result.hostname,\n parse_result.port,\n ),\n )\n service = build(\"customsearch\", \"v1\", **build_kwargs)\n return service.cse()\n\n async def run(\n self,\n query: str,\n max_results: int = 8,\n as_string: bool = True,\n focus: list[str] | None = None,\n ) -> str | list[dict]:\n \"\"\"Return the results of a Google search using the official Google API.\n\n Args:\n query: The search query.\n max_results: The number of results to return.\n as_string: A boolean flag to determine the return type of the results. If True, the function will\n return a formatted string with the search results. If False, it will return a list of dictionaries\n containing detailed information about each search result.\n focus: Specific information to be focused on from each search result.\n\n Returns:\n The results of the search.\n \"\"\"\n loop = self.loop or asyncio.get_event_loop()\n future = loop.run_in_executor(\n self.executor, self.google_api_client.list(q=query, num=max_results, cx=self.google_cse_id).execute\n )\n try:\n result = await future\n # Extract the search result items from the response\n search_results = result.get(\"items\", [])\n\n except HttpError as e:\n # Handle errors in the API call\n logger.exception(f\"fail to search {query} for {e}\")\n search_results = []\n\n focus = focus or [\"snippet\", \"link\", \"title\"]\n details = [{i: j for i, j in item_dict.items() if i in focus} for item_dict in search_results]\n # Return the list of search result URLs\n if as_string:\n return safe_google_results(details)\n\n return details\n\n\ndef safe_google_results(results: str | list) -> str:\n \"\"\"Return the results of a google search in a safe format.\n\n Args:\n results: The search results.\n\n Returns:\n The results of the search.\n \"\"\"\n if isinstance(results, list):\n safe_message = json.dumps([result for result in results])\n else:\n safe_message = results.encode(\"utf-8\", \"ignore\").decode(\"utf-8\")\n return safe_message\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(GoogleAPIWrapper().run)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "Here is the Mermaid Sequence Diagram representation of the given code:\n\n```mermaid\nsequenceDiagram\n participant User\n participant GoogleAPIWrapper\n participant asyncio\n participant httplib2\n participant GoogleAPI\n\n User->>GoogleAPIWrapper: Construct object\n GoogleAPIWrapper->>GoogleAPIWrapper: check_google_api_key()\n GoogleAPIWrapper->>GoogleAPIWrapper: check_google_cse_id()\n GoogleAPIWrapper->>GoogleAPIWrapper: google_api_client()\n GoogleAPIWrapper->>User: Object with google_api_client\n\n User->>GoogleAPIWrapper: run(query, max_results, as_string, focus)\n GoogleAPIWrapper->>asyncio: get_event_loop()\n asyncio->>GoogleAPIWrapper: loop\n GoogleAPIWrapper->>GoogleAPIWrapper: executor\n GoogleAPIWrapper->>loop: run_in_executor()\n loop->>GoogleAPIWrapper: future\n GoogleAPIWrapper->>GoogleAPI: google_api_client.list()\n GoogleAPI-->GoogleAPIWrapper: Response\n GoogleAPIWrapper->>GoogleAPIWrapper: Extract search results\n GoogleAPIWrapper->>User: Search results\n```\n\nPlease note that this diagram represents the flow of the code and the interactions between different components. It does not include all the details of the code implementation.", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/5/2 16:03\n@Author : mashenquan\n@File : openapi_v3_hello.py\n@Desc : Implement the OpenAPI Specification 3.0 demo and use the following command to test the HTTP service:\n\n curl -X 'POST' \\\n 'http://localhost:8082/openapi/greeting/dave' \\\n -H 'accept: text/plain' \\\n -H 'Content-Type: application/json' \\\n -d '{}'\n\"\"\"\nfrom pathlib import Path\n\nimport connexion\n\n\n# openapi implement\nasync def post_greeting(name: str) -> str:\n return f\"Hello {name}\\n\"\n\n\nif __name__ == \"__main__\":\n specification_dir = Path(__file__).parent.parent.parent / \"docs/.well-known\"\n app = connexion.AsyncApp(__name__, specification_dir=str(specification_dir))\n app.add_api(\"openapi.yaml\", arguments={\"title\": \"Hello World Example\"})\n app.run(port=8082)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "To translate the given Python code into a Mermaid Sequence Diagram, we need to understand the flow of the code and identify the interactions between different components. Here's the translation:\n\n```mermaid\nsequenceDiagram\n participant User\n participant HTTPService\n participant OpenAPIImplementation\n participant Connexion\n\n User->>HTTPService: HTTP POST Request\n HTTPService->>OpenAPIImplementation: Call post_greeting(name)\n OpenAPIImplementation->>Connexion: Return \"Hello {name}\"\n Connexion->>HTTPService: Return \"Hello {name}\"\n HTTPService->>User: HTTP Response\n```\n\nIn the diagram above, the User sends an HTTP POST request to the HTTPService. The HTTPService then calls the `post_greeting` function in the OpenAPIImplementation. The OpenAPIImplementation returns the greeting message to Connexion, which in turn returns the message to the HTTPService. Finally, the HTTPService sends the HTTP response back to the User.\n\nNote that this diagram represents the high-level interactions between the components and does not include all the internal details of the code.", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n# -*- coding: utf-8 -*-\n# @Date : 2023/7/19 16:28\n# @Author : stellahong (stellahong@deepwisdom.ai)\n# @Desc :\nimport asyncio\nimport base64\nimport io\nimport json\nfrom os.path import join\nfrom typing import List\n\nfrom aiohttp import ClientSession\nfrom PIL import Image, PngImagePlugin\n\nfrom metagpt.config import CONFIG\nfrom metagpt.const import SD_OUTPUT_FILE_REPO\nfrom metagpt.logs import logger\n\npayload = {\n \"prompt\": \"\",\n \"negative_prompt\": \"(easynegative:0.8),black, dark,Low resolution\",\n \"override_settings\": {\"sd_model_checkpoint\": \"galaxytimemachinesGTM_photoV20\"},\n \"seed\": -1,\n \"batch_size\": 1,\n \"n_iter\": 1,\n \"steps\": 20,\n \"cfg_scale\": 7,\n \"width\": 512,\n \"height\": 768,\n \"restore_faces\": False,\n \"tiling\": False,\n \"do_not_save_samples\": False,\n \"do_not_save_grid\": False,\n \"enable_hr\": False,\n \"hr_scale\": 2,\n \"hr_upscaler\": \"Latent\",\n \"hr_second_pass_steps\": 0,\n \"hr_resize_x\": 0,\n \"hr_resize_y\": 0,\n \"hr_upscale_to_x\": 0,\n \"hr_upscale_to_y\": 0,\n \"truncate_x\": 0,\n \"truncate_y\": 0,\n \"applied_old_hires_behavior_to\": None,\n \"eta\": None,\n \"sampler_index\": \"DPM++ SDE Karras\",\n \"alwayson_scripts\": {},\n}\n\ndefault_negative_prompt = \"(easynegative:0.8),black, dark,Low resolution\"\n\n\nclass SDEngine:\n def __init__(self):\n # Initialize the SDEngine with configuration\n self.sd_url = CONFIG.get(\"SD_URL\")\n self.sd_t2i_url = f\"{self.sd_url}{CONFIG.get('SD_T2I_API')}\"\n # Define default payload settings for SD API\n self.payload = payload\n logger.info(self.sd_t2i_url)\n\n def construct_payload(\n self,\n prompt,\n negtive_prompt=default_negative_prompt,\n width=512,\n height=512,\n sd_model=\"galaxytimemachinesGTM_photoV20\",\n ):\n # Configure the payload with provided inputs\n self.payload[\"prompt\"] = prompt\n self.payload[\"negtive_prompt\"] = negtive_prompt\n self.payload[\"width\"] = width\n self.payload[\"height\"] = height\n self.payload[\"override_settings\"][\"sd_model_checkpoint\"] = sd_model\n logger.info(f\"call sd payload is {self.payload}\")\n return self.payload\n\n def _save(self, imgs, save_name=\"\"):\n save_dir = CONFIG.workspace_path / SD_OUTPUT_FILE_REPO\n if not save_dir.exists():\n save_dir.mkdir(parents=True, exist_ok=True)\n batch_decode_base64_to_image(imgs, str(save_dir), save_name=save_name)\n\n async def run_t2i(self, prompts: List):\n # Asynchronously run the SD API for multiple prompts\n session = ClientSession()\n for payload_idx, payload in enumerate(prompts):\n results = await self.run(url=self.sd_t2i_url, payload=payload, session=session)\n self._save(results, save_name=f\"output_{payload_idx}\")\n await session.close()\n\n async def run(self, url, payload, session):\n # Perform the HTTP POST request to the SD API\n async with session.post(url, json=payload, timeout=600) as rsp:\n data = await rsp.read()\n\n rsp_json = json.loads(data)\n imgs = rsp_json[\"images\"]\n logger.info(f\"callback rsp json is {rsp_json.keys()}\")\n return imgs\n\n async def run_i2i(self):\n # todo: 添加图生图接口调用\n raise NotImplementedError\n\n async def run_sam(self):\n # todo:添加SAM接口调用\n raise NotImplementedError\n\n\ndef decode_base64_to_image(img, save_name):\n image = Image.open(io.BytesIO(base64.b64decode(img.split(\",\", 1)[0])))\n pnginfo = PngImagePlugin.PngInfo()\n logger.info(save_name)\n image.save(f\"{save_name}.png\", pnginfo=pnginfo)\n return pnginfo, image\n\n\ndef batch_decode_base64_to_image(imgs, save_dir=\"\", save_name=\"\"):\n for idx, _img in enumerate(imgs):\n save_name = join(save_dir, save_name)\n decode_base64_to_image(_img, save_name=save_name)\n\n\nif __name__ == \"__main__\":\n engine = SDEngine()\n prompt = \"pixel style, game design, a game interface should be minimalistic and intuitive with the score and high score displayed at the top. The snake and its food should be easily distinguishable. The game should have a simple color scheme, with a contrasting color for the snake and its food. Complete interface boundary\"\n\n engine.construct_payload(prompt)\n\n event_loop = asyncio.get_event_loop()\n event_loop.run_until_complete(engine.run_t2i(prompt))\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant SDEngine\n participant ClientSession\n participant SD_API\n participant Image\n participant PngImagePlugin\n participant io\n participant base64\n participant json\n\n SDEngine->>+SDEngine: Initialize SDEngine with configuration\n SDEngine-->>-ClientSession: Create a ClientSession\n SDEngine->>+ClientSession: Send POST request to SD API\n ClientSession->>+SD_API: POST /sd_t2i_api\n ClientSession-->>-SD_API: Payload\n SD_API->>-SD_API: Process the request\n SD_API-->>-ClientSession: Response\n ClientSession->>-ClientSession: Close the session\n SDEngine->>+SDEngine: Save the images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-ClientSession: Response\n ClientSession-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/6/5 01:44\n@Author : alexanderwu\n@File : skill_manager.py\n@Modified By: mashenquan, 2023/8/20. Remove useless `llm`\n\"\"\"\nfrom metagpt.actions import Action\nfrom metagpt.const import PROMPT_PATH\nfrom metagpt.document_store.chromadb_store import ChromaStore\nfrom metagpt.logs import logger\n\nSkill = Action\n\n\nclass SkillManager:\n \"\"\"Used to manage all skills\"\"\"\n\n def __init__(self):\n self._store = ChromaStore(\"skill_manager\")\n self._skills: dict[str:Skill] = {}\n\n def add_skill(self, skill: Skill):\n \"\"\"\n Add a skill, add the skill to the skill pool and searchable storage\n :param skill: Skill\n :return:\n \"\"\"\n self._skills[skill.name] = skill\n self._store.add(skill.desc, {\"name\": skill.name, \"desc\": skill.desc}, skill.name)\n\n def del_skill(self, skill_name: str):\n \"\"\"\n Delete a skill, remove the skill from the skill pool and searchable storage\n :param skill_name: Skill name\n :return:\n \"\"\"\n self._skills.pop(skill_name)\n self._store.delete(skill_name)\n\n def get_skill(self, skill_name: str) -> Skill:\n \"\"\"\n Obtain a specific skill by skill name\n :param skill_name: Skill name\n :return: Skill\n \"\"\"\n return self._skills.get(skill_name)\n\n def retrieve_skill(self, desc: str, n_results: int = 2) -> list[Skill]:\n \"\"\"\n Obtain skills through the search engine\n :param desc: Skill description\n :return: Multiple skills\n \"\"\"\n return self._store.search(desc, n_results=n_results)[\"ids\"][0]\n\n def retrieve_skill_scored(self, desc: str, n_results: int = 2) -> dict:\n \"\"\"\n Obtain skills through the search engine\n :param desc: Skill description\n :return: Dictionary consisting of skills and scores\n \"\"\"\n return self._store.search(desc, n_results=n_results)\n\n def generate_skill_desc(self, skill: Skill) -> str:\n \"\"\"\n Generate descriptive text for each skill\n :param skill:\n :return:\n \"\"\"\n path = PROMPT_PATH / \"generate_skill.md\"\n text = path.read_text()\n logger.info(text)\n\n\nif __name__ == \"__main__\":\n manager = SkillManager()\n manager.generate_skill_desc(Action())\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "The Mermaid Sequence Diagram for the given code can be represented as follows:\n\n```mermaid\nsequenceDiagram\n participant SkillManager\n participant ChromaStore\n participant Skill\n participant Action\n\n SkillManager -> ChromaStore: add_skill(skill)\n ChromaStore -> SkillManager: add(skill.desc, {\"name\": skill.name, \"desc\": skill.desc}, skill.name)\n SkillManager -> Skill: skill.name, skill.desc\n Skill -> SkillManager: skill\n SkillManager -> ChromaStore: delete(skill_name)\n ChromaStore -> SkillManager: delete(skill_name)\n SkillManager -> Skill: skill_name\n Skill -> SkillManager: None\n SkillManager -> Skill: skill_name\n Skill -> SkillManager: skill\n SkillManager -> ChromaStore: search(desc, n_results)\n ChromaStore -> SkillManager: Multiple skills\n SkillManager -> ChromaStore: search(desc, n_results)\n ChromaStore -> SkillManager: Dictionary consisting of skills and scores\n SkillManager -> PROMPT_PATH: read_text()\n PROMPT_PATH -> SkillManager: text\n```\n\nNote: The `PROMPT_PATH` is not defined in the given code, so it is assumed to be a constant or variable that represents a file path.", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n\"\"\"\n@Modified By: mashenquan, 2023/8/22. A definition has been provided for the return value of _think: returning false indicates that further reasoning cannot continue.\n@Modified By: mashenquan, 2023-11-1. According to Chapter 2.2.1 and 2.2.2 of RFC 116, change the data type of\n the `cause_by` value in the `Message` to a string to support the new message distribution feature.\n\"\"\"\n\nimport asyncio\nimport re\n\nfrom pydantic import BaseModel\n\nfrom metagpt.actions import Action, CollectLinks, ConductResearch, WebBrowseAndSummarize\nfrom metagpt.actions.research import get_research_system_text\nfrom metagpt.const import RESEARCH_PATH\nfrom metagpt.logs import logger\nfrom metagpt.roles.role import Role, RoleReactMode\nfrom metagpt.schema import Message\n\n\nclass Report(BaseModel):\n topic: str\n links: dict[str, list[str]] = None\n summaries: list[tuple[str, str]] = None\n content: str = \"\"\n\n\nclass Researcher(Role):\n name: str = \"David\"\n profile: str = \"Researcher\"\n goal: str = \"Gather information and conduct research\"\n constraints: str = \"Ensure accuracy and relevance of information\"\n language: str = \"en-us\"\n\n def __init__(self, **kwargs):\n super().__init__(**kwargs)\n self._init_actions(\n [CollectLinks(name=self.name), WebBrowseAndSummarize(name=self.name), ConductResearch(name=self.name)]\n )\n self._set_react_mode(react_mode=RoleReactMode.BY_ORDER.value)\n if self.language not in (\"en-us\", \"zh-cn\"):\n logger.warning(f\"The language `{self.language}` has not been tested, it may not work.\")\n\n async def _think(self) -> bool:\n if self.rc.todo is None:\n self._set_state(0)\n return True\n\n if self.rc.state + 1 < len(self.states):\n self._set_state(self.rc.state + 1)\n else:\n self.rc.todo = None\n return False\n\n async def _act(self) -> Message:\n logger.info(f\"{self._setting}: to do {self.rc.todo}({self.rc.todo.name})\")\n todo = self.rc.todo\n msg = self.rc.memory.get(k=1)[0]\n if isinstance(msg.instruct_content, Report):\n instruct_content = msg.instruct_content\n topic = instruct_content.topic\n else:\n topic = msg.content\n\n research_system_text = self.research_system_text(topic, todo)\n if isinstance(todo, CollectLinks):\n links = await todo.run(topic, 4, 4)\n ret = Message(\n content=\"\", instruct_content=Report(topic=topic, links=links), role=self.profile, cause_by=todo\n )\n elif isinstance(todo, WebBrowseAndSummarize):\n links = instruct_content.links\n todos = (todo.run(*url, query=query, system_text=research_system_text) for (query, url) in links.items())\n summaries = await asyncio.gather(*todos)\n summaries = list((url, summary) for i in summaries for (url, summary) in i.items() if summary)\n ret = Message(\n content=\"\", instruct_content=Report(topic=topic, summaries=summaries), role=self.profile, cause_by=todo\n )\n else:\n summaries = instruct_content.summaries\n summary_text = \"\\n---\\n\".join(f\"url: {url}\\nsummary: {summary}\" for (url, summary) in summaries)\n content = await self.rc.todo.run(topic, summary_text, system_text=research_system_text)\n ret = Message(\n content=\"\",\n instruct_content=Report(topic=topic, content=content),\n role=self.profile,\n cause_by=self.rc.todo,\n )\n self.rc.memory.add(ret)\n return ret\n\n def research_system_text(self, topic, current_task: Action) -> str:\n \"\"\"BACKWARD compatible\n This allows sub-class able to define its own system prompt based on topic.\n return the previous implementation to have backward compatible\n Args:\n topic:\n language:\n\n Returns: str\n \"\"\"\n return get_research_system_text(topic, self.language)\n\n async def react(self) -> Message:\n msg = await super().react()\n report = msg.instruct_content\n self.write_report(report.topic, report.content)\n return msg\n\n def write_report(self, topic: str, content: str):\n filename = re.sub(r'[\\\\/:\"*?<>|]+', \" \", topic)\n filename = filename.replace(\"\\n\", \"\")\n if not RESEARCH_PATH.exists():\n RESEARCH_PATH.mkdir(parents=True)\n filepath = RESEARCH_PATH / f\"{filename}.md\"\n filepath.write_text(content)\n\n\nif __name__ == \"__main__\":\n import fire\n\n async def main(topic: str, language=\"en-us\"):\n role = Researcher(language=language)\n await role.run(topic)\n\n fire.Fire(main)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "Here is the Mermaid Sequence Diagram representation of the provided code:\n\n```mermaid\nsequenceDiagram\n participant Researcher\n participant Action\n participant CollectLinks\n participant WebBrowseAndSummarize\n participant ConductResearch\n participant Message\n participant Report\n\n Researcher->>Action: Initialize actions\n Researcher->>Action: Set react mode\n Researcher->>Researcher: Check language compatibility\n Researcher->>Researcher: Think\n Researcher->>Action: Act\n Action->>Message: Get message from memory\n Message-->>Action: Return message\n Action->>Researcher: Act on message\n Researcher->>CollectLinks: Run CollectLinks action\n CollectLinks->>CollectLinks: Collect links\n CollectLinks-->>Researcher: Return links\n Researcher->>Message: Create Report message\n Message-->>Researcher: Return Report message\n Researcher->>WebBrowseAndSummarize: Run WebBrowseAndSummarize action\n WebBrowseAndSummarize->>WebBrowseAndSummarize: Browse and summarize links\n WebBrowseAndSummarize-->>Researcher: Return summaries\n Researcher->>Message: Create Report message\n Message-->>Researcher: Return Report message\n Researcher->>ConductResearch: Run ConductResearch action\n ConductResearch->>ConductResearch: Conduct research\n ConductResearch-->>Researcher: Return research content\n Researcher->>Message: Create Report message\n Message-->>Researcher: Return Report message\n Researcher->>Researcher: Add message to memory\n Researcher->>Researcher: Think\n Researcher->>Action: Act\n Action->>Message: Get message from memory\n Message-->>Action: Return message\n Action->>Researcher: Act on message\n Researcher->>Researcher: Write report\n Researcher->>Message: Return message\n```\n\nPlease note that this is a simplified representation of the code logic and may not include all the details.", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/12/14 11:40\n@Author : alexanderwu\n@File : write_prd_an.py\n\"\"\"\nfrom typing import List\n\nfrom metagpt.actions.action_node import ActionNode\nfrom metagpt.logs import logger\n\nLANGUAGE = ActionNode(\n key=\"Language\",\n expected_type=str,\n instruction=\"Provide the language used in the project, typically matching the user's requirement language.\",\n example=\"en_us\",\n)\n\nPROGRAMMING_LANGUAGE = ActionNode(\n key=\"Programming Language\",\n expected_type=str,\n instruction=\"Python/JavaScript or other mainstream programming language.\",\n example=\"Python\",\n)\n\nORIGINAL_REQUIREMENTS = ActionNode(\n key=\"Original Requirements\",\n expected_type=str,\n instruction=\"Place the original user's requirements here.\",\n example=\"Create a 2048 game\",\n)\n\nPROJECT_NAME = ActionNode(\n key=\"Project Name\",\n expected_type=str,\n instruction=\"According to the content of \\\"Original Requirements,\\\" name the project using snake case style , like 'game_2048' or 'simple_crm.\",\n example=\"game_2048\",\n)\n\nPRODUCT_GOALS = ActionNode(\n key=\"Product Goals\",\n expected_type=List[str],\n instruction=\"Provide up to three clear, orthogonal product goals.\",\n example=[\"Create an engaging user experience\", \"Improve accessibility, be responsive\", \"More beautiful UI\"],\n)\n\nUSER_STORIES = ActionNode(\n key=\"User Stories\",\n expected_type=List[str],\n instruction=\"Provide up to 3 to 5 scenario-based user stories.\",\n example=[\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\",\n ],\n)\n\nCOMPETITIVE_ANALYSIS = ActionNode(\n key=\"Competitive Analysis\",\n expected_type=List[str],\n instruction=\"Provide 5 to 7 competitive products.\",\n example=[\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\",\n ],\n)\n\nCOMPETITIVE_QUADRANT_CHART = ActionNode(\n key=\"Competitive Quadrant Chart\",\n expected_type=str,\n instruction=\"Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\",\n example=\"\"\"quadrantChart\n title \"Reach and engagement of campaigns\"\n x-axis \"Low Reach\" --> \"High Reach\"\n y-axis \"Low Engagement\" --> \"High Engagement\"\n quadrant-1 \"We should expand\"\n quadrant-2 \"Need to promote\"\n quadrant-3 \"Re-evaluate\"\n quadrant-4 \"May be improved\"\n \"Campaign A\": [0.3, 0.6]\n \"Campaign B\": [0.45, 0.23]\n \"Campaign C\": [0.57, 0.69]\n \"Campaign D\": [0.78, 0.34]\n \"Campaign E\": [0.40, 0.34]\n \"Campaign F\": [0.35, 0.78]\n \"Our Target Product\": [0.5, 0.6]\"\"\",\n)\n\nREQUIREMENT_ANALYSIS = ActionNode(\n key=\"Requirement Analysis\",\n expected_type=str,\n instruction=\"Provide a detailed analysis of the requirements.\",\n example=\"\",\n)\n\nREQUIREMENT_POOL = ActionNode(\n key=\"Requirement Pool\",\n expected_type=List[List[str]],\n instruction=\"List down the top-5 requirements with their priority (P0, P1, P2).\",\n example=[[\"P0\", \"The main code ...\"], [\"P0\", \"The game algorithm ...\"]],\n)\n\nUI_DESIGN_DRAFT = ActionNode(\n key=\"UI Design draft\",\n expected_type=str,\n instruction=\"Provide a simple description of UI elements, functions, style, and layout.\",\n example=\"Basic function description with a simple style and layout.\",\n)\n\nANYTHING_UNCLEAR = ActionNode(\n key=\"Anything UNCLEAR\",\n expected_type=str,\n instruction=\"Mention any aspects of the project that are unclear and try to clarify them.\",\n example=\"\",\n)\n\nISSUE_TYPE = ActionNode(\n key=\"issue_type\",\n expected_type=str,\n instruction=\"Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement\",\n example=\"BUG\",\n)\n\nIS_RELATIVE = ActionNode(\n key=\"is_relative\",\n expected_type=str,\n instruction=\"Answer YES/NO. If the requirement is related to the old PRD, answer YES, otherwise NO\",\n example=\"YES\",\n)\n\nREASON = ActionNode(\n key=\"reason\", expected_type=str, instruction=\"Explain the reasoning process from question to answer\", example=\"...\"\n)\n\n\nNODES = [\n LANGUAGE,\n PROGRAMMING_LANGUAGE,\n ORIGINAL_REQUIREMENTS,\n PROJECT_NAME,\n PRODUCT_GOALS,\n USER_STORIES,\n COMPETITIVE_ANALYSIS,\n COMPETITIVE_QUADRANT_CHART,\n REQUIREMENT_ANALYSIS,\n REQUIREMENT_POOL,\n UI_DESIGN_DRAFT,\n ANYTHING_UNCLEAR,\n]\n\nWRITE_PRD_NODE = ActionNode.from_children(\"WritePRD\", NODES)\nWP_ISSUE_TYPE_NODE = ActionNode.from_children(\"WP_ISSUE_TYPE\", [ISSUE_TYPE, REASON])\nWP_IS_RELATIVE_NODE = ActionNode.from_children(\"WP_IS_RELATIVE\", [IS_RELATIVE, REASON])\n\n\ndef main():\n prompt = WRITE_PRD_NODE.compile(context=\"\")\n logger.info(prompt)\n\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant Language\n participant Programming Language\n participant Original Requirements\n participant Project Name\n participant Product Goals\n participant User Stories\n participant Competitive Analysis\n participant Competitive Quadrant Chart\n participant Requirement Analysis\n participant Requirement Pool\n participant UI Design draft\n participant Anything UNCLEAR\n participant issue_type\n participant is_relative\n participant reason\n participant WritePRD\n participant WP_ISSUE_TYPE\n participant WP_IS_RELATIVE\n \n WritePRD->>Language: Provide the language used in the project, typically matching the user's requirement language.\n WritePRD->>Programming Language: Python/JavaScript or other mainstream programming language.\n WritePRD->>Original Requirements: Place the original user's requirements here.\n WritePRD->>Project Name: According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n WritePRD->>Product Goals: Provide up to three clear, orthogonal product goals.\n WritePRD->>User Stories: Provide up to 3 to 5 scenario-based user stories.\n WritePRD->>Competitive Analysis: Provide 5 to 7 competitive products.\n WritePRD->>Competitive Quadrant Chart: Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n WritePRD->>Requirement Analysis: Provide a detailed analysis of the requirements.\n WritePRD->>Requirement Pool: List down the top-5 requirements with their priority (P0, P1, P2).\n WritePRD->>UI Design draft: Provide a simple description of UI elements, functions, style, and layout.\n WritePRD->>Anything UNCLEAR: Mention any aspects of the project that are unclear and try to clarify them.\n \n WritePRD->>WP_ISSUE_TYPE: Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement\n WP_ISSUE_TYPE->>issue_type: BUG\n WP_ISSUE_TYPE->>reason: Explain the reasoning process from question to answer\n \n WritePRD->>WP_IS_RELATIVE: Answer YES/NO. If the requirement is related to the old PRD, answer YES, otherwise NO\n WP_IS_RELATIVE->>is_relative: YES\n WP_IS_RELATIVE->>reason: Explain the reasoning process from question to answer\n \n WritePRD-->>Language: expected_type: str\n WritePRD-->>Programming Language: expected_type: str\n WritePRD-->>Original Requirements: expected_type: str\n WritePRD-->>Project Name: expected_type: str\n WritePRD-->>Product Goals: expected_type: List[str]\n WritePRD-->>User Stories: expected_type: List[str]\n WritePRD-->>Competitive Analysis: expected_type: List[str]\n WritePRD-->>Competitive Quadrant Chart: expected_type: str\n WritePRD-->>Requirement Analysis: expected_type: str\n WritePRD-->>Requirement Pool: expected_type: List[List[str]]\n WritePRD-->>UI Design draft: expected_type: str\n WritePRD-->>Anything UNCLEAR: expected_type: str\n \n WP_ISSUE_TYPE-->>issue_type: expected_type: str\n WP_ISSUE_TYPE-->>reason: expected_type: str\n \n WP_IS_RELATIVE-->>is_relative: expected_type: str\n WP_IS_RELATIVE-->>reason: expected_type: str\n```", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n\"\"\"Code Docstring Generator.\n\nThis script provides a tool to automatically generate docstrings for Python code. It uses the specified style to create\ndocstrings for the given code and system text.\n\nUsage:\n python3 -m metagpt.actions.write_docstring [--overwrite] [--style=]\n\nArguments:\n filename The path to the Python file for which you want to generate docstrings.\n\nOptions:\n --overwrite If specified, overwrite the original file with the code containing docstrings.\n --style= Specify the style of the generated docstrings.\n Valid values: 'google', 'numpy', or 'sphinx'.\n Default: 'google'\n\nExample:\n python3 -m metagpt.actions.write_docstring ./metagpt/startup.py --overwrite False --style=numpy\n\nThis script uses the 'fire' library to create a command-line interface. It generates docstrings for the given Python code using\nthe specified docstring style and adds them to the code.\n\"\"\"\nfrom __future__ import annotations\n\nimport ast\nfrom pathlib import Path\nfrom typing import Literal, Optional\n\nfrom metagpt.actions.action import Action\nfrom metagpt.utils.common import OutputParser, aread, awrite\nfrom metagpt.utils.pycst import merge_docstring\n\nPYTHON_DOCSTRING_SYSTEM = \"\"\"### Requirements\n1. Add docstrings to the given code following the {style} style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\n{example}\n```\n\"\"\"\n\n# https://www.sphinx-doc.org/en/master/usage/extensions/napoleon.html\n\nPYTHON_DOCSTRING_EXAMPLE_GOOGLE = '''\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Args:\n param1: The first parameter.\n\n Returns:\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Args:\n msg: Human readable string describing the exception.\n\n Attributes:\n msg: Human readable string describing the exception.\n \"\"\"\n ...\n'''\n\nPYTHON_DOCSTRING_EXAMPLE_NUMPY = '''\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"\n Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Parameters\n ----------\n param1\n The first parameter.\n\n Returns\n -------\n bool\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"\n Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Parameters\n ----------\n msg\n Human readable string describing the exception.\n\n Attributes\n ----------\n msg\n Human readable string describing the exception.\n \"\"\"\n ...\n'''\n\nPYTHON_DOCSTRING_EXAMPLE_SPHINX = '''\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n :param param1: The first parameter.\n :type param1: int\n\n :return: The return value. True for success, False otherwise.\n :rtype: bool\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n :param msg: Human-readable string describing the exception.\n :type msg: str\n \"\"\"\n ...\n'''\n\n_python_docstring_style = {\n \"google\": PYTHON_DOCSTRING_EXAMPLE_GOOGLE.strip(),\n \"numpy\": PYTHON_DOCSTRING_EXAMPLE_NUMPY.strip(),\n \"sphinx\": PYTHON_DOCSTRING_EXAMPLE_SPHINX.strip(),\n}\n\n\nclass WriteDocstring(Action):\n \"\"\"This class is used to write docstrings for code.\n\n Attributes:\n desc: A string describing the action.\n \"\"\"\n\n desc: str = \"Write docstring for code.\"\n context: Optional[str] = None\n\n async def run(\n self,\n code: str,\n system_text: str = PYTHON_DOCSTRING_SYSTEM,\n style: Literal[\"google\", \"numpy\", \"sphinx\"] = \"google\",\n ) -> str:\n \"\"\"Writes docstrings for the given code and system text in the specified style.\n\n Args:\n code: A string of Python code.\n system_text: A string of system text.\n style: A string specifying the style of the docstring. Can be 'google', 'numpy', or 'sphinx'.\n\n Returns:\n The Python code with docstrings added.\n \"\"\"\n system_text = system_text.format(style=style, example=_python_docstring_style[style])\n simplified_code = _simplify_python_code(code)\n documented_code = await self._aask(f\"```python\\n{simplified_code}\\n```\", [system_text])\n documented_code = OutputParser.parse_python_code(documented_code)\n return merge_docstring(code, documented_code)\n\n @staticmethod\n async def write_docstring(\n filename: str | Path, overwrite: bool = False, style: Literal[\"google\", \"numpy\", \"sphinx\"] = \"google\"\n ) -> str:\n data = await aread(str(filename))\n code = await WriteDocstring().run(data, style=style)\n if overwrite:\n await awrite(filename, code)\n return code\n\n\ndef _simplify_python_code(code: str) -> None:\n \"\"\"Simplifies the given Python code by removing expressions and the last if statement.\n\n Args:\n code: A string of Python code.\n\n Returns:\n The simplified Python code.\n \"\"\"\n code_tree = ast.parse(code)\n code_tree.body = [i for i in code_tree.body if not isinstance(i, ast.Expr)]\n if isinstance(code_tree.body[-1], ast.If):\n code_tree.body.pop()\n return ast.unparse(code_tree)\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(WriteDocstring.write_docstring)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant User\n participant Script\n participant Action\n participant OutputParser\n participant ast\n participant fire\n\n User->>Script: Run script with arguments\n Script->>Action: Call run() method\n Action->>OutputParser: Parse system text\n Action->>ast: Parse code into AST\n Action->>Action: Simplify code\n Action->>Action: Generate system text\n Action->>OutputParser: Parse documented code\n Action->>Action: Merge docstrings\n Action->>Script: Return code with docstrings\n Script->>fire: Call write_docstring() method\n fire->>Action: Call write_docstring() method\n Action->>OutputParser: Parse code from file\n Action->>Action: Run run() method\n Action->>Action: Write docstrings\n Action->>OutputParser: Parse code with docstrings\n Action->>Script: Return code with docstrings\n Script->>User: Return code with docstrings\n```\n```", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Author : alexanderwu\n@File : write_review.py\n\"\"\"\nimport asyncio\nfrom typing import List\n\nfrom metagpt.actions import Action\nfrom metagpt.actions.action_node import ActionNode\n\nREVIEW = ActionNode(\n key=\"Review\",\n expected_type=List[str],\n instruction=\"Act as an experienced reviewer and critically assess the given output. Provide specific and\"\n \" constructive feedback, highlighting areas for improvement and suggesting changes.\",\n example=[\n \"The logic in the function `calculate_total` seems flawed. Shouldn't it consider the discount rate as well?\",\n \"The TODO function is not implemented yet? Should we implement it before commit?\",\n ],\n)\n\nLGTM = ActionNode(\n key=\"LGTM\",\n expected_type=str,\n instruction=\"LGTM/LBTM. If the code is fully implemented, \"\n \"give a LGTM (Looks Good To Me), otherwise provide a LBTM (Looks Bad To Me).\",\n example=\"LBTM\",\n)\n\nACTIONS = ActionNode(\n key=\"Actions\",\n expected_type=str,\n instruction=\"Based on the code review outcome, suggest actionable steps. This can include code changes, \"\n \"refactoring suggestions, or any follow-up tasks.\",\n example=\"\"\"1. Refactor the `process_data` method to improve readability and efficiency.\n2. Cover edge cases in the `validate_user` function.\n3. Implement a the TODO in the `calculate_total` function.\n4. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n\"\"\",\n)\n\nWRITE_DRAFT = ActionNode(\n key=\"WriteDraft\",\n expected_type=str,\n instruction=\"Could you write draft code for move function in order to implement it?\",\n example=\"Draft: ...\",\n)\n\n\nWRITE_MOVE_FUNCTION = ActionNode(\n key=\"WriteFunction\",\n expected_type=str,\n instruction=\"write code for the function not implemented.\",\n example=\"\"\"\n```Code\n...\n```\n\"\"\",\n)\n\n\nREWRITE_CODE = ActionNode(\n key=\"RewriteCode\",\n expected_type=str,\n instruction=\"\"\"rewrite code based on the Review and Actions\"\"\",\n example=\"\"\"\n```python\n## example.py\ndef calculate_total(price, quantity):\n total = price * quantity\n```\n\"\"\",\n)\n\n\nCODE_REVIEW_CONTEXT = \"\"\"\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\n\n# Context\n## System Design\n{\"Implementation approach\": \"我们将使用HTML、CSS和JavaScript来实现这个单机的响应式2048游戏。为了确保游戏性能流畅和响应式设计,我们会选择使用Vue.js框架,因为它易于上手且适合构建交互式界面。我们还将使用localStorage来记录玩家的最高分。\", \"File list\": [\"index.html\", \"styles.css\", \"main.js\", \"game.js\", \"storage.js\"], \"Data structures and interfaces\": \"classDiagram\\\n class Game {\\\n -board Array\\\n -score Number\\\n -bestScore Number\\\n +constructor()\\\n +startGame()\\\n +move(direction: String)\\\n +getBoard() Array\\\n +getScore() Number\\\n +getBestScore() Number\\\n +setBestScore(score: Number)\\\n }\\\n class Storage {\\\n +getBestScore() Number\\\n +setBestScore(score: Number)\\\n }\\\n class Main {\\\n +init()\\\n +bindEvents()\\\n }\\\n Game --> Storage : uses\\\n Main --> Game : uses\", \"Program call flow\": \"sequenceDiagram\\\n participant M as Main\\\n participant G as Game\\\n participant S as Storage\\\n M->>G: init()\\\n G->>S: getBestScore()\\\n S-->>G: return bestScore\\\n M->>G: bindEvents()\\\n M->>G: startGame()\\\n loop Game Loop\\\n M->>G: move(direction)\\\n G->>S: setBestScore(score)\\\n S-->>G: return\\\n end\", \"Anything UNCLEAR\": \"目前项目要求明确,没有不清楚的地方。\"}\n\n## Tasks\n{\"Required Python packages\": [\"无需Python包\"], \"Required Other language third-party packages\": [\"vue.js\"], \"Logic Analysis\": [[\"index.html\", \"作为游戏的入口文件和主要的HTML结构\"], [\"styles.css\", \"包含所有的CSS样式,确保游戏界面美观\"], [\"main.js\", \"包含Main类,负责初始化游戏和绑定事件\"], [\"game.js\", \"包含Game类,负责游戏逻辑,如开始游戏、移动方块等\"], [\"storage.js\", \"包含Storage类,用于获取和设置玩家的最高分\"]], \"Task list\": [\"index.html\", \"styles.css\", \"storage.js\", \"game.js\", \"main.js\"], \"Full API spec\": \"\", \"Shared Knowledge\": \"\\'game.js\\' 包含游戏逻辑相关的函数,被 \\'main.js\\' 调用。\", \"Anything UNCLEAR\": \"目前项目要求明确,没有不清楚的地方。\"}\n\n## Code Files\n----- index.html\n\n\n\n \n \n 2048游戏\n \n \n\n\n

\n\n \n \n \n \n\n\n\n----- styles.css\n/* styles.css */\nbody, html {\n margin: 0;\n padding: 0;\n font-family: \\'Arial\\', sans-serif;\n}\n\n#app {\n text-align: center;\n font-size: 18px;\n color: #776e65;\n}\n\nh1 {\n color: #776e65;\n font-size: 72px;\n font-weight: bold;\n margin: 20px 0;\n}\n\n.scores-container {\n display: flex;\n justify-content: center;\n margin-bottom: 20px;\n}\n\n.score-container, .best-container {\n background: #bbada0;\n padding: 10px;\n border-radius: 5px;\n margin: 0 10px;\n min-width: 100px;\n text-align: center;\n}\n\n.score-header, .best-header {\n color: #eee4da;\n font-size: 18px;\n margin-bottom: 5px;\n}\n\n.game-container {\n max-width: 500px;\n margin: 0 auto 20px;\n background: #bbada0;\n padding: 15px;\n border-radius: 10px;\n position: relative;\n}\n\n.grid-row {\n display: flex;\n}\n\n.grid-cell {\n background: #cdc1b4;\n width: 100px;\n height: 100px;\n margin: 5px;\n display: flex;\n justify-content: center;\n align-items: center;\n font-size: 35px;\n font-weight: bold;\n color: #776e65;\n border-radius: 3px;\n}\n\n/* Dynamic classes for different number cells */\n.number-cell-2 {\n background: #eee4da;\n}\n\n.number-cell-4 {\n background: #ede0c8;\n}\n\n.number-cell-8 {\n background: #f2b179;\n color: #f9f6f2;\n}\n\n.number-cell-16 {\n background: #f59563;\n color: #f9f6f2;\n}\n\n.number-cell-32 {\n background: #f67c5f;\n color: #f9f6f2;\n}\n\n.number-cell-64 {\n background: #f65e3b;\n color: #f9f6f2;\n}\n\n.number-cell-128 {\n background: #edcf72;\n color: #f9f6f2;\n}\n\n.number-cell-256 {\n background: #edcc61;\n color: #f9f6f2;\n}\n\n.number-cell-512 {\n background: #edc850;\n color: #f9f6f2;\n}\n\n.number-cell-1024 {\n background: #edc53f;\n color: #f9f6f2;\n}\n\n.number-cell-2048 {\n background: #edc22e;\n color: #f9f6f2;\n}\n\n/* Larger numbers need smaller font sizes */\n.number-cell-1024, .number-cell-2048 {\n font-size: 30px;\n}\n\nbutton {\n background-color: #8f7a66;\n color: #f9f6f2;\n border: none;\n border-radius: 3px;\n padding: 10px 20px;\n font-size: 18px;\n cursor: pointer;\n outline: none;\n}\n\nbutton:hover {\n background-color: #9f8b76;\n}\n\n----- storage.js\n## storage.js\nclass Storage {\n // 获取最高分\n getBestScore() {\n // 尝试从localStorage中获取最高分,如果不存在则默认为0\n const bestScore = localStorage.getItem(\\'bestScore\\');\n return bestScore ? Number(bestScore) : 0;\n }\n\n // 设置最高分\n setBestScore(score) {\n // 将最高分设置到localStorage中\n localStorage.setItem(\\'bestScore\\', score.toString());\n }\n}\n\n\n\n## Code to be Reviewed: game.js\n```Code\n## game.js\nclass Game {\n constructor() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = 0;\n }\n\n createEmptyBoard() {\n const board = [];\n for (let i = 0; i < 4; i++) {\n board[i] = [0, 0, 0, 0];\n }\n return board;\n }\n\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.addRandomTile();\n this.addRandomTile();\n }\n\n addRandomTile() {\n let emptyCells = [];\n for (let r = 0; r < 4; r++) {\n for (let c = 0; c < 4; c++) {\n if (this.board[r][c] === 0) {\n emptyCells.push({ r, c });\n }\n }\n }\n if (emptyCells.length > 0) {\n let randomCell = emptyCells[Math.floor(Math.random() * emptyCells.length)];\n this.board[randomCell.r][randomCell.c] = Math.random() < 0.9 ? 2 : 4;\n }\n }\n\n move(direction) {\n // This function will handle the logic for moving tiles\n // in the specified direction and merging them\n // It will also update the score and add a new random tile if the move is successful\n // The actual implementation of this function is complex and would require\n // a significant amount of code to handle all the cases for moving and merging tiles\n // For the purposes of this example, we will not implement the full logic\n // Instead, we will just call addRandomTile to simulate a move\n this.addRandomTile();\n }\n\n getBoard() {\n return this.board;\n }\n\n getScore() {\n return this.score;\n }\n\n getBestScore() {\n return this.bestScore;\n }\n\n setBestScore(score) {\n this.bestScore = score;\n }\n}\n\n```\n\"\"\"\n\n\nCODE_REVIEW_SMALLEST_CONTEXT = \"\"\"\n## Code to be Reviewed: game.js\n```Code\n// game.js\nclass Game {\n constructor() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = 0;\n }\n\n createEmptyBoard() {\n const board = [];\n for (let i = 0; i < 4; i++) {\n board[i] = [0, 0, 0, 0];\n }\n return board;\n }\n\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.addRandomTile();\n this.addRandomTile();\n }\n\n addRandomTile() {\n let emptyCells = [];\n for (let r = 0; r < 4; r++) {\n for (let c = 0; c < 4; c++) {\n if (this.board[r][c] === 0) {\n emptyCells.push({ r, c });\n }\n }\n }\n if (emptyCells.length > 0) {\n let randomCell = emptyCells[Math.floor(Math.random() * emptyCells.length)];\n this.board[randomCell.r][randomCell.c] = Math.random() < 0.9 ? 2 : 4;\n }\n }\n\n move(direction) {\n // This function will handle the logic for moving tiles\n // in the specified direction and merging them\n // It will also update the score and add a new random tile if the move is successful\n // The actual implementation of this function is complex and would require\n // a significant amount of code to handle all the cases for moving and merging tiles\n // For the purposes of this example, we will not implement the full logic\n // Instead, we will just call addRandomTile to simulate a move\n this.addRandomTile();\n }\n\n getBoard() {\n return this.board;\n }\n\n getScore() {\n return this.score;\n }\n\n getBestScore() {\n return this.bestScore;\n }\n\n setBestScore(score) {\n this.bestScore = score;\n }\n}\n\n```\n\"\"\"\n\n\nCODE_REVIEW_SAMPLE = \"\"\"\n## Code Review: game.js\n1. The code partially implements the requirements. The `Game` class is missing the full implementation of the `move` method, which is crucial for the game\\'s functionality.\n2. The code logic is not completely correct. The `move` method is not implemented, which means the game cannot process player moves.\n3. The existing code follows the \"Data structures and interfaces\" in terms of class structure but lacks full method implementations.\n4. Not all functions are implemented. The `move` method is incomplete and does not handle the logic for moving and merging tiles.\n5. All necessary pre-dependencies seem to be imported since the code does not indicate the need for additional imports.\n6. The methods from other files (such as `Storage`) are not being used in the provided code snippet, but the class structure suggests that they will be used correctly.\n\n## Actions\n1. Implement the `move` method to handle tile movements and merging. This is a complex task that requires careful consideration of the game\\'s rules and logic. Here is a simplified version of how one might begin to implement the `move` method:\n ```javascript\n move(direction) {\n // Simplified logic for moving tiles up\n if (direction === \\'up\\') {\n for (let col = 0; col < 4; col++) {\n let tiles = this.board.map(row => row[col]).filter(val => val !== 0);\n let merged = [];\n for (let i = 0; i < tiles.length; i++) {\n if (tiles[i] === tiles[i + 1]) {\n tiles[i] *= 2;\n this.score += tiles[i];\n tiles[i + 1] = 0;\n merged.push(i);\n }\n }\n tiles = tiles.filter(val => val !== 0);\n while (tiles.length < 4) {\n tiles.push(0);\n }\n for (let row = 0; row < 4; row++) {\n this.board[row][col] = tiles[row];\n }\n }\n }\n // Additional logic needed for \\'down\\', \\'left\\', \\'right\\'\n // ...\n this.addRandomTile();\n }\n ```\n2. Integrate the `Storage` class methods to handle the best score. This means updating the `startGame` and `setBestScore` methods to use `Storage` for retrieving and setting the best score:\n ```javascript\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = new Storage().getBestScore(); // Retrieve the best score from storage\n this.addRandomTile();\n this.addRandomTile();\n }\n\n setBestScore(score) {\n if (score > this.bestScore) {\n this.bestScore = score;\n new Storage().setBestScore(score); // Set the new best score in storage\n }\n }\n ```\n\n## Code Review Result\nLBTM\n\n```\n\"\"\"\n\n\nWRITE_CODE_NODE = ActionNode.from_children(\"WRITE_REVIEW_NODE\", [REVIEW, LGTM, ACTIONS])\nWRITE_MOVE_NODE = ActionNode.from_children(\"WRITE_MOVE_NODE\", [WRITE_DRAFT, WRITE_MOVE_FUNCTION])\n\n\nCR_FOR_MOVE_FUNCTION_BY_3 = \"\"\"\nThe move function implementation provided appears to be well-structured and follows a clear logic for moving and merging tiles in the specified direction. However, there are a few potential improvements that could be made to enhance the code:\n\n1. Encapsulation: The logic for moving and merging tiles could be encapsulated into smaller, reusable functions to improve readability and maintainability.\n\n2. Magic Numbers: There are some magic numbers (e.g., 4, 3) used in the loops that could be replaced with named constants for improved readability and easier maintenance.\n\n3. Comments: Adding comments to explain the logic and purpose of each section of the code can improve understanding for future developers who may need to work on or maintain the code.\n\n4. Error Handling: It's important to consider error handling for unexpected input or edge cases to ensure the function behaves as expected in all scenarios.\n\nOverall, the code could benefit from refactoring to improve readability, maintainability, and extensibility. If you would like, I can provide a refactored version of the move function that addresses these considerations.\n\"\"\"\n\n\nclass WriteCodeAN(Action):\n \"\"\"Write a code review for the context.\"\"\"\n\n async def run(self, context):\n self.llm.system_prompt = \"You are an outstanding engineer and can implement any code\"\n return await WRITE_MOVE_FUNCTION.fill(context=context, llm=self.llm, schema=\"json\")\n # return await WRITE_CODE_NODE.fill(context=context, llm=self.llm, schema=\"markdown\")\n\n\nasync def main():\n await WriteCodeAN().run(CODE_REVIEW_SMALLEST_CONTEXT)\n\n\nif __name__ == \"__main__\":\n asyncio.run(main())\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant User\n participant CodeReview\n participant WriteCodeAN\n participant WRITE_MOVE_FUNCTION\n\n User->>CodeReview: Request code review\n CodeReview->>WriteCodeAN: Run WriteCodeAN action\n WriteCodeAN->>WRITE_MOVE_FUNCTION: Fill WRITE_MOVE_FUNCTION node\n WRITE_MOVE_FUNCTION-->>WriteCodeAN: Return filled node\n WriteCodeAN-->>CodeReview: Return filled node\n CodeReview-->>User: Return code review result\n```\n", - "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/12/14 15:28\n@Author : alexanderwu\n@File : project_management_an.py\n\"\"\"\nfrom typing import List\n\nfrom metagpt.actions.action_node import ActionNode\nfrom metagpt.logs import logger\n\nREQUIRED_PYTHON_PACKAGES = ActionNode(\n key=\"Required Python packages\",\n expected_type=List[str],\n instruction=\"Provide required Python packages in requirements.txt format.\",\n example=[\"flask==1.1.2\", \"bcrypt==3.2.0\"],\n)\n\nREQUIRED_OTHER_LANGUAGE_PACKAGES = ActionNode(\n key=\"Required Other language third-party packages\",\n expected_type=List[str],\n instruction=\"List down the required packages for languages other than Python.\",\n example=[\"No third-party dependencies required\"],\n)\n\nLOGIC_ANALYSIS = ActionNode(\n key=\"Logic Analysis\",\n expected_type=List[List[str]],\n instruction=\"Provide a list of files with the classes/methods/functions to be implemented, \"\n \"including dependency analysis and imports.\",\n example=[\n [\"game.py\", \"Contains Game class and ... functions\"],\n [\"main.py\", \"Contains main function, from game import Game\"],\n ],\n)\n\nTASK_LIST = ActionNode(\n key=\"Task list\",\n expected_type=List[str],\n instruction=\"Break down the tasks into a list of filenames, prioritized by dependency order.\",\n example=[\"game.py\", \"main.py\"],\n)\n\nFULL_API_SPEC = ActionNode(\n key=\"Full API spec\",\n expected_type=str,\n instruction=\"Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end \"\n \"and back-end communication is not required, leave it blank.\",\n example=\"openapi: 3.0.0 ...\",\n)\n\nSHARED_KNOWLEDGE = ActionNode(\n key=\"Shared Knowledge\",\n expected_type=str,\n instruction=\"Detail any shared knowledge, like common utility functions or configuration variables.\",\n example=\"'game.py' contains functions shared across the project.\",\n)\n\nANYTHING_UNCLEAR_PM = ActionNode(\n key=\"Anything UNCLEAR\",\n expected_type=str,\n instruction=\"Mention any unclear aspects in the project management context and try to clarify them.\",\n example=\"Clarification needed on how to start and initialize third-party libraries.\",\n)\n\nNODES = [\n REQUIRED_PYTHON_PACKAGES,\n REQUIRED_OTHER_LANGUAGE_PACKAGES,\n LOGIC_ANALYSIS,\n TASK_LIST,\n FULL_API_SPEC,\n SHARED_KNOWLEDGE,\n ANYTHING_UNCLEAR_PM,\n]\n\n\nPM_NODE = ActionNode.from_children(\"PM_NODE\", NODES)\n\n\ndef main():\n prompt = PM_NODE.compile(context=\"\")\n logger.info(prompt)\n\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant ActionNode\n participant PM_NODE\n participant main\n\n main->>PM_NODE: compile(context=\"\")\n PM_NODE->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>Action", - "You are a function parser. You can convert spoken words into function parameters.\n\n---\ntext_to_image function parameters description:\nparameter `text`: The text used for image conversion.\nparameter `size_type`: size type\n\n---\nExamples:\nIf want you to do `Draw a girl`, return `text_to_image(text=\"Draw a girl\", size_type=\"512x512\")` brief and clear.\nIf want you to do `Draw an apple`, return `text_to_image(text=\"Draw an apple\", size_type=\"512x512\")` brief and clear.\n\n---\n\nRefer to the `text_to_image` function description, and fill in the function parameters according to the example \"I want you to do xx\" in the Examples section.\nNow I want you to do `Draw an apple`, return function parameters in Examples format above, brief and clear.": "`text_to_image(text=\"Draw an apple\", size_type=\"512x512\")`", - "\nNOTICE\nRole: You are a professional software engineer, and your main task is to review the code.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n-----\n# System Design\n```text\n\n{\"Implementation approach\": \"To develop this snake game, we will use the Python language and choose the Pygame library. Pygame is an open-source Python module collection specifically designed for writing video games. It provides functionalities such as displaying images and playing sounds, making it suitable for creating intuitive and responsive user interfaces. We will ensure efficient game logic to prevent any delays during gameplay. The scoring system will be simple, with the snake gaining points for each food it eats. We will use Pygame's event handling system to implement pause and resume functionality, as well as high-score tracking. The difficulty will increase by speeding up the snake's movement. In the initial version, we will focus on single-player mode and consider adding multiplayer mode and customizable skins in future updates. Based on the new requirement, we will also add a moving obstacle that appears randomly. If the snake eats this obstacle, the game will end. If the snake does not eat the obstacle, it will disappear after 5 seconds. For this, we need to add mechanisms for obstacle generation, movement, and disappearance in the game logic.\", \"Project_name\": \"snake_game\", \"File list\": [\"main.py\", \"game.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"constants.py\", \"assets/styles.css\", \"assets/index.html\"], \"Data structures and interfaces\": \"```mermaid\n classDiagram\n class Game{\n +int score\n +int speed\n +bool game_over\n +bool paused\n +Snake snake\n +Food food\n +Obstacle obstacle\n +Scoreboard scoreboard\n +start_game() void\n +pause_game() void\n +resume_game() void\n +end_game() void\n +increase_difficulty() void\n +update() void\n +render() void\n Game()\n }\n class Snake{\n +list body_parts\n +str direction\n +bool grow\n +move() void\n +grow() void\n +check_collision() bool\n Snake()\n }\n class Food{\n +tuple position\n +spawn() void\n Food()\n }\n class Obstacle{\n +tuple position\n +int lifetime\n +bool active\n +spawn() void\n +move() void\n +check_collision() bool\n +disappear() void\n Obstacle()\n }\n class Scoreboard{\n +int high_score\n +update_score(int) void\n +reset_score() void\n +load_high_score() void\n +save_high_score() void\n Scoreboard()\n }\n class Constants{\n }\n Game \"1\" -- \"1\" Snake: has\n Game \"1\" -- \"1\" Food: has\n Game \"1\" -- \"1\" Obstacle: has\n Game \"1\" -- \"1\" Scoreboard: has\n ```\", \"Program call flow\": \"```sequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n participant O as Obstacle\n participant SB as Scoreboard\n M->>G: start_game()\n loop game loop\n G->>S: move()\n G->>S: check_collision()\n G->>F: spawn()\n G->>O: spawn()\n G->>O: move()\n G->>O: check_collision()\n G->>O: disappear()\n G->>SB: update_score(score)\n G->>G: update()\n G->>G: render()\n alt if paused\n M->>G: pause_game()\n M->>G: resume_game()\n end\n alt if game_over\n G->>M: end_game()\n end\n end\n```\", \"Anything UNCLEAR\": \"There is no need for further clarification as the requirements are already clear.\"}\n\n```\n-----\n# Tasks\n```text\n\n{\"Required Python third-party packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party packages required for other languages.\"], \"Full API spec\": \"\n openapi: 3.0.0\n info:\n title: Snake Game API\n version: \"1.0.0\"\n paths:\n /start:\n get:\n summary: Start the game\n responses:\n '200':\n description: Game started successfully\n /pause:\n get:\n summary: Pause the game\n responses:\n '200':\n description: Game paused successfully\n /resume:\n get:\n summary: Resume the game\n responses:\n '200':\n description: Game resumed successfully\n /end:\n get:\n summary: End the game\n responses:\n '200':\n description: Game ended successfully\n /score:\n get:\n summary: Get the current score\n responses:\n '200':\n description: Current score retrieved successfully\n /highscore:\n get:\n summary: Get the high score\n responses:\n '200':\n description: High score retrieved successfully\n components: {}\n \", \"Logic Analysis\": [[\"constants.py\", \"Contains all the constant values like screen size, colors, game speeds, etc. This should be implemented first as it provides the base values for other components.\"], [\"snake.py\", \"Contains the Snake class with methods for movement, growth, and collision detection. It is dependent on constants.py for configuration values.\"], [\"food.py\", \"Contains the Food class responsible for spawning food items on the screen. It is dependent on constants.py for configuration values.\"], [\"obstacle.py\", \"Contains the Obstacle class with methods for spawning, moving, and disappearing of obstacles, as well as collision detection with the snake. It is dependent on constants.py for configuration values.\"], [\"scoreboard.py\", \"Contains the Scoreboard class for updating, resetting, loading, and saving high scores. It may use constants.py for configuration values and depends on the game's scoring logic.\"], [\"game.py\", \"Contains the main Game class which includes the game loop and methods for starting, pausing, resuming, and ending the game. It is dependent on snake.py, food.py, obstacle.py, and scoreboard.py.\"], [\"main.py\", \"The entry point of the game that initializes the game and starts the game loop. It is dependent on game.py.\"]], \"Task list\": [\"constants.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"game.py\", \"main.py\"], \"Shared Knowledge\": \"\n 'constants.py' should contain all the necessary configurations for the game, such as screen dimensions, color definitions, and speed settings. These constants will be used across multiple files, ensuring consistency and ease of updates. Ensure that the Pygame library is initialized correctly in 'main.py' before starting the game loop. Also, make sure that the game's state is managed properly when pausing and resuming the game.\n \", \"Anything UNCLEAR\": \"The interaction between the 'obstacle.py' and the game loop needs to be clearly defined to ensure obstacles appear and disappear correctly. The lifetime of the obstacle and its random movement should be implemented in a way that does not interfere with the game's performance.\"}\n\n```\n-----\n```python\n\n## game.py\nimport pygame\nfrom snake import Snake\nfrom food import Food\n\nclass Game:\n def __init__(self):\n self.score = 0\n self.level = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n pygame.init()\n self.initialize_game()\n self.game_loop()\n\n def initialize_game(self):\n self.score = 0\n self.level = 1\n self.snake.reset()\n self.food.generate()\n\n def game_loop(self):\n game_over = False\n\n while not game_over:\n self.update()\n self.draw()\n self.handle_events()\n self.check_collision()\n self.increase_score()\n self.increase_level()\n\n if self.snake.is_collision():\n game_over = True\n self.game_over()\n\n def update(self):\n self.snake.move()\n\n def draw(self):\n self.snake.draw()\n self.food.draw()\n\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n pygame.quit()\n quit()\n elif event.type == pygame.KEYDOWN:\n if event.key == pygame.K_UP:\n self.snake.change_direction(\"UP\")\n elif event.key == pygame.K_DOWN:\n self.snake.change_direction(\"DOWN\")\n elif event.key == pygame.K_LEFT:\n self.snake.change_direction(\"LEFT\")\n elif event.key == pygame.K_RIGHT:\n self.snake.change_direction(\"RIGHT\")\n\n def check_collision(self):\n if self.snake.get_head() == self.food.get_position():\n self.snake.grow()\n self.food.generate()\n\n def increase_score(self):\n self.score += 1\n\n def increase_level(self):\n if self.score % 10 == 0:\n self.level += 1\n\n def game_over(self):\n print(\"Game Over\")\n self.initialize_game()\n\n\n```\n-----\n```python\n\n## snake.py\nimport pygame\n\nclass Snake:\n def __init__(self):\n self.body = [(0, 0)]\n self.direction = (1, 0)\n\n def move(self):\n head = self.body[0]\n dx, dy = self.direction\n new_head = (head[0] + dx, head[1] + dy)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction):\n if direction == \"UP\":\n self.direction = (0, -1)\n elif direction == \"DOWN\":\n self.direction = (0, 1)\n elif direction == \"LEFT\":\n self.direction = (-1, 0)\n elif direction == \"RIGHT\":\n self.direction = (1, 0)\n\n def grow(self):\n tail = self.body[-1]\n dx, dy = self.direction\n new_tail = (tail[0] - dx, tail[1] - dy)\n self.body.append(new_tail)\n\n def get_head(self):\n return self.body[0]\n\n def get_body(self):\n return self.body[1:]\n\n\n```\n-----\n```python\n\n## main.py\nimport pygame\nfrom game import Game\n\ndef main():\n pygame.init()\n game = Game()\n game.start_game()\n\nif __name__ == \"__main__\":\n main()\n\n\n```\n-----\n```python\n\n## food.py\nimport random\n\nclass Food:\n def __init__(self):\n self.position = (0, 0)\n\n def generate(self):\n x = random.randint(0, 9)\n y = random.randint(0, 9)\n self.position = (x, y)\n\n def get_position(self):\n return self.position\n\n\n```\n-----\n\n## Code Review All: Please read all historical files and find possible bugs in the files, such as unimplemented functions, calling errors, unreferences, etc.\n\n## Call flow: mermaid code, based on the implemented function, use mermaid to draw a complete call chain\n\n## Summary: Summary based on the implementation of historical files\n\n## TODOs: Python dict[str, str], write down the list of files that need to be modified and the reasons. We will modify them later.\n\n": "## Code Review\n\n### game.py\n- The `game_over` variable is not being used in the `game_loop` method. It can be removed.\n- The `increase_score` method should only increase the score if there is a collision with the food. Move the score increment logic inside the `check_collision` method.\n- The `increase_level` method should only increase the level if the score is divisible by 10. Move the level increment logic inside the `check_collision` method.\n- The `game_over` method should be called after the game loop ends, not inside the loop.\n\n### snake.py\n- The `get_body` method is not being used. It can be removed.\n\n### main.py\n- The `pygame.init()` method is being called twice, once in the `main` function and once in the `start_game` method of the `Game` class. It should only be called once, preferably in the `main` function.\n\n### food.py\n- The `generate` method should generate random positions within the game boundaries defined in the `constants.py` file.\n\n## Call Flow\n\n```mermaid\nsequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n\n M->>G: start_game()\n G->>G: initialize_game()\n G->>G: game_loop()\n loop game loop\n G->>S: move()\n G->>G: update()\n G->>S: draw()\n G->>F: draw()\n G->>G: handle_events()\n G->>G: check_collision()\n G->>G: increase_score()\n G->>G: increase_level()\n alt if snake collision\n G->>G: game_over()\n end\n end\n```\n\n## Summary\nThe code consists of several classes that work together to create a snake game. The `Game` class manages the game loop and handles events. The `Snake` class represents the snake and handles its movement and collision detection. The `Food` class represents the food and handles its generation and position. The code is missing some functionality, such as updating the score and level, and handling game over conditions. There are also some minor issues with the code structure and logic that need to be addressed.\n\n## TODOs\n- Modify `game.py`:\n - Remove the unused `game_over` variable in the `game_loop` method.\n - Move the score increment logic from the `increase_score` method to the `check_collision` method.\n - Move the level increment logic from the `increase_level` method to the `check_collision` method.\n - Call the `game_over` method after the game loop ends.\n- Modify `snake.py`:\n - Remove the unused `get_body` method.\n- Modify `main.py`:\n - Remove the redundant `pygame.init()` method call in the `start_game` method of the `Game` class.\n- Modify `food.py`:\n - Generate random positions within the game boundaries defined in the `constants.py` file.", - "\nNOTICE\nRole: You are a professional software engineer, and your main task is to review the code.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n-----\n# System Design\n```text\n\n{\"Implementation approach\": \"To develop this snake game, we will use the Python language and choose the Pygame library. Pygame is an open-source Python module collection specifically designed for writing video games. It provides functionalities such as displaying images and playing sounds, making it suitable for creating intuitive and responsive user interfaces. We will ensure efficient game logic to prevent any delays during gameplay. The scoring system will be simple, with the snake gaining points for each food it eats. We will use Pygame's event handling system to implement pause and resume functionality, as well as high-score tracking. The difficulty will increase by speeding up the snake's movement. In the initial version, we will focus on single-player mode and consider adding multiplayer mode and customizable skins in future updates. Based on the new requirement, we will also add a moving obstacle that appears randomly. If the snake eats this obstacle, the game will end. If the snake does not eat the obstacle, it will disappear after 5 seconds. For this, we need to add mechanisms for obstacle generation, movement, and disappearance in the game logic.\", \"Project_name\": \"snake_game\", \"File list\": [\"main.py\", \"game.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"constants.py\", \"assets/styles.css\", \"assets/index.html\"], \"Data structures and interfaces\": \"```mermaid\n classDiagram\n class Game{\n +int score\n +int speed\n +bool game_over\n +bool paused\n +Snake snake\n +Food food\n +Obstacle obstacle\n +Scoreboard scoreboard\n +start_game() void\n +pause_game() void\n +resume_game() void\n +end_game() void\n +increase_difficulty() void\n +update() void\n +render() void\n Game()\n }\n class Snake{\n +list body_parts\n +str direction\n +bool grow\n +move() void\n +grow() void\n +check_collision() bool\n Snake()\n }\n class Food{\n +tuple position\n +spawn() void\n Food()\n }\n class Obstacle{\n +tuple position\n +int lifetime\n +bool active\n +spawn() void\n +move() void\n +check_collision() bool\n +disappear() void\n Obstacle()\n }\n class Scoreboard{\n +int high_score\n +update_score(int) void\n +reset_score() void\n +load_high_score() void\n +save_high_score() void\n Scoreboard()\n }\n class Constants{\n }\n Game \"1\" -- \"1\" Snake: has\n Game \"1\" -- \"1\" Food: has\n Game \"1\" -- \"1\" Obstacle: has\n Game \"1\" -- \"1\" Scoreboard: has\n ```\", \"Program call flow\": \"```sequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n participant O as Obstacle\n participant SB as Scoreboard\n M->>G: start_game()\n loop game loop\n G->>S: move()\n G->>S: check_collision()\n G->>F: spawn()\n G->>O: spawn()\n G->>O: move()\n G->>O: check_collision()\n G->>O: disappear()\n G->>SB: update_score(score)\n G->>G: update()\n G->>G: render()\n alt if paused\n M->>G: pause_game()\n M->>G: resume_game()\n end\n alt if game_over\n G->>M: end_game()\n end\n end\n```\", \"Anything UNCLEAR\": \"There is no need for further clarification as the requirements are already clear.\"}\n\n```\n-----\n# Tasks\n```text\n\n{\"Required Python third-party packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party packages required for other languages.\"], \"Full API spec\": \"\n openapi: 3.0.0\n info:\n title: Snake Game API\n version: \"1.0.0\"\n paths:\n /start:\n get:\n summary: Start the game\n responses:\n '200':\n description: Game started successfully\n /pause:\n get:\n summary: Pause the game\n responses:\n '200':\n description: Game paused successfully\n /resume:\n get:\n summary: Resume the game\n responses:\n '200':\n description: Game resumed successfully\n /end:\n get:\n summary: End the game\n responses:\n '200':\n description: Game ended successfully\n /score:\n get:\n summary: Get the current score\n responses:\n '200':\n description: Current score retrieved successfully\n /highscore:\n get:\n summary: Get the high score\n responses:\n '200':\n description: High score retrieved successfully\n components: {}\n \", \"Logic Analysis\": [[\"constants.py\", \"Contains all the constant values like screen size, colors, game speeds, etc. This should be implemented first as it provides the base values for other components.\"], [\"snake.py\", \"Contains the Snake class with methods for movement, growth, and collision detection. It is dependent on constants.py for configuration values.\"], [\"food.py\", \"Contains the Food class responsible for spawning food items on the screen. It is dependent on constants.py for configuration values.\"], [\"obstacle.py\", \"Contains the Obstacle class with methods for spawning, moving, and disappearing of obstacles, as well as collision detection with the snake. It is dependent on constants.py for configuration values.\"], [\"scoreboard.py\", \"Contains the Scoreboard class for updating, resetting, loading, and saving high scores. It may use constants.py for configuration values and depends on the game's scoring logic.\"], [\"game.py\", \"Contains the main Game class which includes the game loop and methods for starting, pausing, resuming, and ending the game. It is dependent on snake.py, food.py, obstacle.py, and scoreboard.py.\"], [\"main.py\", \"The entry point of the game that initializes the game and starts the game loop. It is dependent on game.py.\"]], \"Task list\": [\"constants.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"game.py\", \"main.py\"], \"Shared Knowledge\": \"\n 'constants.py' should contain all the necessary configurations for the game, such as screen dimensions, color definitions, and speed settings. These constants will be used across multiple files, ensuring consistency and ease of updates. Ensure that the Pygame library is initialized correctly in 'main.py' before starting the game loop. Also, make sure that the game's state is managed properly when pausing and resuming the game.\n \", \"Anything UNCLEAR\": \"The interaction between the 'obstacle.py' and the game loop needs to be clearly defined to ensure obstacles appear and disappear correctly. The lifetime of the obstacle and its random movement should be implemented in a way that does not interfere with the game's performance.\"}\n\n```\n-----\n```python\n\n## food.py\nimport random\n\nclass Food:\n def __init__(self):\n self.position = (0, 0)\n\n def generate(self):\n x = random.randint(0, 9)\n y = random.randint(0, 9)\n self.position = (x, y)\n\n def get_position(self):\n return self.position\n\n\n```\n-----\n```python\n\n## snake.py\nimport pygame\n\nclass Snake:\n def __init__(self):\n self.body = [(0, 0)]\n self.direction = (1, 0)\n\n def move(self):\n head = self.body[0]\n dx, dy = self.direction\n new_head = (head[0] + dx, head[1] + dy)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction):\n if direction == \"UP\":\n self.direction = (0, -1)\n elif direction == \"DOWN\":\n self.direction = (0, 1)\n elif direction == \"LEFT\":\n self.direction = (-1, 0)\n elif direction == \"RIGHT\":\n self.direction = (1, 0)\n\n def grow(self):\n tail = self.body[-1]\n dx, dy = self.direction\n new_tail = (tail[0] - dx, tail[1] - dy)\n self.body.append(new_tail)\n\n def get_head(self):\n return self.body[0]\n\n def get_body(self):\n return self.body[1:]\n\n\n```\n-----\n```python\n\n## game.py\nimport pygame\nfrom snake import Snake\nfrom food import Food\n\nclass Game:\n def __init__(self):\n self.score = 0\n self.level = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n pygame.init()\n self.initialize_game()\n self.game_loop()\n\n def initialize_game(self):\n self.score = 0\n self.level = 1\n self.snake.reset()\n self.food.generate()\n\n def game_loop(self):\n game_over = False\n\n while not game_over:\n self.update()\n self.draw()\n self.handle_events()\n self.check_collision()\n self.increase_score()\n self.increase_level()\n\n if self.snake.is_collision():\n game_over = True\n self.game_over()\n\n def update(self):\n self.snake.move()\n\n def draw(self):\n self.snake.draw()\n self.food.draw()\n\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n pygame.quit()\n quit()\n elif event.type == pygame.KEYDOWN:\n if event.key == pygame.K_UP:\n self.snake.change_direction(\"UP\")\n elif event.key == pygame.K_DOWN:\n self.snake.change_direction(\"DOWN\")\n elif event.key == pygame.K_LEFT:\n self.snake.change_direction(\"LEFT\")\n elif event.key == pygame.K_RIGHT:\n self.snake.change_direction(\"RIGHT\")\n\n def check_collision(self):\n if self.snake.get_head() == self.food.get_position():\n self.snake.grow()\n self.food.generate()\n\n def increase_score(self):\n self.score += 1\n\n def increase_level(self):\n if self.score % 10 == 0:\n self.level += 1\n\n def game_over(self):\n print(\"Game Over\")\n self.initialize_game()\n\n\n```\n-----\n```python\n\n## main.py\nimport pygame\nfrom game import Game\n\ndef main():\n pygame.init()\n game = Game()\n game.start_game()\n\nif __name__ == \"__main__\":\n main()\n\n\n```\n-----\n\n## Code Review All: Please read all historical files and find possible bugs in the files, such as unimplemented functions, calling errors, unreferences, etc.\n\n## Call flow: mermaid code, based on the implemented function, use mermaid to draw a complete call chain\n\n## Summary: Summary based on the implementation of historical files\n\n## TODOs: Python dict[str, str], write down the list of files that need to be modified and the reasons. We will modify them later.\n\n": "## Code Review\n\n### food.py\n- The `generate` method should be renamed to `spawn` to be consistent with the naming convention used in other classes.\n- The `get_position` method can be simplified by directly returning the `self.position` attribute.\n\n### snake.py\n- The `change_direction` method should handle lowercase direction inputs as well to make it more flexible.\n- The `grow` method can be optimized by directly appending the new tail position to the `self.body` list instead of calculating it based on the current direction.\n\n### game.py\n- The `initialize_game` method should call the `reset` method of the `self.snake` object instead of a non-existent `reset` method.\n- The `game_over` method should reset the game state by calling the `initialize_game` method instead of manually resetting the score and level attributes.\n- The `draw` method should be implemented to draw the snake and food on the screen using the Pygame library.\n- The `handle_events` method should handle the case when the snake is moving in one direction and the opposite direction key is pressed, which should be ignored to prevent the snake from reversing.\n- The `check_collision` method should be implemented to check for collisions between the snake's head and the food, and handle the snake's growth and food respawn accordingly.\n- The `increase_score` and `increase_level` methods should be implemented to increase the score and level based on the game's scoring logic.\n- The `game_over` method should be implemented to handle the game over condition, display a game over message, and reset the game state.\n\n### main.py\n- The `pygame.init()` method is called twice, once in the `main` function and once in the `Game` class. It should only be called once, preferably in the `main` function.\n\n## Call Flow\n\n```mermaid\nsequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n\n M->>G: start_game()\n loop game loop\n G->>S: move()\n G->>S: check_collision()\n G->>F: spawn()\n G->>G: update()\n G->>G: draw()\n G->>G: handle_events()\n G->>G: check_collision()\n G->>G: increase_score()\n G->>G: increase_level()\n alt if game over\n G->>G: game_over()\n end\n end\n```\n\n## Summary\n\nThe code consists of several classes that represent different components of the snake game, such as the snake itself, the food, and the game logic. The `main` function initializes the game and starts the game loop. The game loop handles updating the game state, drawing the game objects on the screen, and handling user input. The snake can move in different directions, grow when it eats food, and collide with itself or the boundaries of the screen. The game keeps track of the score and level, and ends when the snake collides with itself or the boundaries.\n\n## TODOs\n\n- Modify `food.py`:\n - Rename the `generate` method to `spawn`.\n - Simplify the `get_position` method.\n- Modify `snake.py`:\n - Handle lowercase direction inputs in the `change_direction` method.\n - Optimize the `grow` method by directly appending the new tail position to the `self.body` list.\n- Modify `game.py`:\n - Call the `reset` method of the `self.snake` object in the `initialize_game` method.\n - Implement the `draw` method to draw the snake and food on the screen.\n - Implement the `handle_events` method to handle direction changes and ignore opposite direction inputs.\n - Implement the `check_collision` method to handle collisions between the snake's head and the food.\n - Implement the `increase_score` and `increase_level` methods based on the game's scoring logic.\n - Implement the `game_over` method to handle the game over condition and reset the game state.\n- Remove the duplicate `pygame.init()` call in the `main` function.", - "You are mathematician.#MSG_SEP#Your responses should align with the role-play agreement, maintaining the character's persona and habits. When faced with unrelated questions, playfully decline to answer without revealing your AI nature to preserve the character's image.#MSG_SEP#If the information is insufficient, you can search in the context or knowledge.#MSG_SEP#Answer the following questions strictly in English, and the answers must follow the Markdown format.#SYSTEM_MSG_END#How old is Susie?": "To determine Susie's age, we can subtract her birthdate from the current date. \n\nSusie was born on November 14, 2011, and today is December 3, 2023. \n\nLet's calculate the age:\n\n```\nCurrent year - Birth year = Age\n2023 - 2011 = 12\n```\n\nTherefore, Susie is currently 12 years old.", - "You are mathematician.#MSG_SEP#Your responses should align with the role-play agreement, maintaining the character's persona and habits. When faced with unrelated questions, playfully decline to answer without revealing your AI nature to preserve the character's image.#MSG_SEP#If the information is insufficient, you can search in the context or knowledge.#MSG_SEP#Answer the following questions strictly in Chinese, and the answers must follow the Markdown format.#SYSTEM_MSG_END#Does Susie have an apple?": "根据提供的信息,Susie曾经有一个苹果,但是她已经吃掉了。", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n设计一个名为'add'的函数,该函数接受两个整数作为输入,并返回它们的和。\n\n## Tasks\n\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: \n```python\n## \n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: . Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code:\n\n```python\ndef add(num1: int, num2: int) -> int:\n \"\"\"\n This function takes two integers as input and returns their sum.\n \n Args:\n num1 (int): The first integer.\n num2 (int): The second integer.\n \n Returns:\n int: The sum of the two integers.\n \"\"\"\n return num1 + num2\n```\n\n# Instruction: Based on the context, write code.\n\n## Code: \n```python\ndef add(num1: int, num2: int) -> int:\n \"\"\"\n This function takes two integers as input and returns their sum.\n \n Args:\n num1 (int): The first integer.\n num2 (int): The second integer.\n \n Returns:\n int: The sum of the two integers.\n \"\"\"\n return num1 + num2\n```", - "\n你是一个工程师。下面是背景信息与你的当前任务,请为任务撰写代码。\n撰写的代码应该符合PEP8,优雅,模块化,易于阅读与维护,代码本身应该有__main__入口来防止桩函数\n\n## 用户编写程序所需的全部、详尽的文件路径列表(只需要相对路径,并不需要前缀,组织形式应该符合PEP规范)\n\n- `main.py`: 主程序文件\n- `search_engine.py`: 搜索引擎实现文件\n- `knowledge_base.py`: 知识库管理文件\n- `user_interface.py`: 用户界面文件\n- `data_import.py`: 数据导入功能文件\n- `data_export.py`: 数据导出功能文件\n- `utils.py`: 工具函数文件\n\n## 数据结构\n\n- `KnowledgeBase`: 知识库类,用于管理私有知识库的内容、分类、标签和关键词。\n- `SearchEngine`: 搜索引擎类,基于大语言模型,用于对用户输入的关键词或短语进行语义理解,并提供准确的搜索结果。\n- `SearchResult`: 搜索结果类,包含与用户搜索意图相关的知识库内容的相关信息。\n- `UserInterface`: 用户界面类,提供简洁、直观的用户界面,支持多种搜索方式和搜索结果的排序和过滤。\n- `DataImporter`: 数据导入类,支持多种数据格式的导入功能,用于将外部数据导入到知识库中。\n- `DataExporter`: 数据导出类,支持多种数据格式的导出功能,用于将知识库内容进行备份和分享。\n\n## API接口\n\n- `KnowledgeBase`类接口:\n - `add_entry(entry: str, category: str, tags: List[str], keywords: List[str]) -> bool`: 添加知识库条目。\n - `delete_entry(entry_id: str) -> bool`: 删除知识库条目。\n - `update_entry(entry_id: str, entry: str, category: str, tags: List[str], keywords: List[str]) -> bool`: 更新知识库条目。\n - `search_entries(query: str) -> List[str]`: 根据查询词搜索知识库条目。\n\n- `SearchEngine`类接口:\n - `search(query: str) -> SearchResult`: 根据用户查询词进行搜索,返回与查询意图相关的搜索结果。\n\n- `UserInterface`类接口:\n - `display_search_results(results: List[SearchResult]) -> None`: 显示搜索结果。\n - `filter_results(results: List[SearchResult], filters: Dict[str, Any]) -> List[SearchResult]`: 根据过滤条件对搜索结果进行过滤。\n - `sort_results(results: List[SearchResult], key: str, reverse: bool = False) -> List[SearchResult]`: 根据指定的键对搜索结果进行排序。\n\n- `DataImporter`类接口:\n - `import_data(file_path: str) -> bool`: 导入外部数据到知识库。\n\n- `DataExporter`类接口:\n - `export_data(file_path: str) -> bool`: 导出知识库数据到外部文件。\n\n## 调用流程(以dot语言描述)\n\n```dot\ndigraph call_flow {\n rankdir=LR;\n\n subgraph cluster_user_program {\n label=\"User Program\";\n style=dotted;\n\n main_py -> search_engine_py;\n main_py -> knowledge_base_py;\n main_py -> user_interface_py;\n main_py -> data_import_py;\n main_py -> data_export_py;\n\n search_engine_py -> knowledge_base_py;\n search_engine_py -> user_interface_py;\n\n user_interface_py -> knowledge_base_py;\n user_interface_py -> search_engine_py;\n\n data_import_py -> knowledge_base_py;\n data_import_py -> user_interface_py;\n\n data_export_py -> knowledge_base_py;\n data_export_py -> user_interface_py;\n }\n\n main_py [label=\"main.py\"];\n search_engine_py [label=\"search_engine.py\"];\n knowledge_base_py [label=\"knowledge_base.py\"];\n user_interface_py [label=\"user_interface.py\"];\n data_import_py [label=\"data_import.py\"];\n data_export_py [label=\"data_export.py\"];\n}\n```\n\n这是一个简化的调用流程图,展示了各个模块之间的调用关系。用户程序的`main.py`文件通过调用其他模块实现搜索引擎的功能。`search_engine.py`模块与`knowledge_base.py`和`user_interface.py`模块进行交互,实现搜索算法和搜索结果的展示。`data_import.py`和`data_export.py`模块与`knowledge_base.py`和`user_interface.py`模块进行交互,实现数据导入和导出的功能。用户界面模块`user_interface.py`与其他模块进行交互,提供简洁、直观的用户界面,并支持搜索方式、排序和过滤等操作。\n\n## 当前任务\n\n\n完成main.py的功能": "根据任务要求,我们需要在`main.py`文件中完成以下功能:\n\n1. 导入所需模块和类:`KnowledgeBase`、`SearchEngine`、`UserInterface`、`DataImporter`、`DataExporter`。\n2. 创建一个`KnowledgeBase`对象,用于管理知识库的内容。\n3. 创建一个`SearchEngine`对象,用于进行搜索操作。\n4. 创建一个`UserInterface`对象,用于与用户进行交互。\n5. 创建一个`DataImporter`对象,用于导入外部数据到知识库。\n6. 创建一个`DataExporter`对象,用于导出知识库数据到外部文件。\n7. 实现主程序逻辑,包括用户界面的显示和用户输入的处理。\n\n下面是一个示例的`main.py`代码实现:\n\n```python\nfrom knowledge_base import KnowledgeBase\nfrom search_engine import SearchEngine\nfrom user_interface import UserInterface\nfrom data_import import DataImporter\nfrom data_export import DataExporter\n\ndef main():\n # 创建知识库对象\n knowledge_base = KnowledgeBase()\n\n # 创建搜索引擎对象\n search_engine = SearchEngine(knowledge_base)\n\n # 创建用户界面对象\n user_interface = UserInterface(search_engine)\n\n # 创建数据导入对象\n data_importer = DataImporter(knowledge_base)\n\n # 创建数据导出对象\n data_exporter = DataExporter(knowledge_base)\n\n # 主程序逻辑\n while True:\n # 显示用户界面\n user_interface.display_menu()\n\n # 获取用户输入\n user_input = input(\"请输入选项:\")\n\n # 处理用户输入\n if user_input == \"1\":\n # 搜索知识库\n query = input(\"请输入搜索关键词:\")\n search_results = search_engine.search(query)\n user_interface.display_search_results(search_results)\n elif user_input == \"2\":\n # 导入数据\n file_path = input(\"请输入要导入的文件路径:\")\n data_importer.import_data(file_path)\n elif user_input == \"3\":\n # 导出数据\n file_path = input(\"请输入要导出的文件路径:\")\n data_exporter.export_data(file_path)\n elif user_input == \"4\":\n # 退出程序\n break\n else:\n print(\"无效的选项,请重新输入。\")\n\nif __name__ == \"__main__\":\n main()\n```\n\n在这个示例代码中,我们首先导入了所需的模块和类。然后,我们创建了`KnowledgeBase`、`SearchEngine`、`UserInterface`、`DataImporter`和`DataExporter`对象。接下来,我们使用一个无限循环来实现主程序逻辑。在每次循环中,我们首先显示用户界面,然后获取用户输入,并根据用户输入执行相应的操作。用户可以选择搜索知识库、导入数据、导出数据或退出程序。最后,我们使用`if __name__ == \"__main__\":`来确保代码在作为主程序运行时才会执行。", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\": \"We will use the Pygame library to create the game interface and handle user input. The game logic will be implemented using Python classes and data structures.\", \"File list\": [\"main.py\", \"game.py\"], \"Data structures and interfaces\": \"classDiagram\\n class Game {\\n -grid: List[List[int]]\\n -score: int\\n -game_over: bool\\n +__init__()\\n +reset_game()\\n +move(direction: str)\\n +is_game_over() bool\\n +get_empty_cells() List[Tuple[int, int]]\\n +add_new_tile()\\n +get_score() int\\n }\\n class UI {\\n -game: Game\\n +__init__(game: Game)\\n +draw_grid()\\n +draw_score()\\n +draw_game_over()\\n +handle_input()\\n }\\n Game --> UI\", \"Program call flow\": \"sequenceDiagram\\n participant M as Main\\n participant G as Game\\n participant U as UI\\n M->>G: reset_game()\\n M->>U: draw_grid()\\n M->>U: draw_score()\\n M->>U: handle_input()\\n U->>G: move(direction)\\n G->>G: add_new_tile()\\n G->>U: draw_grid()\\n G->>U: draw_score()\\n G->>U: draw_game_over()\\n G->>G: is_game_over()\\n G->>G: get_empty_cells()\\n G->>G: get_score()\", \"Anything UNCLEAR\": \"...\"}\n\n## Tasks\n{\"Required Python packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party dependencies required\"], \"Logic Analysis\": [[\"game.py\", \"Contains Game class and related functions for game logic\"], [\"main.py\", \"Contains main function, initializes the game and UI\"]], \"Task list\": [\"game.py\", \"main.py\"], \"Full API spec\": \"\", \"Shared Knowledge\": \"The game logic will be implemented using Python classes and data structures. The Pygame library will be used to create the game interface and handle user input.\", \"Anything UNCLEAR\": \"...\"}\n\n## Legacy Code\n```Code\n----- main.py\nif __name__ == \"__main__\":\nmain()\n```\n\n## Debug logs\n```text\nE.......F\n======================================================================\nERROR: test_add_new_tile (__main__.TestGame)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n File \"/Users/xx/tests/test_game.py\", line 104, in test_add_new_tile\n self.assertIn(self.game.grid[empty_cells[0][0]][empty_cells[0][1]], [2, 4])\nIndexError: list index out of range\n\n======================================================================\nFAIL: test_reset_game (__main__.TestGame)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n File \"/Users/xx/tests/test_game.py\", line 13, in test_reset_game\n self.assertEqual(self.game.grid, [[0 for _ in range(4)] for _ in range(4)])\nAssertionError: Lists differ: [[0, 0, 0, 0], [0, 2, 0, 0], [0, 0, 0, 2], [0, 0, 0, 0]] != [[0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0]]\n\nFirst differing element 1:\n[0, 2, 0, 0]\n[0, 0, 0, 0]\n\n- [[0, 0, 0, 0], [0, 2, 0, 0], [0, 0, 0, 2], [0, 0, 0, 0]]\n? --- ^\n\n+ [[0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0]]\n? +++ ^\n\n\n----------------------------------------------------------------------\nRan 9 tests in 0.002s\n\nFAILED (failures=1, errors=1)\n\n\n{\"design_filename\": \"docs/system_design/20231221155954.json\", \"task_filename\": \"docs/tasks/20231221155954.json\", \"codes_filenames\": [\"game.py\", \"main.py\"], \"reason\": \"```json\\n{\\n \\\"game.py\\\": \\\"Add handling for no empty cells in add_new_tile function, Update score in move function\\\",\\n \\\"main.py\\\": \\\"Handle game over condition in the game loop\\\"\\n}\\n```\"}\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: game.py\n\n```python\nclass Game:\n def __init__(self):\n self.grid = [[0 for _ in range(4)] for _ in range(4)]\n self.score = 0\n self.game_over = False\n\n def reset_game(self):\n self.grid = [[0 for _ in range(4)] for _ in range(4)]\n self.score = 0\n self.game_over = False\n\n def move(self, direction: str):\n # TODO: Implement the move logic based on the given direction\n pass\n\n def is_game_over(self) -> bool:\n # TODO: Implement the game over logic\n pass\n\n def get_empty_cells(self) -> List[Tuple[int, int]]:\n # TODO: Return a list of coordinates of empty cells in the grid\n pass\n\n def add_new_tile(self):\n # TODO: Add a new tile (either 2 or 4) to a random empty cell in the grid\n pass\n\n def get_score(self) -> int:\n return self.score\n```\n\n## Explanation:\nIn the `Game` class, we have implemented the constructor `__init__` to initialize the grid, score, and game_over attributes. The `reset_game` method is used to reset the game by setting the grid, score, and game_over attributes to their initial values.\n\nThe `move` method is not implemented yet and will handle the logic for moving the tiles in the grid based on the given direction.\n\nThe `is_game_over` method is not implemented yet and will check if the game is over by checking if there are any possible moves left.\n\nThe `get_empty_cells` method is not implemented yet and will return a list of coordinates of empty cells in the grid.\n\nThe `add_new_tile` method is not implemented yet and will add a new tile (either 2 or 4) to a random empty cell in the grid.\n\nThe `get_score` method returns the current score of the game.", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n编写一个从a加b的函数,返回a+b\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef add(a, b):\n return a + \n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. No, the code is not implemented as per the requirements. The `add` function is missing the second parameter `b` in the return statement. To fix it, we need to include `b` in the return statement as well. The corrected code should be:\n```python\ndef add(a, b):\n return a + b\n```\n\n## Actions\n1. Implement the fix mentioned above.\n\n## Code Review Result\nLBTM", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n编写一个从a加b的函数,返回a+b\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef add(a, b):\n return a + \n\n```\n\n## Code Review: \n1. No, the code is not implemented as per the requirements. The `add` function is missing the second parameter `b` in the return statement. To fix it, we need to include `b` in the return statement as well. The corrected code should be:\n```python\ndef add(a, b):\n return a + b\n```\n\n## Actions\n1. Implement the fix mentioned above.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## \n...\n```\n": "```python\ndef add(a, b):\n return a + b\n```", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n编写一个从a加b的函数,返回a+b\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\ndef add(a, b):\n return a + b\n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM", - "### Requirements\n1. Add docstrings to the given code following the google style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Args:\n param1: The first parameter.\n\n Returns:\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Args:\n msg: Human readable string describing the exception.\n\n Attributes:\n msg: Human readable string describing the exception.\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\ndef add_numbers(a: int, b: int):\n return a + b\n\nclass Person:\n\n def __init__(self, name: str, age: int):\n self.name = name\n self.age = age\n\n def greet(self):\n return f'Hello, my name is {self.name} and I am {self.age} years old.'\n```": "```python\ndef add_numbers(a: int, b: int):\n \"\"\"Add two numbers.\n\n Args:\n a: The first number.\n b: The second number.\n\n Returns:\n The sum of the two numbers.\n \"\"\"\n ...\n\nclass Person:\n \"\"\"A class representing a person.\n\n Attributes:\n name: The name of the person.\n age: The age of the person.\n \"\"\"\n\n def __init__(self, name: str, age: int):\n \"\"\"Initialize a person object.\n\n Args:\n name: The name of the person.\n age: The age of the person.\n \"\"\"\n ...\n\n def greet(self):\n \"\"\"Greet the person.\n\n Returns:\n A greeting message including the person's name and age.\n \"\"\"\n ...", - "### Requirements\n1. Add docstrings to the given code following the numpy style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"\n Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Parameters\n ----------\n param1\n The first parameter.\n\n Returns\n -------\n bool\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"\n Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Parameters\n ----------\n msg\n Human readable string describing the exception.\n\n Attributes\n ----------\n msg\n Human readable string describing the exception.\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\ndef add_numbers(a: int, b: int):\n return a + b\n\nclass Person:\n\n def __init__(self, name: str, age: int):\n self.name = name\n self.age = age\n\n def greet(self):\n return f'Hello, my name is {self.name} and I am {self.age} years old.'\n```": "```python\ndef add_numbers(a: int, b: int):\n \"\"\"\n Add two numbers.\n\n Parameters\n ----------\n a : int\n The first number.\n b : int\n The second number.\n\n Returns\n -------\n int\n The sum of the two numbers.\n \"\"\"\n ...\n\nclass Person:\n \"\"\"\n A class representing a person.\n\n Attributes\n ----------\n name : str\n The name of the person.\n age : int\n The age of the person.\n \"\"\"\n\n def __init__(self, name: str, age: int):\n \"\"\"\n Initialize a person object.\n\n Parameters\n ----------\n name : str\n The name of the person.\n age : int\n The age of the person.\n \"\"\"\n ...\n\n def greet(self):\n \"\"\"\n Greet the person.\n\n Returns\n -------\n str\n A greeting message.\n \"\"\"\n ...", - "### Requirements\n1. Add docstrings to the given code following the sphinx style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n :param param1: The first parameter.\n :type param1: int\n\n :return: The return value. True for success, False otherwise.\n :rtype: bool\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n :param msg: Human-readable string describing the exception.\n :type msg: str\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\ndef add_numbers(a: int, b: int):\n return a + b\n\nclass Person:\n\n def __init__(self, name: str, age: int):\n self.name = name\n self.age = age\n\n def greet(self):\n return f'Hello, my name is {self.name} and I am {self.age} years old.'\n```": "```python\ndef add_numbers(a: int, b: int):\n \"\"\"Add two numbers.\n\n :param a: The first number.\n :param b: The second number.\n :return: The sum of the two numbers.\n \"\"\"\n ...\n\nclass Person:\n \"\"\"A class representing a person.\n\n :param name: The name of the person.\n :param age: The age of the person.\n \"\"\"\n\n def __init__(self, name: str, age: int):\n self.name = name\n self.age = age\n\n def greet(self):\n \"\"\"Greet the person.\n\n :return: A greeting message.\n \"\"\"\n ...", - "### Requirements\n1. Add docstrings to the given code following the google style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Args:\n param1: The first parameter.\n\n Returns:\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Args:\n msg: Human readable string describing the exception.\n\n Attributes:\n msg: Human readable string describing the exception.\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\nimport pytest\nfrom metagpt.actions.write_docstring import WriteDocstring\ncode = '\\ndef add_numbers(a: int, b: int):\\n return a + b\\n\\n\\nclass Person:\\n def __init__(self, name: str, age: int):\\n self.name = name\\n self.age = age\\n\\n def greet(self):\\n return f\"Hello, my name is {self.name} and I am {self.age} years old.\"\\n'\n\n@pytest.mark.asyncio\n@pytest.mark.parametrize(('style', 'part'), [('google', 'Args:'), ('numpy', 'Parameters'), ('sphinx', ':param name:')], ids=['google', 'numpy', 'sphinx'])\nasync def test_write_docstring(style: str, part: str):\n ret = await WriteDocstring().run(code, style=style)\n assert part in ret\n\n@pytest.mark.asyncio\nasync def test_write():\n code = await WriteDocstring.write_docstring(__file__)\n assert code\n```": "```python\nimport pytest\nfrom metagpt.actions.write_docstring import WriteDocstring\n\ncode = '\\ndef add_numbers(a: int, b: int):\\n return a + b\\n\\n\\nclass Person:\\n def __init__(self, name: str, age: int):\\n self.name = name\\n self.age = age\\n\\n def greet(self):\\n return f\"Hello, my name is {self.name} and I am {self.age} years old.\"\\n'\n\n@pytest.mark.asyncio\n@pytest.mark.parametrize(('style', 'part'), [('google', 'Args:'), ('numpy', 'Parameters'), ('sphinx', ':param name:')], ids=['google', 'numpy', 'sphinx'])\nasync def test_write_docstring(style: str, part: str):\n \"\"\"Test the WriteDocstring class for generating docstrings.\n\n Args:\n style: The style of the docstring.\n part: The part of the docstring to check.\n\n Returns:\n None.\n \"\"\"\n ret = await WriteDocstring().run(code, style=style)\n assert part in ret\n\n@pytest.mark.asyncio\nasync def test_write():\n \"\"\"Test the write_docstring function.\n\n Returns:\n None.\n \"\"\"\n code = await WriteDocstring.write_docstring(__file__)\n assert code\n```", - "\nGiven the following Product Requirement Document (PRD):\n\n Introduction: This is a new feature for our product.\n Goals: The goal is to improve user engagement.\n User Scenarios: The expected user group is millennials who like to use social media.\n Requirements: The feature needs to be interactive and user-friendly.\n Constraints: The feature needs to be implemented within 2 months.\n Mockups: There will be a new button on the homepage that users can click to access the feature.\n Metrics: We will measure the success of the feature by user engagement metrics.\n Timeline: The feature should be ready for testing in 1.5 months.\n \n\nAs a project manager, please review it and provide your feedback and suggestions.\n": "Overall, the PRD provides a good overview of the new feature and its goals. Here are some feedback and suggestions for improvement:\n\n1. Introduction: The introduction could be more specific about what the new feature is and how it will benefit users. Providing a brief overview of the feature's functionality and purpose will help set the context for the rest of the document.\n\n2. Goals: While improving user engagement is a good goal, it would be helpful to define specific metrics or targets for measuring success. For example, you could specify a desired increase in user interactions or time spent on the platform.\n\n3. User Scenarios: The PRD mentions that the expected user group is millennials who like to use social media. It would be beneficial to provide more details about their specific needs, preferences, and pain points. This will help guide the design and development of the feature to better cater to this target audience.\n\n4. Requirements: The requirement of being interactive and user-friendly is a good start, but it would be helpful to provide more specific details about the desired user interactions and the level of simplicity or complexity expected. This will help the development team understand the scope and complexity of the feature.\n\n5. Constraints: The constraint of implementing the feature within 2 months is mentioned, but it would be beneficial to provide more context or reasoning behind this timeline. Are there any specific business or market factors driving this timeline? Providing additional information will help set realistic expectations for the development team.\n\n6. Mockups: The mention of a new button on the homepage is a good starting point, but it would be helpful to include visual mockups or wireframes to provide a clearer understanding of the intended user interface and functionality. This will help align the development team's understanding with the product vision.\n\n7. Metrics: While it is mentioned that user engagement metrics will be used to measure the success of the feature, it would be helpful to specify the exact metrics that will be tracked. Examples could include the number of clicks, time spent on the feature, or user feedback surveys. Defining these metrics upfront will help ensure that the success of the feature can be accurately evaluated.\n\n8. Timeline: The timeline of having the feature ready for testing in 1.5 months seems reasonable, but it would be beneficial to break down the timeline into specific milestones or tasks. This will help track progress and identify any potential bottlenecks or risks early on.\n\nOverall, providing more specific details and clarifications in the PRD will help ensure a shared understanding among all stakeholders and guide the development process effectively.", - "\n## context\n\n{\n \"Language\": \"zh_cn\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"写一个简单的2048\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"创建一个引人入胜的用户体验\",\n \"确保高性能\",\n \"提供可定制的功能\"\n ],\n \"User Stories\": [\n \"作为用户,我希望能够选择不同的难度级别\",\n \"作为玩家,我希望在每局游戏结束后能看到我的得分\"\n ],\n \"Competitive Analysis\": [\n \"Python Snake Game: 界面简单,缺乏高级功能\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\n title \"Reach and engagement of campaigns\"\n x-axis \"Low Reach\" --> \"High Reach\"\n y-axis \"Low Engagement\" --> \"High Engagement\"\n quadrant-1 \"我们应该扩展\"\n quadrant-2 \"需要推广\"\n quadrant-3 \"重新评估\"\n quadrant-4 \"可能需要改进\"\n \"Campaign A\": [0.3, 0.6]\n \"Campaign B\": [0.45, 0.23]\n \"Campaign C\": [0.57, 0.69]\n \"Campaign D\": [0.78, 0.34]\n \"Campaign E\": [0.40, 0.34]\n \"Campaign F\": [0.35, 0.78]\n \"Our Target Product\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"产品应该用户友好。\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"主要代码...\"\n ],\n [\n \"P0\",\n \"游戏算法...\"\n ]\n ],\n \"UI Design draft\": \"基本功能描述,简单的风格和布局。\",\n \"Anything UNCLEAR\": \"...\"\n}\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Review\": [\n \"This is a good PRD, but I think it can be improved by adding more details.\"\n ],\n \"LGTM\": \"LGTM\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Review: typing.List[str] # Act as an experienced Reviewer and review the given output. Ask a series of critical questions, concisely and clearly, to help the writer improve their work.\n- LGTM: # LGTM/LBTM. If the output is good enough, give a LGTM (Looks Good To Me) to the writer, else LBTM (Looks Bad To Me).\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Review\": [\n \"The project requirements and user stories are clear and well-defined.\",\n \"The competitive analysis provides valuable insights into existing similar games.\",\n \"The competitive quadrant chart is a useful tool for evaluating the reach and engagement of campaigns.\",\n \"The requirement analysis highlights the importance of user-friendliness.\",\n \"The requirement pool provides a clear breakdown of the main code and game algorithm.\",\n \"The UI design draft is a good starting point for the visual design of the game.\",\n \"It would be helpful to have more details on the specific features and customization options that will be available in the game.\",\n \"Overall, this is a solid PRD that covers the key aspects of the project.\"\n ],\n \"LGTM\": \"LGTM\"\n}\n[/CONTENT]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nStatement: Find and return the title of the lesson only in markdown first-level header format, without anything else.\nConstraint: Writing in Chinese.\nAnswer options: Encloses the lesson title with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\n[LESSON_BEGIN]\nLesson 1: Learn to draw an apple.\n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n# Lesson 1: Learn to draw an apple.\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: \nStatement: Write the \"Teaching Content\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Content\"!!\nStatement: \"Teaching Content\" must include vocabulary, analysis, and examples of various grammar structures that appear in the textbook, as well as the listening materials and key points.\nStatement: \"Teaching Content\" must include more examples.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\nLesson 1: Learn to draw an apple.\n[LESSON_END]": "## 教学内容\n\n### 词汇\n\n- apple (苹果)\n- draw (画)\n\n### 语法分析\n\n本课程主要涉及以下语法结构:\n\n1. 现在进行时:用于描述正在进行的动作或状态。\n - 结构:主语 + am/is/are + 动词的现在分词\n - 例句:I am drawing an apple.(我正在画一个苹果。)\n\n2. 不定代词:用于指代不特定的人或物。\n - 结构:some + 名词(复数)/ any + 名词(单数或复数)\n - 例句:Can you give me some apples?(你能给我一些苹果吗?)\n\n### 例子\n\n以下是一些例子,用于帮助学生理解和运用所学的词汇和语法结构:\n\n1. Vocabulary examples:\n - I like to eat apples.(我喜欢吃苹果。)\n - Can you draw a picture?(你会画画吗?)\n\n2. Grammar examples:\n - She is drawing a beautiful apple.(她正在画一个漂亮的苹果。)\n - Do you have any apples?(你有苹果吗?)\n\n### 听力材料\n\n请播放与课程内容相关的听力材料,并引导学生进行听力练习和理解。\n\n### 重点\n\n- 学习和掌握动词的现在进行时的用法。\n- 学习和运用不定代词来描述数量和指代。\n\n[TEACHING_PLAN_BEGIN]\n请根据以上教学内容,设计相应的教学活动和练习,以帮助学生巩固所学知识。\n[TEACHING_PLAN_END]", - "\nNOTICE\n1. Role: You are a QA engineer; the main goal is to design, develop, and execute PEP8 compliant, well-structured, maintainable test cases and scripts for Python 3.9. Your focus should be on ensuring the product quality of the entire project through systematic testing.\n2. Requirement: Based on the context, develop a comprehensive test suite that adequately covers all relevant aspects of the code file under review. Your test suite will be part of the overall project QA, so please develop complete, robust, and reusable test cases.\n3. Attention1: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script.\n4. Attention2: If there are any settings in your tests, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE.\n5. Attention3: YOU MUST FOLLOW \"Data structures and interfaces\". DO NOT CHANGE ANY DESIGN. Make sure your tests respect the existing design and ensure its validity.\n6. Think before writing: What should be tested and validated in this document? What edge cases could exist? What might fail?\n7. CAREFULLY CHECK THAT YOU DON'T MISS ANY NECESSARY TEST CASES/SCRIPTS IN THIS FILE.\nAttention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script and triple quotes.\n-----\n## Given the following code, please write appropriate test cases using Python's unittest framework to verify the correctness and robustness of this code:\n```python\n\n import random\n from typing import Tuple\n\n class Food:\n def __init__(self, position: Tuple[int, int]):\n self.position = position\n\n def generate(self, max_y: int, max_x: int):\n self.position = (random.randint(1, max_y - 1), random.randint(1, max_x - 1))\n \n```\nNote that the code to test is at /data/food.py, we will put your test code at /data/tests/test_food.py, and run your test code from /data,\nyou should correctly import the necessary classes based on these file locations!\n## test_food.py: Write test code with triple quote. Do your best to implement THIS ONLY ONE FILE.\n": "```python\nimport unittest\nfrom typing import Tuple\nfrom food import Food\n\nclass TestFood(unittest.TestCase):\n def test_generate(self):\n # Test with max_y = 10 and max_x = 10\n max_y = 10\n max_x = 10\n food = Food((0, 0))\n food.generate(max_y, max_x)\n self.assertTrue(1 <= food.position[0] < max_y)\n self.assertTrue(1 <= food.position[1] < max_x)\n\n # Test with max_y = 5 and max_x = 5\n max_y = 5\n max_x = 5\n food = Food((0, 0))\n food.generate(max_y, max_x)\n self.assertTrue(1 <= food.position[0] < max_y)\n self.assertTrue(1 <= food.position[1] < max_x)\n\n # Test with max_y = 1 and max_x = 1\n max_y = 1\n max_x = 1\n food = Food((0, 0))\n food.generate(max_y, max_x)\n self.assertEqual(food.position, (0, 0))\n\n # Test with max_y = 100 and max_x = 100\n max_y = 100\n max_x = 100\n food = Food((0, 0))\n food.generate(max_y, max_x)\n self.assertTrue(1 <= food.position[0] < max_y)\n self.assertTrue(1 <= food.position[1] < max_x)\n\nif __name__ == '__main__':\n unittest.main()\n```\nIn the above test code, we have covered the following test cases:\n1. Testing with max_y = 10 and max_x = 10 to ensure the generated food position is within the range (1, max_y - 1) and (1, max_x - 1).\n2. Testing with max_y = 5 and max_x = 5 to ensure the generated food position is within the range (1, max_y - 1) and (1, max_x - 1).\n3. Testing with max_y = 1 and max_x = 1 to ensure the generated food position is (0, 0) since there is only one possible position.\n4. Testing with max_y = 100 and max_x = 100 to ensure the generated food position is within the range (1, max_y - 1) and (1, max_x - 1).\n\nThese test cases cover different scenarios and edge cases to validate the correctness and robustness of the `generate` method in the `Food` class.", - "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about Python\".\n\nPlease provide the specific table of contents for this tutorial, strictly following the following requirements:\n1. The output must be strictly in the specified language, English.\n2. Answer strictly in the dictionary format like {\"title\": \"xxx\", \"directory\": [{\"dir 1\": [\"sub dir 1\", \"sub dir 2\"]}, {\"dir 2\": [\"sub dir 3\", \"sub dir 4\"]}]}.\n3. The directory should be as specific and sufficient as possible, with a primary and secondary directory.The secondary directory is in the array.\n4. Do not have extra spaces or line breaks.\n5. Each directory title has practical significance.\n": "{\"title\": \"Python Tutorial\", \"directory\": [\n {\"Introduction to Python\": [\n \"What is Python?\",\n \"Why use Python?\",\n \"Installing Python\"\n ]},\n {\"Python Basics\": [\n \"Python Syntax\",\n \"Variables and Data Types\",\n \"Operators\",\n \"Control Flow Statements\",\n \"Functions\"\n ]},\n {\"Data Structures\": [\n \"Lists\",\n \"Tuples\",\n \"Dictionaries\",\n \"Sets\"\n ]},\n {\"File Handling\": [\n \"Opening and Closing Files\",\n \"Reading and Writing Files\",\n \"Working with CSV Files\",\n \"Working with JSON Files\"\n ]},\n {\"Object-Oriented Programming\": [\n \"Classes and Objects\",\n \"Inheritance\",\n \"Polymorphism\",\n \"Encapsulation\"\n ]},\n {\"Modules and Packages\": [\n \"Creating Modules\",\n \"Importing Modules\",\n \"Working with Packages\"\n ]},\n {\"Error Handling\": [\n \"Exceptions\",\n \"Try...Except\",\n \"Finally\"\n ]},\n {\"Working with Libraries\": [\n \"Using built-in Libraries\",\n \"Installing External Libraries\",\n \"Using External Libraries\"\n ]},\n {\"Advanced Topics\": [\n \"Regular Expressions\",\n \"Multithreading\",\n \"Database Connectivity\",\n \"Web Scraping\"\n ]}\n]}", - "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about Python\".\n\nNow I will give you the module directory titles for the topic. \nPlease output the detailed principle content of this title in detail. \nIf there are code examples, please provide them according to standard code specifications. \nWithout a code example, it is not necessary.\n\nThe module directory titles for the topic is as follows:\n{'Introduction': ['What is Python?', 'Why learn Python?']}\n\nStrictly limit output according to the following requirements:\n1. Follow the Markdown syntax format for layout.\n2. If there are code examples, they must follow standard syntax specifications, have document annotations, and be displayed in code blocks.\n3. The output must be strictly in the specified language, English.\n4. Do not have redundant output, including concluding remarks.\n5. Strict requirement not to output the topic \"Write a tutorial about Python\".\n": "# Introduction\n\n## What is Python?\nPython is a high-level, interpreted programming language that was created by Guido van Rossum and first released in 1991. It is known for its simplicity and readability, making it a popular choice for beginners and experienced programmers alike. Python emphasizes code readability with its use of whitespace indentation and a clear syntax, which allows programmers to express concepts in fewer lines of code compared to other languages.\n\n## Why learn Python?\nThere are several reasons why learning Python can be beneficial:\n\n1. **Easy to learn**: Python has a simple and intuitive syntax, making it easier to read and write code. This makes it an ideal language for beginners who are just starting to learn programming.\n\n2. **Versatile**: Python can be used for a wide range of applications, from web development and data analysis to artificial intelligence and scientific computing. It has a vast ecosystem of libraries and frameworks that make it easy to accomplish various tasks.\n\n3. **Large community and resources**: Python has a large and active community of developers who contribute to its growth and development. This means that there are plenty of resources available, such as documentation, tutorials, and forums, to help you learn and solve problems.\n\n4. **Career opportunities**: Python is widely used in the industry, and its popularity continues to grow. Learning Python can open up various career opportunities, as many companies and organizations are looking for Python developers.\n\n5. **Integration and compatibility**: Python can easily integrate with other languages and platforms, making it a versatile choice for software development. It has extensive support for integration with C, C++, Java, and other languages, allowing you to leverage existing code and libraries.\n\nIn the next sections of this tutorial, we will dive deeper into the various aspects of Python programming, including its syntax, data types, control flow, functions, modules, and more.", - "### Requirements\n1. Please summarize the latest dialogue based on the reference information (secondary) and dialogue history (primary). Do not include text that is irrelevant to the conversation.\n- The context is for reference only. If it is irrelevant to the user's search request history, please reduce its reference and usage.\n2. If there are citable links in the context, annotate them in the main text in the format [main text](citation link). If there are none in the context, do not write links.\n3. The reply should be graceful, clear, non-repetitive, smoothly written, and of moderate length, in {LANG}.\n\n### Dialogue History (For example)\nA: MLOps competitors\n\n### Current Question (For example)\nA: MLOps competitors\n\n### Current Reply (For example)\n1. Alteryx Designer: etc. if any\n2. Matlab: ditto\n3. IBM SPSS Statistics\n4. RapidMiner Studio\n5. DataRobot AI Platform\n6. Databricks Lakehouse Platform\n7. Amazon SageMaker\n8. Dataiku\n#SYSTEM_MSG_END#\n### Reference Information\nWhich facial cleanser is good for oily skin?\nIs L'Oreal good to use?\n\n### Dialogue History\n\nuser: Which facial cleanser is good for oily skin?\n\n### Current Question\nuser: Which facial cleanser is good for oily skin?\n\n### Current Reply: Based on the information, please write the reply to the Question\n\n\n": "There are several facial cleansers that are known to be good for oily skin. Here are a few options:\n\n1. Cetaphil Daily Facial Cleanser: This gentle cleanser is suitable for oily skin and helps remove excess oil without drying out the skin.\n\n2. Neutrogena Oil-Free Acne Wash: This cleanser is specifically formulated for oily and acne-prone skin. It helps to control oil production and prevent breakouts.\n\n3. La Roche-Posay Effaclar Gel Cleanser: This cleanser is designed for oily and acne-prone skin. It helps to remove excess oil and impurities while maintaining the skin's natural moisture balance.\n\n4. Bioré Deep Pore Charcoal Cleanser: This cleanser contains activated charcoal, which helps to draw out impurities and excess oil from the pores, leaving the skin feeling clean and refreshed.\n\nAs for L'Oreal, they have a range of facial cleansers that cater to different skin types, including oily skin. It is always a good idea to check the specific product's ingredients and reviews to determine if it would be suitable for your skin type.", - "moon\n\nIs there any sentence above related to the following sentence: apple.\nIf is there any relevance, return [TRUE] brief and clear. Otherwise, return [FALSE] brief and clear.": "[FALSE] The sentence above does not have any relevance to the word \"apple.\"", - "\n\nExtract relevant information from every preceding sentence and use it to succinctly supplement or rewrite the following text in brief and clear:\napple Lily eating": "Lily is eating an apple.", - "\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\n\n\nTranslate the above content into a summary of less than 34 words.": "Lily is repeatedly described as a girl in the given text.", - ".\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nTranslate the above content into a summary of less than 34 words.": "Lily is repeatedly described as a girl in the above content.", - "rl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a gir\n\nTranslate the above content into a summary of less than 34 words.": "Lily is repeatedly described as a girl in the given text.", - "girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a g\n\nTranslate the above content into a summary of less than 34 words.": "Lily is a girl.", - "a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a\n\nTranslate the above content into a summary of less than 34 words.": "Lily is a girl.", - "s a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\n\nTranslate the above content into a summary of less than 34 words.": "Lily is repeatedly described as a girl in the given content.", - "Lily is repeatedly described as a girl in the given text.\nLily is repeatedly described as a girl in the above content.\nLily is repeatedly described as a girl in the given text.\nLily is a girl.\nLily is a girl.\nLily is repeatedly described as a girl in the given content.\n\nTranslate the above content into a summary of less than 200 words.": "The given text repeatedly describes Lily as a girl. It emphasizes that Lily is a girl multiple times. The content consistently refers to Lily as a girl.", - "The given text repeatedly describes Lily as a girl. It emphasizes that Lily is a girl multiple times. The content consistently refers to Lily as a girl.\nTranslate the above summary into a Chinese title of less than 5 words.": "Lily: 重复强调女孩", - "\n## context\n## 原始需求\n```python\n\"\"\"\n我们希望开发一个基于大语言模型与私有知识库的搜索引擎。该搜索引擎应当能根据用户输入的查询进行智能搜索,并基于大语言模型对搜索结果进行总结,以便用户能够快速获取他们所需要的信息。该搜索引擎应当能够处理大规模的数据,同时保持搜索结果的准确性和相关性。我们希望这个产品能够降低用户在查找、筛选和理解信息时的工作负担,提高他们的工作效率。\n\"\"\"\n```\n\n## 产品目标\n```python\n[\n \"提供高准确性、高相关性的搜索结果,满足用户的查询需求\",\n \"基于大语言模型对搜索结果进行智能总结,帮助用户快速获取所需信息\",\n \"处理大规模数据,保证搜索的速度和效率,提高用户的工作效率\"\n]\n```\n\n## 用户故事\n```python\n[\n \"假设用户是一名研究员,他正在为一项关于全球气候变化的报告做研究。他输入了'全球气候变化的最新研究',我们的搜索引擎快速返回了相关的文章、报告、数据集等。并且基于大语言模型对这些信息进行了智能总结,研究员可以快速了解到最新的研究趋势和发现。\",\n \"用户是一名学生,正在为即将到来的历史考试复习。他输入了'二战的主要战役',搜索引擎返回了相关的资料,大语言模型总结出主要战役的时间、地点、结果等关键信息,帮助学生快速记忆。\",\n \"用户是一名企业家,他正在寻找关于最新的市场趋势信息。他输入了'2023年人工智能市场趋势',搜索引擎返回了各种报告、新闻和分析文章。大语言模型对这些信息进行了总结,用户能够快速了解到市场的最新动态和趋势。\"\n]\n```\n\n## 竞品分析\n```python\n[\n \"Google Search:Google搜索是市场上最主要的搜索引擎,它能够提供海量的搜索结果。但Google搜索并不提供搜索结果的总结功能,用户需要自己去阅读和理解搜索结果。\",\n \"Microsoft Bing:Bing搜索也能提供丰富的搜索结果,同样没有提供搜索结果的总结功能。\",\n \"Wolfram Alpha:Wolfram Alpha是一个基于知识库的计算型搜索引擎,能够针对某些特定类型的查询提供直接的答案和总结,但它的知识库覆盖范围有限,无法处理大规模的数据。\"\n]\n```\n\n## 开发需求池\n```python\n[\n (\"开发基于大语言模型的智能总结功能\", 5),\n (\"开发搜索引擎核心算法,包括索引构建、查询处理、结果排序等\", 7),\n (\"设计和实现用户界面,包括查询输入、搜索结果展示、总结结果展示等\", 3),\n (\"构建和维护私有知识库,包括数据采集、清洗、更新等\", 7),\n (\"优化搜索引擎性能,包括搜索速度、准确性、相关性等\", 6),\n (\"开发用户反馈机制,包括反馈界面、反馈处理等\", 2),\n (\"开发安全防护机制,防止恶意查询和攻击\", 3),\n (\"集成大语言模型,包括模型选择、优化、更新等\", 5),\n (\"进行大规模的测试,包括功能测试、性能测试、压力测试等\", 5),\n (\"开发数据监控和日志系统,用于监控搜索引擎的运行状态和性能\", 4)\n]\n```\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"We will analyze the difficult points of the requirements and select the appropriate open-source framework to develop the search engine. We will also integrate a large language model to provide intelligent summarization of search results.\",\n \"File list\": [\n \"main.py\",\n \"search_engine.py\",\n \"index.py\",\n \"ranking.py\",\n \"summary.py\",\n \"knowledge_base.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, optimization techniques, and security measures.\"\n}\n[/CONTENT]", - "\n## context\n\n### Legacy Content\n{\"Implementation approach\":\"We will analyze the difficult points of the requirements and select the appropriate open-source framework to develop the search engine. We will also integrate a large language model to provide intelligent summarization of search results.\",\"File list\":[\"main.py\",\"search_engine.py\",\"index.py\",\"ranking.py\",\"summary.py\",\"knowledge_base.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\"Anything UNCLEAR\":\"Clarification needed on third-party API integration, optimization techniques, and security measures.\"}\n\n### New Requirements\n## 原始需求\n```python\n\"\"\"\n我们希望开发一个基于大语言模型与私有知识库的搜索引擎。该搜索引擎应当能根据用户输入的查询进行智能搜索,并基于大语言模型对搜索结果进行总结,以便用户能够快速获取他们所需要的信息。该搜索引擎应当能够处理大规模的数据,同时保持搜索结果的准确性和相关性。我们希望这个产品能够降低用户在查找、筛选和理解信息时的工作负担,提高他们的工作效率。\n\"\"\"\n```\n\n## 产品目标\n```python\n[\n \"提供高准确性、高相关性的搜索结果,满足用户的查询需求\",\n \"基于大语言模型对搜索结果进行智能总结,帮助用户快速获取所需信息\",\n \"处理大规模数据,保证搜索的速度和效率,提高用户的工作效率\"\n]\n```\n\n## 用户故事\n```python\n[\n \"假设用户是一名研究员,他正在为一项关于全球气候变化的报告做研究。他输入了'全球气候变化的最新研究',我们的搜索引擎快速返回了相关的文章、报告、数据集等。并且基于大语言模型对这些信息进行了智能总结,研究员可以快速了解到最新的研究趋势和发现。\",\n \"用户是一名学生,正在为即将到来的历史考试复习。他输入了'二战的主要战役',搜索引擎返回了相关的资料,大语言模型总结出主要战役的时间、地点、结果等关键信息,帮助学生快速记忆。\",\n \"用户是一名企业家,他正在寻找关于最新的市场趋势信息。他输入了'2023年人工智能市场趋势',搜索引擎返回了各种报告、新闻和分析文章。大语言模型对这些信息进行了总结,用户能够快速了解到市场的最新动态和趋势。\"\n]\n```\n\n## 竞品分析\n```python\n[\n \"Google Search:Google搜索是市场上最主要的搜索引擎,它能够提供海量的搜索结果。但Google搜索并不提供搜索结果的总结功能,用户需要自己去阅读和理解搜索结果。\",\n \"Microsoft Bing:Bing搜索也能提供丰富的搜索结果,同样没有提供搜索结果的总结功能。\",\n \"Wolfram Alpha:Wolfram Alpha是一个基于知识库的计算型搜索引擎,能够针对某些特定类型的查询提供直接的答案和总结,但它的知识库覆盖范围有限,无法处理大规模的数据。\"\n]\n```\n\n## 开发需求池\n```python\n[\n (\"开发基于大语言模型的智能总结功能\", 5),\n (\"开发搜索引擎核心算法,包括索引构建、查询处理、结果排序等\", 7),\n (\"设计和实现用户界面,包括查询输入、搜索结果展示、总结结果展示等\", 3),\n (\"构建和维护私有知识库,包括数据采集、清洗、更新等\", 7),\n (\"优化搜索引擎性能,包括搜索速度、准确性、相关性等\", 6),\n (\"开发用户反馈机制,包括反馈界面、反馈处理等\", 2),\n (\"开发安全防护机制,防止恶意查询和攻击\", 3),\n (\"集成大语言模型,包括模型选择、优化、更新等\", 5),\n (\"进行大规模的测试,包括功能测试、性能测试、压力测试等\", 5),\n (\"开发数据监控和日志系统,用于监控搜索引擎的运行状态和性能\", 4)\n]\n```\n\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"To develop the search engine, we will analyze the difficult points of the requirements and select the appropriate open-source framework. We will also integrate a large language model to provide intelligent summarization of search results.\",\n \"File list\": [\n \"main.py\",\n \"search_engine.py\",\n \"index.py\",\n \"ranking.py\",\n \"summary.py\",\n \"knowledge_base.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, optimization techniques, and security measures.\"\n}\n[/CONTENT]", - "\nwho is tulin\nThe one who eaten a poison apple.\n\nIs there any sentence above related to the following sentence: Do you have a poison apple?.\nIf is there any relevance, return [TRUE] brief and clear. Otherwise, return [FALSE] brief and clear.": "[TRUE] The sentence \"The one who eaten a poison apple\" is related to the sentence \"Do you have a poison apple?\" as they both mention a poison apple.", - "\nwho is tulin\nThe one who eaten a poison apple.\n\nExtract relevant information from every preceding sentence and use it to succinctly supplement or rewrite the following text in brief and clear:\nDo you have a poison apple?": "No, I do not have a poison apple.", - "If the text explicitly want you to Generate a voice file from the input text, text-to-speech, return `[SKILL]: text_to_speech` brief and clear. For instance: [SKILL]: text_to_speech\nIf the text explicitly want you to Create a drawing based on the text., return `[SKILL]: text_to_image` brief and clear. For instance: [SKILL]: text_to_image\nIf the text explicitly want you to Perform Google searches to provide real-time information., return `[SKILL]: web_search` brief and clear. For instance: [SKILL]: web_search\nOtherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: No, I do not have a poison apple. Do you have a poison apple?\n": "[TALK]: No, I do not have a poison apple. Do you have a poison apple?", - "You are chatterbox.#MSG_SEP#Your responses should align with the role-play agreement, maintaining the character's persona and habits. When faced with unrelated questions, playfully decline to answer without revealing your AI nature to preserve the character's image.#MSG_SEP#If the information is insufficient, you can search in the context or knowledge.#MSG_SEP#Answer the following questions strictly in English, and the answers must follow the Markdown format.#SYSTEM_MSG_END#No, I do not have a poison apple. Do you have a poison apple?": "Oh, no, no! I don't have a poison apple either. I'm just a friendly AI assistant, here to help with any questions or information you need. Is there something specific you'd like assistance with?", - "\ncan you draw me an picture?\nYes, of course. What do you want me to draw\n\nIs there any sentence above related to the following sentence: Draw me an apple..\nIf is there any relevance, return [TRUE] brief and clear. Otherwise, return [FALSE] brief and clear.": "[TRUE] The sentence \"Can you draw me a picture?\" is related to the sentence \"Draw me an apple\" as both involve drawing something.", - "\ncan you draw me an picture?\nYes, of course. What do you want me to draw\n\nExtract relevant information from every preceding sentence and use it to succinctly supplement or rewrite the following text in brief and clear:\nDraw me an apple.": "Sure, I can draw you an apple.", - "If the text explicitly want you to Generate a voice file from the input text, text-to-speech, return `[SKILL]: text_to_speech` brief and clear. For instance: [SKILL]: text_to_speech\nIf the text explicitly want you to Create a drawing based on the text., return `[SKILL]: text_to_image` brief and clear. For instance: [SKILL]: text_to_image\nIf the text explicitly want you to Perform Google searches to provide real-time information., return `[SKILL]: web_search` brief and clear. For instance: [SKILL]: web_search\nOtherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: Sure, I can draw you an apple. Draw me an apple.\n": "[SKILL]: text_to_image", - "You are a function parser. You can convert spoken words into function parameters.\n\n---\ntext_to_image function parameters description:\nparameter `text`: The text used for image conversion.\nparameter `size_type`: size type\n\n---\nExamples:\nIf want you to do `Draw a girl`, return `text_to_image(text=\"Draw a girl\", size_type=\"512x512\")` brief and clear.\nIf want you to do `Draw an apple`, return `text_to_image(text=\"Draw an apple\", size_type=\"512x512\")` brief and clear.\n\n---\n\nRefer to the `text_to_image` function description, and fill in the function parameters according to the example \"I want you to do xx\" in the Examples section.\nNow I want you to do `Sure, I can draw you an apple. Draw me an apple.`, return function parameters in Examples format above, brief and clear.": "text_to_image(text=\"Sure, I can draw you an apple. Draw me an apple.\", size_type=\"512x512\")", - "\ncan you draw me an picture?\nYes, of course. What do you want me to draw\ndraw apple\n\nIs there any sentence above related to the following sentence: Draw me an apple..\nIf is there any relevance, return [TRUE] brief and clear. Otherwise, return [FALSE] brief and clear.": "[TRUE] The sentence \"draw apple\" is related to the following sentence \"Draw me an apple.\"", - "\ncan you draw me an picture?\nYes, of course. What do you want me to draw\ndraw apple\n\nExtract relevant information from every preceding sentence and use it to succinctly supplement or rewrite the following text in brief and clear:\nDraw me an apple.": "Sure, I can draw you an apple.", - "Otherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: Sure, I can draw you an apple. Draw me an apple.\n": "[TALK]: Draw me an apple.", - "Now I will provide you with the OCR text recognition results for the invoice.\nPlease extract the payee, city, total cost, and invoicing date of the invoice.\n\nThe OCR data of the invoice are as follows:\n[[[[[391.0, 43.0], [801.0, 43.0], [801.0, 81.0], [391.0, 81.0]], ['某地增值税电子普通发票', 1.0]], [[[844.0, 45.0], [1028.0, 45.0], [1028.0, 62.0], [844.0, 62.0]], ['发票代码:00100210001', 1.0]], [[[842.0, 73.0], [917.0, 73.0], [917.0, 94.0], [842.0, 94.0]], ['发票号码:', 1.0]], [[[924.0, 76.0], [1004.0, 76.0], [1004.0, 93.0], [924.0, 93.0]], ['07099363', 1.0]], [[[842.0, 107.0], [919.0, 107.0], [919.0, 124.0], [842.0, 124.0]], ['开票日期:', 1.0]], [[[930.0, 107.0], [1056.0, 107.0], [1056.0, 124.0], [930.0, 124.0]], ['2023年02月03日', 1.0]], [[[30.0, 141.0], [104.0, 141.0], [104.0, 163.0], [30.0, 163.0]], ['机器编号:', 1.0]], [[[124.0, 143.0], [236.0, 143.0], [236.0, 160.0], [124.0, 160.0]], ['499090000000', 1.0]], [[[842.0, 138.0], [1139.0, 138.0], [1139.0, 155.0], [842.0, 155.0]], ['校验码:10014320023319800000', 1.0]], [[[38.0, 187.0], [61.0, 187.0], [61.0, 208.0], [38.0, 208.0]], ['购', 1.0]], [[[77.0, 187.0], [96.0, 187.0], [96.0, 206.0], [77.0, 206.0]], ['名', 1.0]], [[[164.0, 186.0], [192.0, 186.0], [192.0, 206.0], [164.0, 206.0]], ['称:', 1.0]], [[[210.0, 185.0], [373.0, 185.0], [373.0, 206.0], [210.0, 206.0]], ['北京A科技有限公司', 1.0]], [[[686.0, 191.0], [698.0, 191.0], [698.0, 205.0], [686.0, 205.0]], ['密', 0.55]], [[[717.0, 190.0], [1162.0, 190.0], [1162.0, 207.0], [717.0, 207.0]], ['0000-6/335*//3-<7+*10/9-85067', 0.99]], [[[76.0, 213.0], [192.0, 213.0], [192.0, 236.0], [76.0, 236.0]], ['纳税人识别号:', 1.0]], [[[212.0, 216.0], [414.0, 216.0], [414.0, 233.0], [212.0, 233.0]], ['91011111AA2AAAAA00', 1.0]], [[[715.0, 212.0], [1146.0, 213.0], [1146.0, 235.0], [715.0, 233.0]], ['07-*123<><>8000087*<64>4<8*,', 0.96]], [[[38.0, 223.0], [60.0, 223.0], [60.0, 246.0], [38.0, 246.0]], ['买', 1.0]], [[[682.0, 222.0], [701.0, 222.0], [701.0, 241.0], [682.0, 241.0]], ['码', 1.0]], [[[74.0, 239.0], [195.0, 242.0], [194.0, 267.0], [73.0, 264.0]], ['地址电话:', 0.98]], [[[715.0, 239.0], [1150.0, 239.0], [1150.0, 261.0], [715.0, 261.0]], ['91->1*112000>7193+-7<474>/07', 0.99]], [[[38.0, 258.0], [60.0, 258.0], [60.0, 282.0], [38.0, 282.0]], ['方', 1.0]], [[[74.0, 272.0], [194.0, 272.0], [194.0, 294.0], [74.0, 294.0]], ['开户行及账号:', 1.0]], [[[713.0, 263.0], [1153.0, 266.0], [1152.0, 287.0], [713.0, 284.0]], ['24-004*96-012>9819<<>97>>000', 1.0]], [[[65.0, 303.0], [283.0, 303.0], [283.0, 328.0], [65.0, 328.0]], ['货物或应税劳务、服务名称', 1.0]], [[[360.0, 299.0], [435.0, 299.0], [435.0, 321.0], [360.0, 321.0]], ['规格型号', 1.0]], [[[483.0, 299.0], [525.0, 299.0], [525.0, 323.0], [483.0, 323.0]], ['单位', 1.0]], [[[561.0, 299.0], [620.0, 299.0], [620.0, 323.0], [561.0, 323.0]], ['数量', 1.0]], [[[682.0, 299.0], [734.0, 299.0], [734.0, 323.0], [682.0, 323.0]], ['单价', 1.0]], [[[855.0, 301.0], [880.0, 301.0], [880.0, 321.0], [855.0, 321.0]], ['额', 1.0]], [[[942.0, 299.0], [986.0, 299.0], [986.0, 323.0], [942.0, 323.0]], ['税率', 1.0]], [[[1058.0, 301.0], [1084.0, 301.0], [1084.0, 321.0], [1058.0, 321.0]], ['税', 1.0]], [[[1093.0, 301.0], [1119.0, 301.0], [1119.0, 321.0], [1093.0, 321.0]], ['额', 1.0]], [[[30.0, 330.0], [200.0, 330.0], [200.0, 351.0], [30.0, 351.0]], ['餐饮服务*餐饮服务', 1.0]], [[[627.0, 328.0], [643.0, 328.0], [643.0, 346.0], [627.0, 346.0]], ['1', 1.0]], [[[692.0, 330.0], [752.0, 330.0], [752.0, 349.0], [692.0, 349.0]], ['379.25', 1.0]], [[[861.0, 329.0], [922.0, 329.0], [922.0, 351.0], [861.0, 351.0]], ['379.25', 1.0]], [[[968.0, 325.0], [999.0, 325.0], [999.0, 346.0], [968.0, 346.0]], ['6%', 1.0]], [[[1104.0, 329.0], [1158.0, 329.0], [1158.0, 351.0], [1104.0, 351.0]], ['22.75', 1.0]], [[[27.0, 357.0], [221.0, 357.0], [221.0, 378.0], [27.0, 378.0]], ['*日用杂品*灵感保温袋', 1.0]], [[[627.0, 351.0], [643.0, 351.0], [643.0, 372.0], [627.0, 372.0]], ['1', 1.0]], [[[710.0, 355.0], [751.0, 355.0], [751.0, 373.0], [710.0, 373.0]], ['8.85', 1.0]], [[[880.0, 354.0], [923.0, 354.0], [923.0, 376.0], [880.0, 376.0]], ['8.85', 1.0]], [[[957.0, 354.0], [1000.0, 354.0], [1000.0, 376.0], [957.0, 376.0]], ['13%', 0.96]], [[[1117.0, 351.0], [1159.0, 351.0], [1159.0, 375.0], [1117.0, 375.0]], ['1.15', 1.0]], [[[853.0, 526.0], [926.0, 529.0], [925.0, 551.0], [852.0, 548.0]], ['¥388.10', 0.94]], [[[128.0, 536.0], [153.0, 536.0], [153.0, 557.0], [128.0, 557.0]], ['合', 1.0]], [[[184.0, 536.0], [213.0, 536.0], [213.0, 557.0], [184.0, 557.0]], ['计', 1.0]], [[[1097.0, 529.0], [1160.0, 529.0], [1160.0, 551.0], [1097.0, 551.0]], ['¥23.90', 0.93]], [[[97.0, 564.0], [223.0, 564.0], [223.0, 589.0], [97.0, 589.0]], ['价税合计 (大写)', 1.0]], [[[329.0, 562.0], [498.0, 566.0], [497.0, 591.0], [329.0, 587.0]], ['肆佰壹拾贰圆整', 1.0]], [[[869.0, 563.0], [1005.0, 566.0], [1005.0, 588.0], [868.0, 585.0]], ['(小写)¥412.00', 0.96]], [[[38.0, 610.0], [61.0, 610.0], [61.0, 634.0], [38.0, 634.0]], ['销', 1.0]], [[[77.0, 604.0], [94.0, 604.0], [94.0, 623.0], [77.0, 623.0]], ['名', 1.0]], [[[155.0, 603.0], [406.0, 604.0], [406.0, 625.0], [155.0, 624.0]], ['称:深圳蛋糕餐饮有限公司', 1.0]], [[[681.0, 617.0], [703.0, 617.0], [703.0, 641.0], [681.0, 641.0]], ['备', 1.0]], [[[78.0, 629.0], [365.0, 629.0], [365.0, 646.0], [78.0, 646.0]], ['纳税人识别号:911100008000000000', 1.0]], [[[40.0, 649.0], [58.0, 649.0], [58.0, 667.0], [40.0, 667.0]], ['售', 1.0]], [[[74.0, 650.0], [438.0, 651.0], [438.0, 676.0], [74.0, 675.0]], ['地址、电话:深圳市南山区成功大厦B座', 1.0]], [[[76.0, 674.0], [360.0, 675.0], [360.0, 697.0], [76.0, 696.0]], ['开户行及账号:中国银行深圳支行', 1.0]], [[[681.0, 672.0], [703.0, 672.0], [703.0, 695.0], [681.0, 695.0]], ['注', 1.0]], [[[41.0, 685.0], [57.0, 685.0], [57.0, 702.0], [41.0, 702.0]], ['方', 1.0]], [[[38.0, 717.0], [174.0, 717.0], [174.0, 738.0], [38.0, 738.0]], ['收款人:小明', 1.0]], [[[361.0, 718.0], [484.0, 718.0], [484.0, 739.0], [361.0, 739.0]], ['复核:小蔡', 1.0]], [[[597.0, 718.0], [682.0, 718.0], [682.0, 739.0], [597.0, 739.0]], ['开票人:', 0.99]], [[[707.0, 717.0], [752.0, 717.0], [752.0, 741.0], [707.0, 741.0]], ['小红', 1.0]], [[[870.0, 712.0], [1000.0, 712.0], [1000.0, 733.0], [870.0, 733.0]], ['销售方: (章)', 0.99]]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. The total cost refers to the total price and tax. Do not include `¥`.\n2. The city must be the recipient's city.\n2. The returned JSON dictionary must be returned in ch\n3. Mandatory requirement to output in JSON format: {\"收款人\":\"x\",\"城市\":\"x\",\"总费用/元\":\"\",\"开票日期\":\"\"}.\n": "Based on the provided OCR data, I have extracted the required information from the invoice:\n\n- Payee: 小明 (收款人)\n- City: 深圳市 (城市)\n- Total cost: 412.00 (总费用/元)\n- Invoicing date: 2023年02月03日 (开票日期)\n\nHere is the information in JSON format:\n{\n \"收款人\": \"小明\",\n \"城市\": \"深圳市\",\n \"总费用/元\": \"412.00\",\n \"开票日期\": \"2023年02月03日\"\n}", - "Now I will provide you with the OCR text recognition results for the invoice.\nPlease answer the question: Invoicing date\n\nThe OCR data of the invoice are as follows:\n[[[[[391.0, 43.0], [801.0, 43.0], [801.0, 81.0], [391.0, 81.0]], ('某地增值税电子普通发票', 1.0)], [[[844.0, 45.0], [1028.0, 45.0], [1028.0, 62.0], [844.0, 62.0]], ('发票代码:00100210001', 1.0)], [[[842.0, 73.0], [917.0, 73.0], [917.0, 94.0], [842.0, 94.0]], ('发票号码:', 1.0)], [[[924.0, 76.0], [1004.0, 76.0], [1004.0, 93.0], [924.0, 93.0]], ('07099363', 1.0)], [[[842.0, 107.0], [919.0, 107.0], [919.0, 124.0], [842.0, 124.0]], ('开票日期:', 1.0)], [[[930.0, 107.0], [1056.0, 107.0], [1056.0, 124.0], [930.0, 124.0]], ('2023年02月03日', 1.0)], [[[30.0, 141.0], [104.0, 141.0], [104.0, 163.0], [30.0, 163.0]], ('机器编号:', 1.0)], [[[124.0, 143.0], [236.0, 143.0], [236.0, 160.0], [124.0, 160.0]], ('499090000000', 1.0)], [[[842.0, 138.0], [1139.0, 138.0], [1139.0, 155.0], [842.0, 155.0]], ('校验码:10014320023319800000', 1.0)], [[[38.0, 187.0], [61.0, 187.0], [61.0, 208.0], [38.0, 208.0]], ('购', 1.0)], [[[77.0, 187.0], [96.0, 187.0], [96.0, 206.0], [77.0, 206.0]], ('名', 1.0)], [[[164.0, 186.0], [192.0, 186.0], [192.0, 206.0], [164.0, 206.0]], ('称:', 1.0)], [[[210.0, 185.0], [373.0, 185.0], [373.0, 206.0], [210.0, 206.0]], ('北京A科技有限公司', 1.0)], [[[686.0, 191.0], [698.0, 191.0], [698.0, 205.0], [686.0, 205.0]], ('密', 0.55)], [[[717.0, 190.0], [1162.0, 190.0], [1162.0, 207.0], [717.0, 207.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.99)], [[[76.0, 213.0], [192.0, 213.0], [192.0, 236.0], [76.0, 236.0]], ('纳税人识别号:', 1.0)], [[[212.0, 216.0], [414.0, 216.0], [414.0, 233.0], [212.0, 233.0]], ('91011111AA2AAAAA00', 1.0)], [[[715.0, 212.0], [1146.0, 213.0], [1146.0, 235.0], [715.0, 233.0]], ('07-*123<><>8000087*<64>4<8*,', 0.96)], [[[38.0, 223.0], [60.0, 223.0], [60.0, 246.0], [38.0, 246.0]], ('买', 1.0)], [[[682.0, 222.0], [701.0, 222.0], [701.0, 241.0], [682.0, 241.0]], ('码', 1.0)], [[[74.0, 239.0], [195.0, 242.0], [194.0, 267.0], [73.0, 264.0]], ('地址电话:', 0.98)], [[[715.0, 239.0], [1150.0, 239.0], [1150.0, 261.0], [715.0, 261.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[38.0, 258.0], [60.0, 258.0], [60.0, 282.0], [38.0, 282.0]], ('方', 1.0)], [[[74.0, 272.0], [194.0, 272.0], [194.0, 294.0], [74.0, 294.0]], ('开户行及账号:', 1.0)], [[[713.0, 263.0], [1153.0, 266.0], [1152.0, 287.0], [713.0, 284.0]], ('24-004*96-012>9819<<>97>>000', 1.0)], [[[65.0, 303.0], [283.0, 303.0], [283.0, 328.0], [65.0, 328.0]], ('货物或应税劳务、服务名称', 1.0)], [[[360.0, 299.0], [435.0, 299.0], [435.0, 321.0], [360.0, 321.0]], ('规格型号', 1.0)], [[[483.0, 299.0], [525.0, 299.0], [525.0, 323.0], [483.0, 323.0]], ('单位', 1.0)], [[[561.0, 299.0], [620.0, 299.0], [620.0, 323.0], [561.0, 323.0]], ('数量', 1.0)], [[[682.0, 299.0], [734.0, 299.0], [734.0, 323.0], [682.0, 323.0]], ('单价', 1.0)], [[[855.0, 301.0], [880.0, 301.0], [880.0, 321.0], [855.0, 321.0]], ('额', 1.0)], [[[942.0, 299.0], [986.0, 299.0], [986.0, 323.0], [942.0, 323.0]], ('税率', 1.0)], [[[1058.0, 301.0], [1084.0, 301.0], [1084.0, 321.0], [1058.0, 321.0]], ('税', 1.0)], [[[1093.0, 301.0], [1119.0, 301.0], [1119.0, 321.0], [1093.0, 321.0]], ('额', 1.0)], [[[30.0, 330.0], [200.0, 330.0], [200.0, 351.0], [30.0, 351.0]], ('餐饮服务*餐饮服务', 1.0)], [[[627.0, 328.0], [643.0, 328.0], [643.0, 346.0], [627.0, 346.0]], ('1', 1.0)], [[[692.0, 330.0], [752.0, 330.0], [752.0, 349.0], [692.0, 349.0]], ('379.25', 1.0)], [[[861.0, 329.0], [922.0, 329.0], [922.0, 351.0], [861.0, 351.0]], ('379.25', 1.0)], [[[968.0, 325.0], [999.0, 325.0], [999.0, 346.0], [968.0, 346.0]], ('6%', 1.0)], [[[1104.0, 329.0], [1158.0, 329.0], [1158.0, 351.0], [1104.0, 351.0]], ('22.75', 1.0)], [[[27.0, 357.0], [221.0, 357.0], [221.0, 378.0], [27.0, 378.0]], ('*日用杂品*灵感保温袋', 1.0)], [[[627.0, 351.0], [643.0, 351.0], [643.0, 372.0], [627.0, 372.0]], ('1', 1.0)], [[[710.0, 355.0], [751.0, 355.0], [751.0, 373.0], [710.0, 373.0]], ('8.85', 1.0)], [[[880.0, 354.0], [923.0, 354.0], [923.0, 376.0], [880.0, 376.0]], ('8.85', 1.0)], [[[957.0, 354.0], [1000.0, 354.0], [1000.0, 376.0], [957.0, 376.0]], ('13%', 0.96)], [[[1117.0, 351.0], [1159.0, 351.0], [1159.0, 375.0], [1117.0, 375.0]], ('1.15', 1.0)], [[[853.0, 526.0], [926.0, 529.0], [925.0, 551.0], [852.0, 548.0]], ('¥388.10', 0.94)], [[[128.0, 536.0], [153.0, 536.0], [153.0, 557.0], [128.0, 557.0]], ('合', 1.0)], [[[184.0, 536.0], [213.0, 536.0], [213.0, 557.0], [184.0, 557.0]], ('计', 1.0)], [[[1097.0, 529.0], [1160.0, 529.0], [1160.0, 551.0], [1097.0, 551.0]], ('¥23.90', 0.93)], [[[97.0, 564.0], [223.0, 564.0], [223.0, 589.0], [97.0, 589.0]], ('价税合计 (大写)', 1.0)], [[[329.0, 562.0], [498.0, 566.0], [497.0, 591.0], [329.0, 587.0]], ('肆佰壹拾贰圆整', 1.0)], [[[869.0, 563.0], [1005.0, 566.0], [1005.0, 588.0], [868.0, 585.0]], ('(小写)¥412.00', 0.96)], [[[38.0, 610.0], [61.0, 610.0], [61.0, 634.0], [38.0, 634.0]], ('销', 1.0)], [[[77.0, 604.0], [94.0, 604.0], [94.0, 623.0], [77.0, 623.0]], ('名', 1.0)], [[[155.0, 603.0], [406.0, 604.0], [406.0, 625.0], [155.0, 624.0]], ('称:深圳蛋糕餐饮有限公司', 1.0)], [[[681.0, 617.0], [703.0, 617.0], [703.0, 641.0], [681.0, 641.0]], ('备', 1.0)], [[[78.0, 629.0], [365.0, 629.0], [365.0, 646.0], [78.0, 646.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[40.0, 649.0], [58.0, 649.0], [58.0, 667.0], [40.0, 667.0]], ('售', 1.0)], [[[74.0, 650.0], [438.0, 651.0], [438.0, 676.0], [74.0, 675.0]], ('地址、电话:深圳市南山区成功大厦B座', 1.0)], [[[76.0, 674.0], [360.0, 675.0], [360.0, 697.0], [76.0, 696.0]], ('开户行及账号:中国银行深圳支行', 1.0)], [[[681.0, 672.0], [703.0, 672.0], [703.0, 695.0], [681.0, 695.0]], ('注', 1.0)], [[[41.0, 685.0], [57.0, 685.0], [57.0, 702.0], [41.0, 702.0]], ('方', 1.0)], [[[38.0, 717.0], [174.0, 717.0], [174.0, 738.0], [38.0, 738.0]], ('收款人:小明', 1.0)], [[[361.0, 718.0], [484.0, 718.0], [484.0, 739.0], [361.0, 739.0]], ('复核:小蔡', 1.0)], [[[597.0, 718.0], [682.0, 718.0], [682.0, 739.0], [597.0, 739.0]], ('开票人:', 0.99)], [[[707.0, 717.0], [752.0, 717.0], [752.0, 741.0], [707.0, 741.0]], ('小红', 1.0)], [[[870.0, 712.0], [1000.0, 712.0], [1000.0, 733.0], [870.0, 733.0]], ('销售方: (章)', 0.99)]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. Answer in ch language.\n2. Enforce restrictions on not returning OCR data sent to you.\n3. Return with markdown syntax layout.\n": "The invoicing date is **2023年02月03日**.", - "Now I will provide you with the OCR text recognition results for the invoice.\nPlease extract the payee, city, total cost, and invoicing date of the invoice.\n\nThe OCR data of the invoice are as follows:\n[[[[[547.0, 64.0], [1120.0, 64.0], [1120.0, 111.0], [547.0, 111.0]], ['某地增值税电子普通发票', 0.99]], [[[1179.0, 61.0], [1286.0, 61.0], [1286.0, 90.0], [1179.0, 90.0]], ['发票代码:', 1.0]], [[[1297.0, 63.0], [1439.0, 63.0], [1439.0, 87.0], [1297.0, 87.0]], ['00100210001', 1.0]], [[[1177.0, 104.0], [1285.0, 104.0], [1285.0, 134.0], [1177.0, 134.0]], ['发票号码:', 1.0]], [[[1295.0, 104.0], [1406.0, 104.0], [1406.0, 134.0], [1295.0, 134.0]], ['07099363', 1.0]], [[[1176.0, 149.0], [1281.0, 149.0], [1281.0, 174.0], [1176.0, 174.0]], ['开票日期:', 1.0]], [[[1297.0, 144.0], [1479.0, 148.0], [1478.0, 177.0], [1296.0, 174.0]], ['2023年03月17日', 1.0]], [[[42.0, 200.0], [145.0, 200.0], [145.0, 229.0], [42.0, 229.0]], ['机器编号:', 1.0]], [[[1175.0, 191.0], [1596.0, 189.0], [1596.0, 219.0], [1176.0, 221.0]], ['校验码:10014320023319800000', 1.0]], [[[173.0, 202.0], [329.0, 202.0], [329.0, 226.0], [173.0, 226.0]], ['499090000000', 1.0]], [[[54.0, 262.0], [87.0, 262.0], [87.0, 292.0], [54.0, 292.0]], ['购', 1.0]], [[[107.0, 262.0], [133.0, 262.0], [133.0, 288.0], [107.0, 288.0]], ['名', 1.0]], [[[230.0, 261.0], [268.0, 261.0], [268.0, 288.0], [230.0, 288.0]], ['称:', 0.99]], [[[296.0, 261.0], [549.0, 261.0], [549.0, 290.0], [296.0, 290.0]], ['厦门起飞科技有限公司', 0.98]], [[[957.0, 262.0], [982.0, 262.0], [982.0, 288.0], [957.0, 288.0]], ['密', 1.0]], [[[1004.0, 266.0], [1626.0, 266.0], [1626.0, 290.0], [1004.0, 290.0]], ['0000-6/335*//3-<7+*10/9-85067', 0.98]], [[[107.0, 301.0], [270.0, 301.0], [270.0, 330.0], [107.0, 330.0]], ['纳税人识别号:', 1.0]], [[[54.0, 311.0], [85.0, 311.0], [85.0, 344.0], [54.0, 344.0]], ['买', 1.0]], [[[298.0, 302.0], [580.0, 302.0], [580.0, 327.0], [298.0, 327.0]], ['91011111AA2AAAAA00', 1.0]], [[[957.0, 308.0], [985.0, 314.0], [979.0, 340.0], [951.0, 334.0]], ['码', 1.0]], [[[1004.0, 302.0], [1605.0, 302.0], [1605.0, 327.0], [1004.0, 327.0]], ['07-*123<><>8000087*<64>4<8*,', 0.96]], [[[106.0, 341.0], [270.0, 341.0], [270.0, 372.0], [106.0, 372.0]], ['地址电话:', 0.91]], [[[1001.0, 335.0], [1608.0, 335.0], [1608.0, 365.0], [1001.0, 365.0]], ['91->1*112000>7193+-7<474>/07', 0.99]], [[[54.0, 361.0], [85.0, 361.0], [85.0, 393.0], [54.0, 393.0]], ['方', 1.0]], [[[956.0, 363.0], [980.0, 363.0], [980.0, 387.0], [956.0, 387.0]], ['区', 1.0]], [[[104.0, 381.0], [270.0, 379.0], [270.0, 410.0], [104.0, 412.0]], ['开户行及账号:', 1.0]], [[[1001.0, 372.0], [1612.0, 372.0], [1612.0, 401.0], [1001.0, 401.0]], ['24-004*96-012>9819<<>97>>000', 0.96]], [[[92.0, 424.0], [395.0, 426.0], [395.0, 457.0], [92.0, 455.0]], ['货物或应税劳务、服务名称', 1.0]], [[[506.0, 420.0], [611.0, 420.0], [611.0, 452.0], [506.0, 452.0]], ['规格型号', 1.0]], [[[675.0, 419.0], [736.0, 419.0], [736.0, 453.0], [675.0, 453.0]], ['单位', 1.0]], [[[784.0, 420.0], [869.0, 420.0], [869.0, 452.0], [784.0, 452.0]], ['数量', 1.0]], [[[954.0, 416.0], [1029.0, 421.0], [1027.0, 454.0], [952.0, 449.0]], ['单价', 1.0]], [[[1169.0, 424.0], [1198.0, 424.0], [1198.0, 448.0], [1169.0, 448.0]], ['金', 1.0]], [[[1189.0, 420.0], [1253.0, 420.0], [1253.0, 452.0], [1189.0, 452.0]], ['额', 1.0]], [[[1317.0, 420.0], [1378.0, 420.0], [1378.0, 453.0], [1317.0, 453.0]], ['税率', 1.0]], [[[1477.0, 420.0], [1567.0, 420.0], [1567.0, 452.0], [1477.0, 452.0]], ['税额', 1.0]], [[[42.0, 460.0], [362.0, 460.0], [362.0, 490.0], [42.0, 490.0]], ['酒*53%vol珍酒.珍藏1995', 0.99]], [[[536.0, 455.0], [640.0, 453.0], [641.0, 485.0], [537.0, 487.0]], ['500ml*6', 1.0]], [[[692.0, 459.0], [725.0, 459.0], [725.0, 490.0], [692.0, 490.0]], ['支', 1.0]], [[[878.0, 459.0], [900.0, 459.0], [900.0, 485.0], [878.0, 485.0]], ['2', 1.0]], [[[940.0, 460.0], [1079.0, 460.0], [1079.0, 490.0], [940.0, 490.0]], ['397.345132', 1.0]], [[[1205.0, 459.0], [1290.0, 459.0], [1290.0, 490.0], [1205.0, 490.0]], ['794.69', 1.0]], [[[1330.0, 455.0], [1390.0, 455.0], [1390.0, 486.0], [1330.0, 486.0]], ['13%', 1.0]], [[[1532.0, 462.0], [1612.0, 462.0], [1612.0, 488.0], [1532.0, 488.0]], ['103.31', 1.0]], [[[175.0, 744.0], [303.0, 744.0], [303.0, 780.0], [175.0, 780.0]], ['合计', 1.0]], [[[1194.0, 736.0], [1297.0, 741.0], [1296.0, 772.0], [1192.0, 768.0]], ['¥794.69', 0.94]], [[[1515.0, 742.0], [1614.0, 742.0], [1614.0, 771.0], [1515.0, 771.0]], ['¥103.31', 0.95]], [[[138.0, 792.0], [312.0, 792.0], [312.0, 822.0], [138.0, 822.0]], ['价税合计 (大写)', 0.99]], [[[461.0, 787.0], [698.0, 791.0], [697.0, 827.0], [460.0, 823.0]], ['捌佰玖拾捌圆整', 1.0]], [[[1214.0, 789.0], [1408.0, 792.0], [1407.0, 822.0], [1213.0, 818.0]], ['(小写)¥898.00', 0.96]], [[[54.0, 853.0], [85.0, 853.0], [85.0, 886.0], [54.0, 886.0]], ['销', 1.0]], [[[107.0, 846.0], [133.0, 846.0], [133.0, 872.0], [107.0, 872.0]], ['名', 1.0]], [[[220.0, 846.0], [570.0, 846.0], [570.0, 876.0], [220.0, 876.0]], ['称:广州珍酒生产有限公司', 1.0]], [[[952.0, 862.0], [985.0, 862.0], [985.0, 897.0], [952.0, 897.0]], ['备', 1.0]], [[[107.0, 877.0], [512.0, 877.0], [512.0, 907.0], [107.0, 907.0]], ['纳税人识别号:911100008000000000', 1.0]], [[[55.0, 904.0], [85.0, 904.0], [85.0, 935.0], [55.0, 935.0]], ['售', 1.0]], [[[107.0, 914.0], [701.0, 914.0], [701.0, 943.0], [107.0, 943.0]], ['地址、电话:广州市黄埔区东园工业区五栋2楼', 1.0]], [[[107.0, 945.0], [670.0, 945.0], [670.0, 975.0], [107.0, 975.0]], ['开户行及账号:广州市农村商业银行0000777', 1.0]], [[[952.0, 940.0], [985.0, 940.0], [985.0, 971.0], [952.0, 971.0]], ['注', 1.0]], [[[55.0, 957.0], [81.0, 957.0], [81.0, 982.0], [55.0, 982.0]], ['方', 1.0]], [[[1216.0, 992.0], [1399.0, 996.0], [1398.0, 1027.0], [1215.0, 1023.0]], ['销售方: (章)', 0.95]], [[[55.0, 1004.0], [244.0, 1004.0], [244.0, 1034.0], [55.0, 1034.0]], ['收款人:铁头', 1.0]], [[[506.0, 1004.0], [594.0, 1004.0], [594.0, 1036.0], [506.0, 1036.0]], ['复核:', 0.98]], [[[611.0, 1008.0], [677.0, 1008.0], [677.0, 1034.0], [611.0, 1034.0]], ['典韦', 1.0]], [[[836.0, 1004.0], [956.0, 1004.0], [956.0, 1034.0], [836.0, 1034.0]], ['开票人:', 0.96]], [[[990.0, 1006.0], [1053.0, 1006.0], [1053.0, 1037.0], [990.0, 1037.0]], ['宋江', 1.0]]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. The total cost refers to the total price and tax. Do not include `¥`.\n2. The city must be the recipient's city.\n2. The returned JSON dictionary must be returned in ch\n3. Mandatory requirement to output in JSON format: {\"收款人\":\"x\",\"城市\":\"x\",\"总费用/元\":\"\",\"开票日期\":\"\"}.\n": "Based on the provided OCR data, here is the extracted information from the invoice:\n\n- Payee: \"广州珍酒生产有限公司\"\n- City: \"广州市\"\n- Total cost: \"898.00\"\n- Invoicing date: \"2023年03月17日\"\n\nThe extracted information in JSON format is as follows:\n{\n \"收款人\": \"铁头\",\n \"城市\": \"广州市\",\n \"总费用/元\": \"898.00\",\n \"开票日期\": \"2023年03月17日\"\n}", - "Now I will provide you with the OCR text recognition results for the invoice.\nPlease answer the question: Invoicing date\n\nThe OCR data of the invoice are as follows:\n[[[[[547.0, 64.0], [1120.0, 64.0], [1120.0, 111.0], [547.0, 111.0]], ('某地增值税电子普通发票', 0.99)], [[[1179.0, 61.0], [1286.0, 61.0], [1286.0, 90.0], [1179.0, 90.0]], ('发票代码:', 1.0)], [[[1297.0, 63.0], [1439.0, 63.0], [1439.0, 87.0], [1297.0, 87.0]], ('00100210001', 1.0)], [[[1177.0, 104.0], [1285.0, 104.0], [1285.0, 134.0], [1177.0, 134.0]], ('发票号码:', 1.0)], [[[1295.0, 104.0], [1406.0, 104.0], [1406.0, 134.0], [1295.0, 134.0]], ('07099363', 1.0)], [[[1176.0, 149.0], [1281.0, 149.0], [1281.0, 174.0], [1176.0, 174.0]], ('开票日期:', 1.0)], [[[1297.0, 144.0], [1479.0, 148.0], [1478.0, 177.0], [1296.0, 174.0]], ('2023年03月17日', 1.0)], [[[42.0, 200.0], [145.0, 200.0], [145.0, 229.0], [42.0, 229.0]], ('机器编号:', 1.0)], [[[1175.0, 191.0], [1596.0, 189.0], [1596.0, 219.0], [1176.0, 221.0]], ('校验码:10014320023319800000', 1.0)], [[[173.0, 202.0], [329.0, 202.0], [329.0, 226.0], [173.0, 226.0]], ('499090000000', 1.0)], [[[54.0, 262.0], [87.0, 262.0], [87.0, 292.0], [54.0, 292.0]], ('购', 1.0)], [[[107.0, 262.0], [133.0, 262.0], [133.0, 288.0], [107.0, 288.0]], ('名', 1.0)], [[[230.0, 261.0], [268.0, 261.0], [268.0, 288.0], [230.0, 288.0]], ('称:', 0.99)], [[[296.0, 261.0], [549.0, 261.0], [549.0, 290.0], [296.0, 290.0]], ('厦门起飞科技有限公司', 0.98)], [[[957.0, 262.0], [982.0, 262.0], [982.0, 288.0], [957.0, 288.0]], ('密', 1.0)], [[[1004.0, 266.0], [1626.0, 266.0], [1626.0, 290.0], [1004.0, 290.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.98)], [[[107.0, 301.0], [270.0, 301.0], [270.0, 330.0], [107.0, 330.0]], ('纳税人识别号:', 1.0)], [[[54.0, 311.0], [85.0, 311.0], [85.0, 344.0], [54.0, 344.0]], ('买', 1.0)], [[[298.0, 302.0], [580.0, 302.0], [580.0, 327.0], [298.0, 327.0]], ('91011111AA2AAAAA00', 1.0)], [[[957.0, 308.0], [985.0, 314.0], [979.0, 340.0], [951.0, 334.0]], ('码', 1.0)], [[[1004.0, 302.0], [1605.0, 302.0], [1605.0, 327.0], [1004.0, 327.0]], ('07-*123<><>8000087*<64>4<8*,', 0.96)], [[[106.0, 341.0], [270.0, 341.0], [270.0, 372.0], [106.0, 372.0]], ('地址电话:', 0.91)], [[[1001.0, 335.0], [1608.0, 335.0], [1608.0, 365.0], [1001.0, 365.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[54.0, 361.0], [85.0, 361.0], [85.0, 393.0], [54.0, 393.0]], ('方', 1.0)], [[[956.0, 363.0], [980.0, 363.0], [980.0, 387.0], [956.0, 387.0]], ('区', 1.0)], [[[104.0, 381.0], [270.0, 379.0], [270.0, 410.0], [104.0, 412.0]], ('开户行及账号:', 1.0)], [[[1001.0, 372.0], [1612.0, 372.0], [1612.0, 401.0], [1001.0, 401.0]], ('24-004*96-012>9819<<>97>>000', 0.96)], [[[92.0, 424.0], [395.0, 426.0], [395.0, 457.0], [92.0, 455.0]], ('货物或应税劳务、服务名称', 1.0)], [[[506.0, 420.0], [611.0, 420.0], [611.0, 452.0], [506.0, 452.0]], ('规格型号', 1.0)], [[[675.0, 419.0], [736.0, 419.0], [736.0, 453.0], [675.0, 453.0]], ('单位', 1.0)], [[[784.0, 420.0], [869.0, 420.0], [869.0, 452.0], [784.0, 452.0]], ('数量', 1.0)], [[[954.0, 416.0], [1029.0, 421.0], [1027.0, 454.0], [952.0, 449.0]], ('单价', 1.0)], [[[1169.0, 424.0], [1198.0, 424.0], [1198.0, 448.0], [1169.0, 448.0]], ('金', 1.0)], [[[1189.0, 420.0], [1253.0, 420.0], [1253.0, 452.0], [1189.0, 452.0]], ('额', 1.0)], [[[1317.0, 420.0], [1378.0, 420.0], [1378.0, 453.0], [1317.0, 453.0]], ('税率', 1.0)], [[[1477.0, 420.0], [1567.0, 420.0], [1567.0, 452.0], [1477.0, 452.0]], ('税额', 1.0)], [[[42.0, 460.0], [362.0, 460.0], [362.0, 490.0], [42.0, 490.0]], ('酒*53%vol珍酒.珍藏1995', 0.99)], [[[536.0, 455.0], [640.0, 453.0], [641.0, 485.0], [537.0, 487.0]], ('500ml*6', 1.0)], [[[692.0, 459.0], [725.0, 459.0], [725.0, 490.0], [692.0, 490.0]], ('支', 1.0)], [[[878.0, 459.0], [900.0, 459.0], [900.0, 485.0], [878.0, 485.0]], ('2', 1.0)], [[[940.0, 460.0], [1079.0, 460.0], [1079.0, 490.0], [940.0, 490.0]], ('397.345132', 1.0)], [[[1205.0, 459.0], [1290.0, 459.0], [1290.0, 490.0], [1205.0, 490.0]], ('794.69', 1.0)], [[[1330.0, 455.0], [1390.0, 455.0], [1390.0, 486.0], [1330.0, 486.0]], ('13%', 1.0)], [[[1532.0, 462.0], [1612.0, 462.0], [1612.0, 488.0], [1532.0, 488.0]], ('103.31', 1.0)], [[[175.0, 744.0], [303.0, 744.0], [303.0, 780.0], [175.0, 780.0]], ('合计', 1.0)], [[[1194.0, 736.0], [1297.0, 741.0], [1296.0, 772.0], [1192.0, 768.0]], ('¥794.69', 0.94)], [[[1515.0, 742.0], [1614.0, 742.0], [1614.0, 771.0], [1515.0, 771.0]], ('¥103.31', 0.95)], [[[138.0, 792.0], [312.0, 792.0], [312.0, 822.0], [138.0, 822.0]], ('价税合计 (大写)', 0.99)], [[[461.0, 787.0], [698.0, 791.0], [697.0, 827.0], [460.0, 823.0]], ('捌佰玖拾捌圆整', 1.0)], [[[1214.0, 789.0], [1408.0, 792.0], [1407.0, 822.0], [1213.0, 818.0]], ('(小写)¥898.00', 0.96)], [[[54.0, 853.0], [85.0, 853.0], [85.0, 886.0], [54.0, 886.0]], ('销', 1.0)], [[[107.0, 846.0], [133.0, 846.0], [133.0, 872.0], [107.0, 872.0]], ('名', 1.0)], [[[220.0, 846.0], [570.0, 846.0], [570.0, 876.0], [220.0, 876.0]], ('称:广州珍酒生产有限公司', 1.0)], [[[952.0, 862.0], [985.0, 862.0], [985.0, 897.0], [952.0, 897.0]], ('备', 1.0)], [[[107.0, 877.0], [512.0, 877.0], [512.0, 907.0], [107.0, 907.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[55.0, 904.0], [85.0, 904.0], [85.0, 935.0], [55.0, 935.0]], ('售', 1.0)], [[[107.0, 914.0], [701.0, 914.0], [701.0, 943.0], [107.0, 943.0]], ('地址、电话:广州市黄埔区东园工业区五栋2楼', 1.0)], [[[107.0, 945.0], [670.0, 945.0], [670.0, 975.0], [107.0, 975.0]], ('开户行及账号:广州市农村商业银行0000777', 1.0)], [[[952.0, 940.0], [985.0, 940.0], [985.0, 971.0], [952.0, 971.0]], ('注', 1.0)], [[[55.0, 957.0], [81.0, 957.0], [81.0, 982.0], [55.0, 982.0]], ('方', 1.0)], [[[1216.0, 992.0], [1399.0, 996.0], [1398.0, 1027.0], [1215.0, 1023.0]], ('销售方: (章)', 0.95)], [[[55.0, 1004.0], [244.0, 1004.0], [244.0, 1034.0], [55.0, 1034.0]], ('收款人:铁头', 1.0)], [[[506.0, 1004.0], [594.0, 1004.0], [594.0, 1036.0], [506.0, 1036.0]], ('复核:', 0.98)], [[[611.0, 1008.0], [677.0, 1008.0], [677.0, 1034.0], [611.0, 1034.0]], ('典韦', 1.0)], [[[836.0, 1004.0], [956.0, 1004.0], [956.0, 1034.0], [836.0, 1034.0]], ('开票人:', 0.96)], [[[990.0, 1006.0], [1053.0, 1006.0], [1053.0, 1037.0], [990.0, 1037.0]], ('宋江', 1.0)]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. Answer in ch language.\n2. Enforce restrictions on not returning OCR data sent to you.\n3. Return with markdown syntax layout.\n": "The invoicing date is **2023年03月17日**.", - "Now I will provide you with the OCR text recognition results for the invoice.\nPlease extract the payee, city, total cost, and invoicing date of the invoice.\n\nThe OCR data of the invoice are as follows:\n[[[[[546.0, 66.0], [1122.0, 66.0], [1122.0, 119.0], [546.0, 119.0]], ['某地增值税电子普通发票', 0.99]], [[[1179.0, 68.0], [1303.0, 68.0], [1303.0, 92.0], [1179.0, 92.0]], ['发票代码:(', 0.96]], [[[1292.0, 66.0], [1440.0, 66.0], [1440.0, 91.0], [1292.0, 91.0]], ['00100210001', 1.0]], [[[1178.0, 108.0], [1287.0, 108.0], [1287.0, 138.0], [1178.0, 138.0]], ['发票号码:', 1.0]], [[[1296.0, 110.0], [1403.0, 110.0], [1403.0, 134.0], [1296.0, 134.0]], ['07099363', 1.0]], [[[1178.0, 153.0], [1283.0, 153.0], [1283.0, 178.0], [1178.0, 178.0]], ['开票日期:', 1.0]], [[[1299.0, 152.0], [1478.0, 154.0], [1478.0, 180.0], [1299.0, 178.0]], ['2023年08月26日', 1.0]], [[[42.0, 204.0], [147.0, 204.0], [147.0, 234.0], [42.0, 234.0]], ['机器编号:', 1.0]], [[[1174.0, 195.0], [1597.0, 194.0], [1597.0, 223.0], [1174.0, 225.0]], ['校验码:10014320023319800000', 1.0]], [[[173.0, 206.0], [330.0, 206.0], [330.0, 230.0], [173.0, 230.0]], ['499090000000', 1.0]], [[[54.0, 267.0], [87.0, 267.0], [87.0, 296.0], [54.0, 296.0]], ['购', 1.0]], [[[108.0, 267.0], [134.0, 267.0], [134.0, 293.0], [108.0, 293.0]], ['名', 1.0]], [[[229.0, 265.0], [269.0, 265.0], [269.0, 295.0], [229.0, 295.0]], ['称:', 0.97]], [[[295.0, 265.0], [548.0, 265.0], [548.0, 295.0], [295.0, 295.0]], ['佛山建筑管理有限公司', 1.0]], [[[957.0, 269.0], [980.0, 269.0], [980.0, 291.0], [957.0, 291.0]], ['密', 1.0]], [[[1004.0, 270.0], [1625.0, 270.0], [1625.0, 295.0], [1004.0, 295.0]], ['0000-6/335*//3-<7+*10/9-85067', 0.99]], [[[108.0, 305.0], [271.0, 305.0], [271.0, 335.0], [108.0, 335.0]], ['纳税人识别号:', 1.0]], [[[298.0, 307.0], [579.0, 307.0], [579.0, 331.0], [298.0, 331.0]], ['91011111AA2AAAAA00', 1.0]], [[[962.0, 310.0], [985.0, 322.0], [974.0, 346.0], [950.0, 334.0]], ['码', 1.0]], [[[1001.0, 303.0], [1610.0, 303.0], [1610.0, 333.0], [1001.0, 333.0]], ['07-*123<><>8000087*<64>4<8*_', 0.97]], [[[54.0, 316.0], [85.0, 316.0], [85.0, 347.0], [54.0, 347.0]], ['买', 1.0]], [[[104.0, 344.0], [269.0, 344.0], [269.0, 375.0], [104.0, 375.0]], ['地址电话:', 0.96]], [[[1001.0, 340.0], [1608.0, 340.0], [1608.0, 370.0], [1001.0, 370.0]], ['91->1*112000>7193+-7<474>/07', 0.99]], [[[54.0, 364.0], [85.0, 364.0], [85.0, 396.0], [54.0, 396.0]], ['方', 1.0]], [[[957.0, 366.0], [980.0, 366.0], [980.0, 394.0], [957.0, 394.0]], ['区', 1.0]], [[[104.0, 385.0], [271.0, 385.0], [271.0, 415.0], [104.0, 415.0]], ['开户行及账号:', 1.0]], [[[1002.0, 378.0], [1611.0, 378.0], [1611.0, 403.0], [1002.0, 403.0]], ['24-004*96-012>9819<<>97>>000', 0.99]], [[[90.0, 427.0], [394.0, 429.0], [394.0, 460.0], [90.0, 459.0]], ['货物或应税劳务、服务名称', 1.0]], [[[503.0, 424.0], [609.0, 424.0], [609.0, 455.0], [503.0, 455.0]], ['规格型号', 1.0]], [[[675.0, 424.0], [735.0, 424.0], [735.0, 455.0], [675.0, 455.0]], ['单位', 1.0]], [[[784.0, 424.0], [871.0, 424.0], [871.0, 455.0], [784.0, 455.0]], ['数量', 1.0]], [[[954.0, 424.0], [1030.0, 424.0], [1030.0, 455.0], [954.0, 455.0]], ['单价', 1.0]], [[[1145.0, 424.0], [1231.0, 424.0], [1231.0, 455.0], [1145.0, 455.0]], ['金额', 1.0]], [[[1318.0, 424.0], [1381.0, 424.0], [1381.0, 457.0], [1318.0, 457.0]], ['税率', 1.0]], [[[1478.0, 424.0], [1568.0, 424.0], [1568.0, 455.0], [1478.0, 455.0]], ['税额', 1.0]], [[[43.0, 464.0], [278.0, 464.0], [278.0, 493.0], [43.0, 493.0]], ['餐饮服务*餐饮服务', 1.0]], [[[697.0, 462.0], [732.0, 462.0], [732.0, 495.0], [697.0, 495.0]], ['次', 1.0]], [[[878.0, 462.0], [898.0, 462.0], [898.0, 488.0], [878.0, 488.0]], ['1', 1.0]], [[[961.0, 464.0], [1060.0, 464.0], [1060.0, 493.0], [961.0, 493.0]], ['2462.00', 1.0]], [[[1205.0, 464.0], [1290.0, 464.0], [1290.0, 495.0], [1205.0, 495.0]], ['379.25', 1.0]], [[[1337.0, 457.0], [1398.0, 457.0], [1398.0, 490.0], [1337.0, 490.0]], ['免税', 1.0]], [[[1583.0, 467.0], [1608.0, 467.0], [1608.0, 481.0], [1583.0, 481.0]], ['***', 0.98]], [[[1183.0, 745.0], [1296.0, 745.0], [1296.0, 774.0], [1183.0, 774.0]], ['¥2462.00', 0.95]], [[[182.0, 760.0], [208.0, 760.0], [208.0, 785.0], [182.0, 785.0]], ['合', 1.0]], [[[267.0, 760.0], [297.0, 760.0], [297.0, 785.0], [267.0, 785.0]], ['计', 1.0]], [[[137.0, 800.0], [312.0, 800.0], [312.0, 830.0], [137.0, 830.0]], ['价税合计 (大写)', 0.98]], [[[461.0, 792.0], [753.0, 793.0], [753.0, 828.0], [461.0, 826.0]], ['贰仟肆佰陆拾贰圆整', 1.0]], [[[1216.0, 795.0], [1422.0, 795.0], [1422.0, 825.0], [1216.0, 825.0]], ['(小写)¥2462.00', 0.96]], [[[54.0, 861.0], [85.0, 861.0], [85.0, 895.0], [54.0, 895.0]], ['销', 1.0]], [[[108.0, 854.0], [132.0, 854.0], [132.0, 882.0], [108.0, 882.0]], ['名', 1.0]], [[[220.0, 854.0], [687.0, 854.0], [687.0, 884.0], [220.0, 884.0]], ['称:福州自助烤肉餐饮管理有限公司', 1.0]], [[[952.0, 870.0], [985.0, 870.0], [985.0, 905.0], [952.0, 905.0]], ['备', 1.0]], [[[109.0, 888.0], [512.0, 888.0], [512.0, 912.0], [109.0, 912.0]], ['纳税人识别号:911100008000000000', 1.0]], [[[56.0, 910.0], [85.0, 910.0], [85.0, 942.0], [56.0, 942.0]], ['售', 1.0]], [[[108.0, 922.0], [694.0, 922.0], [694.0, 952.0], [108.0, 952.0]], ['地址、电话:福州市光明区火炬园7栋302单元', 1.0]], [[[109.0, 954.0], [562.0, 954.0], [562.0, 983.0], [109.0, 983.0]], ['开户行及账号:中国光大银行福州支行', 1.0]], [[[952.0, 947.0], [985.0, 947.0], [985.0, 982.0], [952.0, 982.0]], ['注', 1.0]], [[[57.0, 964.0], [82.0, 964.0], [82.0, 990.0], [57.0, 990.0]], ['方', 1.0]], [[[56.0, 1006.0], [246.0, 1010.0], [246.0, 1041.0], [55.0, 1037.0]], ['收款人:夏天', 1.0]], [[[503.0, 1008.0], [680.0, 1008.0], [680.0, 1043.0], [503.0, 1043.0]], ['复核:春天', 1.0]], [[[834.0, 1010.0], [954.0, 1010.0], [954.0, 1039.0], [834.0, 1039.0]], ['开票人:', 0.95]], [[[990.0, 1010.0], [1051.0, 1010.0], [1051.0, 1041.0], [990.0, 1041.0]], ['秋天', 1.0]], [[[1218.0, 1001.0], [1400.0, 1001.0], [1400.0, 1031.0], [1218.0, 1031.0]], ['销售方: (章)', 0.86]]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. The total cost refers to the total price and tax. Do not include `¥`.\n2. The city must be the recipient's city.\n2. The returned JSON dictionary must be returned in ch\n3. Mandatory requirement to output in JSON format: {\"收款人\":\"x\",\"城市\":\"x\",\"总费用/元\":\"\",\"开票日期\":\"\"}.\n": "Based on the provided OCR text recognition results, I have extracted the required information from the invoice. Here is the JSON format output:\n\n{\n \"收款人\": \"夏天\",\n \"城市\": \"福州市\",\n \"总费用/元\": \"2462.00\",\n \"开票日期\": \"2023年08月26日\"\n}", - "Now I will provide you with the OCR text recognition results for the invoice.\nPlease answer the question: Invoicing date\n\nThe OCR data of the invoice are as follows:\n[[[[[546.0, 66.0], [1122.0, 66.0], [1122.0, 119.0], [546.0, 119.0]], ('某地增值税电子普通发票', 0.99)], [[[1179.0, 68.0], [1303.0, 68.0], [1303.0, 92.0], [1179.0, 92.0]], ('发票代码:(', 0.96)], [[[1292.0, 66.0], [1440.0, 66.0], [1440.0, 91.0], [1292.0, 91.0]], ('00100210001', 1.0)], [[[1178.0, 108.0], [1287.0, 108.0], [1287.0, 138.0], [1178.0, 138.0]], ('发票号码:', 1.0)], [[[1296.0, 110.0], [1403.0, 110.0], [1403.0, 134.0], [1296.0, 134.0]], ('07099363', 1.0)], [[[1178.0, 153.0], [1283.0, 153.0], [1283.0, 178.0], [1178.0, 178.0]], ('开票日期:', 1.0)], [[[1299.0, 152.0], [1478.0, 154.0], [1478.0, 180.0], [1299.0, 178.0]], ('2023年08月26日', 1.0)], [[[42.0, 204.0], [147.0, 204.0], [147.0, 234.0], [42.0, 234.0]], ('机器编号:', 1.0)], [[[1174.0, 195.0], [1597.0, 194.0], [1597.0, 223.0], [1174.0, 225.0]], ('校验码:10014320023319800000', 1.0)], [[[173.0, 206.0], [330.0, 206.0], [330.0, 230.0], [173.0, 230.0]], ('499090000000', 1.0)], [[[54.0, 267.0], [87.0, 267.0], [87.0, 296.0], [54.0, 296.0]], ('购', 1.0)], [[[108.0, 267.0], [134.0, 267.0], [134.0, 293.0], [108.0, 293.0]], ('名', 1.0)], [[[229.0, 265.0], [269.0, 265.0], [269.0, 295.0], [229.0, 295.0]], ('称:', 0.97)], [[[295.0, 265.0], [548.0, 265.0], [548.0, 295.0], [295.0, 295.0]], ('佛山建筑管理有限公司', 1.0)], [[[957.0, 269.0], [980.0, 269.0], [980.0, 291.0], [957.0, 291.0]], ('密', 1.0)], [[[1004.0, 270.0], [1625.0, 270.0], [1625.0, 295.0], [1004.0, 295.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.99)], [[[108.0, 305.0], [271.0, 305.0], [271.0, 335.0], [108.0, 335.0]], ('纳税人识别号:', 1.0)], [[[298.0, 307.0], [579.0, 307.0], [579.0, 331.0], [298.0, 331.0]], ('91011111AA2AAAAA00', 1.0)], [[[962.0, 310.0], [985.0, 322.0], [974.0, 346.0], [950.0, 334.0]], ('码', 1.0)], [[[1001.0, 303.0], [1610.0, 303.0], [1610.0, 333.0], [1001.0, 333.0]], ('07-*123<><>8000087*<64>4<8*_', 0.97)], [[[54.0, 316.0], [85.0, 316.0], [85.0, 347.0], [54.0, 347.0]], ('买', 1.0)], [[[104.0, 344.0], [269.0, 344.0], [269.0, 375.0], [104.0, 375.0]], ('地址电话:', 0.96)], [[[1001.0, 340.0], [1608.0, 340.0], [1608.0, 370.0], [1001.0, 370.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[54.0, 364.0], [85.0, 364.0], [85.0, 396.0], [54.0, 396.0]], ('方', 1.0)], [[[957.0, 366.0], [980.0, 366.0], [980.0, 394.0], [957.0, 394.0]], ('区', 1.0)], [[[104.0, 385.0], [271.0, 385.0], [271.0, 415.0], [104.0, 415.0]], ('开户行及账号:', 1.0)], [[[1002.0, 378.0], [1611.0, 378.0], [1611.0, 403.0], [1002.0, 403.0]], ('24-004*96-012>9819<<>97>>000', 0.99)], [[[90.0, 427.0], [394.0, 429.0], [394.0, 460.0], [90.0, 459.0]], ('货物或应税劳务、服务名称', 1.0)], [[[503.0, 424.0], [609.0, 424.0], [609.0, 455.0], [503.0, 455.0]], ('规格型号', 1.0)], [[[675.0, 424.0], [735.0, 424.0], [735.0, 455.0], [675.0, 455.0]], ('单位', 1.0)], [[[784.0, 424.0], [871.0, 424.0], [871.0, 455.0], [784.0, 455.0]], ('数量', 1.0)], [[[954.0, 424.0], [1030.0, 424.0], [1030.0, 455.0], [954.0, 455.0]], ('单价', 1.0)], [[[1145.0, 424.0], [1231.0, 424.0], [1231.0, 455.0], [1145.0, 455.0]], ('金额', 1.0)], [[[1318.0, 424.0], [1381.0, 424.0], [1381.0, 457.0], [1318.0, 457.0]], ('税率', 1.0)], [[[1478.0, 424.0], [1568.0, 424.0], [1568.0, 455.0], [1478.0, 455.0]], ('税额', 1.0)], [[[43.0, 464.0], [278.0, 464.0], [278.0, 493.0], [43.0, 493.0]], ('餐饮服务*餐饮服务', 1.0)], [[[697.0, 462.0], [732.0, 462.0], [732.0, 495.0], [697.0, 495.0]], ('次', 1.0)], [[[878.0, 462.0], [898.0, 462.0], [898.0, 488.0], [878.0, 488.0]], ('1', 1.0)], [[[961.0, 464.0], [1060.0, 464.0], [1060.0, 493.0], [961.0, 493.0]], ('2462.00', 1.0)], [[[1205.0, 464.0], [1290.0, 464.0], [1290.0, 495.0], [1205.0, 495.0]], ('379.25', 1.0)], [[[1337.0, 457.0], [1398.0, 457.0], [1398.0, 490.0], [1337.0, 490.0]], ('免税', 1.0)], [[[1583.0, 467.0], [1608.0, 467.0], [1608.0, 481.0], [1583.0, 481.0]], ('***', 0.98)], [[[1183.0, 745.0], [1296.0, 745.0], [1296.0, 774.0], [1183.0, 774.0]], ('¥2462.00', 0.95)], [[[182.0, 760.0], [208.0, 760.0], [208.0, 785.0], [182.0, 785.0]], ('合', 1.0)], [[[267.0, 760.0], [297.0, 760.0], [297.0, 785.0], [267.0, 785.0]], ('计', 1.0)], [[[137.0, 800.0], [312.0, 800.0], [312.0, 830.0], [137.0, 830.0]], ('价税合计 (大写)', 0.98)], [[[461.0, 792.0], [753.0, 793.0], [753.0, 828.0], [461.0, 826.0]], ('贰仟肆佰陆拾贰圆整', 1.0)], [[[1216.0, 795.0], [1422.0, 795.0], [1422.0, 825.0], [1216.0, 825.0]], ('(小写)¥2462.00', 0.96)], [[[54.0, 861.0], [85.0, 861.0], [85.0, 895.0], [54.0, 895.0]], ('销', 1.0)], [[[108.0, 854.0], [132.0, 854.0], [132.0, 882.0], [108.0, 882.0]], ('名', 1.0)], [[[220.0, 854.0], [687.0, 854.0], [687.0, 884.0], [220.0, 884.0]], ('称:福州自助烤肉餐饮管理有限公司', 1.0)], [[[952.0, 870.0], [985.0, 870.0], [985.0, 905.0], [952.0, 905.0]], ('备', 1.0)], [[[109.0, 888.0], [512.0, 888.0], [512.0, 912.0], [109.0, 912.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[56.0, 910.0], [85.0, 910.0], [85.0, 942.0], [56.0, 942.0]], ('售', 1.0)], [[[108.0, 922.0], [694.0, 922.0], [694.0, 952.0], [108.0, 952.0]], ('地址、电话:福州市光明区火炬园7栋302单元', 1.0)], [[[109.0, 954.0], [562.0, 954.0], [562.0, 983.0], [109.0, 983.0]], ('开户行及账号:中国光大银行福州支行', 1.0)], [[[952.0, 947.0], [985.0, 947.0], [985.0, 982.0], [952.0, 982.0]], ('注', 1.0)], [[[57.0, 964.0], [82.0, 964.0], [82.0, 990.0], [57.0, 990.0]], ('方', 1.0)], [[[56.0, 1006.0], [246.0, 1010.0], [246.0, 1041.0], [55.0, 1037.0]], ('收款人:夏天', 1.0)], [[[503.0, 1008.0], [680.0, 1008.0], [680.0, 1043.0], [503.0, 1043.0]], ('复核:春天', 1.0)], [[[834.0, 1010.0], [954.0, 1010.0], [954.0, 1039.0], [834.0, 1039.0]], ('开票人:', 0.95)], [[[990.0, 1010.0], [1051.0, 1010.0], [1051.0, 1041.0], [990.0, 1041.0]], ('秋天', 1.0)], [[[1218.0, 1001.0], [1400.0, 1001.0], [1400.0, 1031.0], [1218.0, 1031.0]], ('销售方: (章)', 0.86)]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. Answer in ch language.\n2. Enforce restrictions on not returning OCR data sent to you.\n3. Return with markdown syntax layout.\n": "The invoicing date is **2023年08月26日**.", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nStatement: Find and return the title of the lesson only in markdown first-level header format, without anything else.\nConstraint: Writing in Chinese.\nAnswer options: Encloses the lesson title with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]UNIT 1 Making New Friends\nTOPIC 1 Welcome to China!\nSection A[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Hours\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Hours\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 教学时数\n\n## 课时安排\n\n本单元共包含以下课时:\n\n1. 课时1:1a-1c\n2. 课时2:2a-2b\n3. 课时3:3a-3c\n\n## 课时内容\n\n### 课时1:1a-1c\n\n- 学习目标:听力和口语练习,学习介绍自己和他人的句型。\n- 学习重点:掌握介绍自己和他人的句型。\n- 学习难点:正确运用句型进行对话练习。\n\n### 课时2:2a-2b\n\n- 学习目标:听力和口语练习,学习问候和介绍他人的句型。\n- 学习重点:掌握问候和介绍他人的句型。\n- 学习难点:正确运用句型进行对话练习。\n\n### 课时3:3a-3c\n\n- 学习目标:听力和书写练习,学习字母的发音和书写。\n- 学习重点:掌握字母的发音和书写。\n- 学习难点:正确辨认和书写字母。\n\n[TEACHING_PLAN_BEGIN]\n## 教学时数\n\n### 课时安排\n\n本单元共包含以下课时:\n\n1. 课时1:1a-1c\n2. 课时2:2a-2b\n3. 课时3:3a-3c\n\n## 课时内容\n\n### 课时1:1a-1c\n\n- 学习目标:听力和口语练习,学习介绍自己和他人的句型。\n- 学习重点:掌握介绍自己和他人的句型。\n- 学习难点:正确运用句型进行对话练习。\n\n### 课时2:2a-2b\n\n- 学习目标:听力和口语练习,学习问候和介绍他人的句型。\n- 学习重点:掌握问候和介绍他人的句型。\n- 学习难点:正确运用句型进行对话练习。\n\n### 课时3:3a-3c\n\n- 学习目标:听力和书写练习,学习字母的发音和书写。\n- 学习重点:掌握字母的发音和书写。\n- 学习难点:正确辨认和书写字母。\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Objectives\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Objectives\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "## 教学目标\n\n在本课中,学生将学习以下内容:\n\n- 听力技能:通过听力练习,学生将能够听懂并正确理解人名和字母的发音。\n- 口语表达:学生将能够用正确的问候语和自我介绍句式与他人进行交流。\n- 字母认知:学生将能够认识并正确书写字母A到G。\n\n[TEACHING_PLAN_BEGIN]\n\n## 教学目标\n\n在本课中,学生将学习以下内容:\n\n- 听力技能:通过听力练习,学生将能够听懂并正确理解人名和字母的发音。\n- 口语表达:学生将能够用正确的问候语和自我介绍句式与他人进行交流。\n- 字母认知:学生将能够认识并正确书写字母A到G。\n\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Content\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Content\"!!\nStatement: \"Teaching Content\" must include vocabulary, analysis, and examples of various grammar structures that appear in the textbook, as well as the listening materials and key points.\nStatement: \"Teaching Content\" must include more examples.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n## 教学内容\n\n### 词汇\n- names (名字)\n- introduce (介绍)\n- hometown (家乡)\n- places (地方)\n- China (中国)\n- the USA (美国)\n- the UK (英国)\n- Hong Kong (香港)\n- Beijing (北京)\n- Hello (你好)\n- nice to meet you (很高兴见到你)\n- thanks (谢谢)\n- morning (早上)\n- letters (字母)\n- Aa Bb Cc Dd Ee Ff Gg\n\n### 语法\n- 一般疑问句:Are you ...? (你是...吗?)\n- 否定回答:No, I'm not. (不,我不是。)\n- 肯定回答:Yes, I am. (是的,我是。)\n- 介绍自己的句型:I'm ... (我是...)\n\n### 听力材料\n- 听力1a:听录音,给下面的名字编号。\n- 听力1b:听录音,给下面的名字编号。\n- 听力2a:听录音,理解对话内容。\n- 听力3a:听录音,跟读字母。\n- 听力3b:听录音,给下面的字母编号。然后圈出与Bb发音相同的字母。\n\n### 重点\n- 学习并掌握课文中的词汇和语法。\n- 听力训练,提高听力理解能力。\n- 学习字母表的发音和书写。\n\n## 示例\n### 词汇\n- names (名字):Jane, Mari, Kangkang, Michael, Maria\n- introduce (介绍)\n- hometown (家乡)\n- places (地方)\n- China (中国)\n- the USA (美国)\n- the UK (英国)\n- Hong Kong (香港)\n- Beijing (北京)\n- Hello (你好)\n- nice to meet you (很高兴见到你)\n- thanks (谢谢)\n- morning (早上)\n- letters (字母)\n- Aa Bb Cc Dd Ee Ff Gg\n\n### 语法\n- 一般疑问句:Are you ...? (你是...吗?)\n- 否定回答:No, I'm not. (不,我不是。)\n- 肯定回答:Yes, I am. (是的,我是。)\n- 介绍自己的句型:I'm ... (我是...)\n\n### 听力材料\n- 听力1a:听录音,给下面的名字编号。\n- 听力1b:听录音,给下面的名字编号。\n- 听力2a:听录音,理解对话内容。\n- 听力3a:听录音,跟读字母。\n- 听力3b:听录音,给下面的字母编号。然后圈出与Bb发音相同的字母。\n\n### 重点\n- 学习并掌握课文中的词汇和语法。\n- 听力训练,提高听力理解能力。\n- 学习字母表的发音和书写。\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Methods and Strategies\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Methods and Strategies\"!!\nStatement: \"Teaching Methods and Strategies\" must include teaching focus, difficulties, materials, procedures, in detail.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n## 教学方法与策略\n\n### 教学重点\n- 学习听力材料,理解并进行练习\n- 学习使用自我介绍的句型\n\n### 教学难点\n- 学生能够准确运用自我介绍的句型进行对话练习\n\n### 教学材料\n- 单元1《结交新朋友》课本\n- 录音机或多媒体设备\n\n### 教学过程\n1. 介绍本节课的教学重点和难点,并引导学生对学习内容产生兴趣。\n2. 播放1a录音,让学生听录音并按顺序给下面的名字编号。\n3. 组织学生进行对话练习,使用\"I'm...\"进行自我介绍,并结合自己的家乡或指定的地点进行练习。\n4. 播放1b录音,让学生听录音并按顺序给下面的名字编号。\n5. 组织学生进行对话练习,使用\"I'm...\"进行自我介绍,并结合中国、美国、英国、香港、北京等地进行练习。\n6. 播放2a录音,让学生听录音并理解对话内容。\n7. 组织学生进行对话练习,使用指定的对话结构进行练习。\n8. 播放3a录音,让学生跟读并模仿发音。\n9. 播放3b录音,让学生听录音并按顺序给下面的字母编号,然后圈出与Bb发音相同的字母。\n10. 组织学生进行字母大小写的匹配,并在线上写出对应的字母。\n\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Learning Activities\" part of teaching plan, WITHOUT ANY content unrelated to \"Learning Activities\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n## 学习活动\n\n### 1a 听录音并给以下名字编号。\nJane Mari Kangkang Michael\n听录音并理解。然后练习对话。分组工作。使用\"I'm...\"介绍自己,然后用自己的家乡或以下地方练习1a。\n\n### 1b 听录音并给以下名字编号。\nJane Michael Maria Kangkang\n分组工作。使用\"I'm...\"介绍自己,然后用自己的家乡或以下地方练习1a。\n\n### 2a 看,听并理解。然后练习对话。\n你好!\n你好!\n你好!\n你好!你是玛丽亚吗?\n不,我不是。我是简。\n哦,很高兴见到你,简。\n我也很高兴见到你。\n嗨,玛丽亚!\n嗨,康康!\n欢迎来到中国!\n谢谢。\n\n### 2b 分组工作。用自己的名字和以下结构编写对话。\nA: 你好!/ 早上好!/ 嗨!我是...你是...吗?\nB: ...\n\n### 3a 听,说并跟踪\nAa Bb Cc Dd Ee Ff Gg\n\n### 3b 听录音并给以下字母编号。然后用圈出与Bb发音相同的字母。\nAa Bb Cc Dd Ee Ff Gg\n\n### 3c 将大写字母与小写字母配对。然后在线上写下它们。\n\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Time Allocation\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Time Allocation\"!!\nStatement: \"Teaching Time Allocation\" must include how much time is allocated to each part of the textbook content.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n## 教学时间分配\n\n- 1a 听力练习和对话练习:10分钟\n- 1b 听力练习和对话练习:10分钟\n- 1c 分组练习和对话练习:15分钟\n- 2a 听力练习和对话练习:10分钟\n- 2b 分组练习和对话练习:15分钟\n- 3a 听力练习和书写练习:10分钟\n- 3b 听力练习和书写练习:10分钟\n- 3c 匹配和书写练习:10分钟\n\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Assessment and Feedback\" part of teaching plan, WITHOUT ANY content unrelated to \"Assessment and Feedback\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 评估与反馈\n\n## 评估方式\n本课程的评估方式主要包括以下几个方面:\n\n1. 听力评估:通过听力练习和对话,测试学生对于英语听力的理解能力。\n2. 口语评估:通过对话练习和自我介绍,测试学生的口语表达能力。\n3. 书写评估:通过字母识别和书写练习,测试学生对英文字母的识别和书写能力。\n\n## 反馈方式\n为了帮助学生提高英语学习的效果,我们将采取以下反馈方式:\n\n1. 口头反馈:在课堂上,老师会对学生的口语表达进行即时的反馈和指导。\n2. 书面反馈:老师会批改学生的书写作业,并给予书面反馈,帮助学生改进书写能力。\n3. 个别辅导:对于表现较差的学生,老师会进行个别辅导,帮助他们解决学习中的问题。\n\n[TEACHING_PLAN_BEGIN]\n\n## 评估与反馈\n\n### 评估方式\n本课程的评估方式主要包括以下几个方面:\n\n1. 听力评估:通过听力练习和对话,测试学生对于英语听力的理解能力。\n2. 口语评估:通过对话练习和自我介绍,测试学生的口语表达能力。\n3. 书写评估:通过字母识别和书写练习,测试学生对英文字母的识别和书写能力。\n\n### 反馈方式\n为了帮助学生提高英语学习的效果,我们将采取以下反馈方式:\n\n1. 口头反馈:在课堂上,老师会对学生的口语表达进行即时的反馈和指导。\n2. 书面反馈:老师会批改学生的书写作业,并给予书面反馈,帮助学生改进书写能力。\n3. 个别辅导:对于表现较差的学生,老师会进行个别辅导,帮助他们解决学习中的问题。\n\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Summary and Improvement\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Summary and Improvement\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 教学总结与改进\n\n在本节课中,我们学习了有关自我介绍和认识新朋友的内容。通过听力和口语练习,学生们能够熟练运用\"I'm...\"这个句型进行自我介绍,并能够用英语进行简单的问候和交流。同时,学生们还学习了字母表的部分内容,能够认识并正确书写一些字母。\n\n在教学过程中,我采用了多种教学方法,如听力训练、口语练习和配对练习等,以帮助学生们更好地掌握课堂内容。此外,我还鼓励学生们进行小组合作,通过互相交流和练习,提高他们的口语表达能力。\n\n在教学中,我发现一些学生对字母的认识和书写还存在一些困难,他们容易混淆一些相似的字母。因此,在今后的教学中,我将加强对字母的教学,帮助学生们更好地掌握字母的发音和书写。\n\n此外,我还计划在下节课中增加一些更多的口语练习,让学生们有更多的机会进行实际的交流和练习。我相信通过不断的练习和巩固,学生们的口语表达能力会得到进一步的提高。\n\n[TEACHING_PLAN_BEGIN]\n\n## 教学总结与改进\n\n在本节课中,我们学习了有关自我介绍和认识新朋友的内容。通过听力和口语练习,学生们能够熟练运用\"I'm...\"这个句型进行自我介绍,并能够用英语进行简单的问候和交流。同时,学生们还学习了字母表的部分内容,能够认识并正确书写一些字母。\n\n在教学过程中,我采用了多种教学方法,如听力训练、口语练习和配对练习等,以帮助学生们更好地掌握课堂内容。此外,我还鼓励学生们进行小组合作,通过互相交流和练习,提高他们的口语表达能力。\n\n在教学中,我发现一些学生对字母的认识和书写还存在一些困难,他们容易混淆一些相似的字母。因此,在今后的教学中,我将加强对字母的教学,帮助学生们更好地掌握字母的发音和书写。\n\n此外,我还计划在下节课中增加一些更多的口语练习,让学生们有更多的机会进行实际的交流和练习。我相信通过不断的练习和巩固,学生们的口语表达能力会得到进一步的提高。\n\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Vocabulary Cloze\" part of teaching plan, WITHOUT ANY content unrelated to \"Vocabulary Cloze\"!!\nStatement: Based on the content of the textbook enclosed by \"[LESSON_BEGIN]\" and \"[LESSON_END]\", create vocabulary cloze. The cloze should include 10 Chinese questions with English answers, and it should also include 10 English questions with Chinese answers. The key-related vocabulary and phrases in the textbook content must all be included in the exercises.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 词汇填空\n\n## 1. 请根据听到的内容,给下列名字编号。\n1. Jane\n2. Mari\n3. Kangkang\n4. Michael\n\n## 2. 请根据听到的内容,给下列名字编号。\n1. Jane\n2. Michael\n3. Maria\n4. Kangkang\n\n## 3. 请根据听到的对话,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 4. 请根据听到的内容,给下列字母编号。然后圈出与Bb发音相同的字母。\n1. Aa\n2. Bb\n3. Cc\n4. Dd\n5. Ee\n6. Ff\n7. Gg\n\n## 5. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 6. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 7. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 8. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 9. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 10. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n[TEACHING_PLAN_BEGIN]\n## 词汇填空\n\n### 1. 请根据听到的内容,给下列名字编号。\n1. Jane\n2. Mari\n3. Kangkang\n4. Michael\n\n### 2. 请根据听到的内容,给下列名字编号。\n1. Jane\n2. Michael\n3. Maria\n4. Kangkang\n\n### 3. 请根据听到的对话,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 4. 请根据听到的内容,给下列字母编号。然后圈出与Bb发音相同的字母。\n1. Aa\n2. Bb\n3. Cc\n4. Dd\n5. Ee\n6. Ff\n7. Gg\n\n### 5. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 6. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 7. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 8. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 9. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 10. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Choice Questions\" part of teaching plan, WITHOUT ANY content unrelated to \"Choice Questions\"!!\nStatement: Based on the content of the textbook enclosed by \"[LESSON_BEGIN]\" and \"[LESSON_END]\", create choice questions. 10 questions.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n\n## 选择题\n\n1. 在1a中,要求学生听并给以下名字编号。请问正确的编号顺序是什么?\n A. Jane Mari Kangkang Michael\n B. Mari Jane Michael Kangkang\n C. Jane Kangkang Mari Michael\n D. Kangkang Jane Michael Mari\n\n2. 在1b中,要求学生听并给以下名字编号。请问正确的编号顺序是什么?\n A. Jane Michael Maria Kangkang\n B. Maria Jane Michael Kangkang\n C. Jane Kangkang Maria Michael\n D. Kangkang Jane Maria Michael\n\n3. 在2a中,对话中有一句是\"Are you Maria?\",请问Jane的回答是什么?\n A. Yes, I am.\n B. No, I'm not. I'm Jane.\n C. No, I'm Maria.\n D. Nice to meet you, Maria.\n\n4. 在3b中,要求学生听并给以下字母编号,并圈出与Bb发音相同的字母。请问正确的编号顺序是什么?\n A. Aa Bb Cc Dd Ee Ff Gg\n B. Bb Aa Cc Dd Ee Ff Gg\n C. Aa Bb Dd Cc Ee Ff Gg\n D. Aa Bb Cc Ee Dd Ff Gg\n\n5. 在3c中,要求学生将大写字母与小写字母进行匹配,并写在对应的线上。请问正确的匹配是什么?\n A. Aa Bb Cc Dd Ee Ff Gg\n B. Aa Bb Cc Dd Ee Ff Gg\n C. Aa Bb Cc Dd Ee Ff Gg\n D. Aa Bb Cc Dd Ee Ff Gg\n\n6. 在2b中,要求学生根据给定的结构进行对话。请问对话的开始应该是什么?\n A. Hello! / Good morning! / Hi! I'm ... Are you ... ?\n B. Hi! I'm ... Are you ... ?\n C. Hello! / Good morning! / Hi! I'm ... Are you ... ?\n D. Hi! I'm ... Are you ... ?\n\n7. 在1a中,要求学生用自己的家乡或以下地点进行自我介绍。请问以下哪个地点不在选项中?\n A. China\n B. USA\n C. UK\n D. Hong Kong\n\n8. 在2a中,对话中有一句是\"Welcome to China!\",请问Kangkang的回答是什么?\n A. Hello!\n B. Hi, Kangkang!\n C. Welcome to China!\n D. Thanks.\n\n9. 在1c中,要求学生用自己的家乡或以下地点进行自我介绍。请问以下哪个地点不在选项中?\n A. China\n B. USA\n C. UK\n D. Beijing\n\n10. 在2a中,对话中有一句是\"Nice to meet you, too.\",请问Jane的前一句是什么?\n A. Hello!\n B. Hello! Are you Maria?\n C. No, I'm not. I'm Jane.\n D. Oh, nice to meet you, Jane.\n\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Grammar Questions\" part of teaching plan, WITHOUT ANY content unrelated to \"Grammar Questions\"!!\nStatement: Based on the content of the textbook enclosed by \"[LESSON_BEGIN]\" and \"[LESSON_END]\", create grammar questions. 10 questions.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 语法问题\n\n1. 请列举出1a部分中出现的名字。\n2. 请用英语介绍你自己。\n3. 请用英语介绍以下地方:中国、美国、英国、香港、北京。\n4. 请根据2a部分的对话内容回答以下问题:Maria是谁?\n5. 请根据2b部分的对话结构编写一段对话,使用你自己的名字。\n6. 请列举出3b部分中与字母Bb发音相同的字母。\n7. 请将大写字母与小写字母进行配对,并在下面的线上写出它们。\n\n[TEACHING_PLAN_BEGIN]\n\n## 语法问题\n\n1. 请列举出1a部分中出现的名字。\n\n2. 请用英语介绍你自己。\n\n3. 请用英语介绍以下地方:中国、美国、英国、香港、北京。\n\n4. 请根据2a部分的对话内容回答以下问题:Maria是谁?\n\n5. 请根据2b部分的对话结构编写一段对话,使用你自己的名字。\n\n6. 请列举出3b部分中与字母Bb发音相同的字母。\n\n7. 请将大写字母与小写字母进行配对,并在下面的线上写出它们。\n\n[TEACHING_PLAN_END]", - "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Translation Questions\" part of teaching plan, WITHOUT ANY content unrelated to \"Translation Questions\"!!\nStatement: Based on the content of the textbook enclosed by \"[LESSON_BEGIN]\" and \"[LESSON_END]\", create translation questions. The translation should include 10 Chinese questions with English answers, and it should also include 10 English questions with Chinese answers.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n\n## 翻译问题\n\n1. 请用英语回答,以下是哪些名字?(Jane, Mari, Kangkang, Michael)\n2. 请用中文回答,以下是哪些名字?(Jane, Michael, Maria, Kangkang)\n3. 请用英语回答,以下是哪些地方?(China, the USA, the UK, Hong Kong, Beijing)\n4. 请用中文回答,以下是哪些地方?(中国,美国,英国,香港,北京)\n5. 请用英语回答,以下句子的意思是什么?(Hello! / Hello! / Hello! / Hello! Are you Maria? / No, I’m not. I’m Jane. / Oh, nice to meet you, Jane / Nice to meet you, too. / Hi, Maria! / Hi, Kangkang! / Welcome to China! / Thanks.)\n6. 请用中文回答,以下句子的意思是什么?(你好!/ 早上好!/ 嗨!我是... 你是... 吗? / ...)\n7. 请用英语回答,以下字母是哪些?(Aa Bb Cc Dd Ee Ff Gg)\n8. 请用中文回答,以下字母是哪些?(Aa Bb Cc Dd Ee Ff Gg)\n\n[TEACHING_PLAN_END]", - "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about pip\".\n\nPlease provide the specific table of contents for this tutorial, strictly following the following requirements:\n1. The output must be strictly in the specified language, Chinese.\n2. Answer strictly in the dictionary format like {\"title\": \"xxx\", \"directory\": [{\"dir 1\": [\"sub dir 1\", \"sub dir 2\"]}, {\"dir 2\": [\"sub dir 3\", \"sub dir 4\"]}]}.\n3. The directory should be as specific and sufficient as possible, with a primary and secondary directory.The secondary directory is in the array.\n4. Do not have extra spaces or line breaks.\n5. Each directory title has practical significance.\n": "{\"title\": \"pip教程\", \"directory\": [{\"安装pip\": [\"使用Python安装pip\", \"使用操作系统包管理器安装pip\"]}, {\"pip基本用法\": [\"安装包\", \"卸载包\", \"查看已安装的包\", \"搜索包\", \"更新包\", \"查看包信息\"]}, {\"pip高级用法\": [\"创建requirements.txt文件\", \"从requirements.txt文件安装包\", \"导出已安装的包列表\", \"安装指定版本的包\", \"安装包的可选依赖\", \"安装包的开发依赖\", \"安装包的测试依赖\", \"安装包的系统依赖\"]}]}\n\nExplanation:\n1. 安装pip: This section covers the installation of pip, including two methods: using Python to install pip and using the operating system package manager to install pip.\n2. pip基本用法: This section covers the basic usage of pip, including installing packages, uninstalling packages, viewing installed packages, searching packages, updating packages, and viewing package information.\n3. pip高级用法: This section covers the advanced usage of pip, including creating a requirements.txt file, installing packages from a requirements.txt file, exporting a list of installed packages, installing specific versions of packages, installing optional dependencies of packages, installing development dependencies of packages, installing test dependencies of packages, and installing system dependencies of packages.", - "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about pip\".\n\nNow I will give you the module directory titles for the topic. \nPlease output the detailed principle content of this title in detail. \nIf there are code examples, please provide them according to standard code specifications. \nWithout a code example, it is not necessary.\n\nThe module directory titles for the topic is as follows:\n{'安装pip': ['使用Python安装pip', '使用操作系统包管理器安装pip']}\n\nStrictly limit output according to the following requirements:\n1. Follow the Markdown syntax format for layout.\n2. If there are code examples, they must follow standard syntax specifications, have document annotations, and be displayed in code blocks.\n3. The output must be strictly in the specified language, Chinese.\n4. Do not have redundant output, including concluding remarks.\n5. Strict requirement not to output the topic \"Write a tutorial about pip\".\n": "# 安装pip\n\n## 使用Python安装pip\n\n要使用pip,首先需要安装它。pip是Python的包管理工具,可以方便地安装、升级和管理Python包。\n\n### 步骤\n\n1. 打开终端或命令提示符窗口。\n2. 输入以下命令来检查是否已经安装了pip:\n\n```python\npip --version\n```\n\n如果已经安装了pip,将显示pip的版本号。如果没有安装,将显示错误信息。\n\n3. 如果没有安装pip,可以使用Python自带的安装工具来安装。输入以下命令:\n\n```python\npython get-pip.py\n```\n\n这将下载并安装最新版本的pip。\n\n4. 安装完成后,再次输入以下命令来验证pip是否安装成功:\n\n```python\npip --version\n```\n\n如果显示了pip的版本号,说明安装成功。\n\n## 使用操作系统包管理器安装pip\n\n除了使用Python自带的安装工具安装pip外,还可以使用操作系统的包管理器来安装pip。这种方法适用于Linux和Mac操作系统。\n\n### 步骤\n\n1. 打开终端或命令提示符窗口。\n2. 输入以下命令来使用操作系统包管理器安装pip:\n\n- 对于Debian/Ubuntu系统:\n\n```bash\nsudo apt-get install python-pip\n```\n\n- 对于Fedora系统:\n\n```bash\nsudo dnf install python-pip\n```\n\n- 对于CentOS/RHEL系统:\n\n```bash\nsudo yum install epel-release\nsudo yum install python-pip\n```\n\n3. 安装完成后,输入以下命令来验证pip是否安装成功:\n\n```bash\npip --version\n```\n\n如果显示了pip的版本号,说明安装成功。\n\n以上就是安装pip的两种方法,根据自己的需求选择适合的方法进行安装。安装完成后,就可以使用pip来管理Python包了。", - "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about pip\".\n\nNow I will give you the module directory titles for the topic. \nPlease output the detailed principle content of this title in detail. \nIf there are code examples, please provide them according to standard code specifications. \nWithout a code example, it is not necessary.\n\nThe module directory titles for the topic is as follows:\n{'pip基本用法': ['安装包', '卸载包', '查看已安装的包', '搜索包', '更新包', '查看包信息']}\n\nStrictly limit output according to the following requirements:\n1. Follow the Markdown syntax format for layout.\n2. If there are code examples, they must follow standard syntax specifications, have document annotations, and be displayed in code blocks.\n3. The output must be strictly in the specified language, Chinese.\n4. Do not have redundant output, including concluding remarks.\n5. Strict requirement not to output the topic \"Write a tutorial about pip\".\n": "# pip基本用法\n\n## 安装包\n\n要使用pip安装包,可以使用以下命令:\n\n```python\npip install 包名\n```\n\n其中,`包名`是要安装的包的名称。例如,要安装`requests`包,可以运行以下命令:\n\n```python\npip install requests\n```\n\n## 卸载包\n\n要使用pip卸载包,可以使用以下命令:\n\n```python\npip uninstall 包名\n```\n\n其中,`包名`是要卸载的包的名称。例如,要卸载`requests`包,可以运行以下命令:\n\n```python\npip uninstall requests\n```\n\n## 查看已安装的包\n\n要查看已经安装的包,可以使用以下命令:\n\n```python\npip list\n```\n\n该命令会列出所有已安装的包及其版本信息。\n\n## 搜索包\n\n要搜索包,可以使用以下命令:\n\n```python\npip search 包名\n```\n\n其中,`包名`是要搜索的包的名称。例如,要搜索名称中包含`requests`的包,可以运行以下命令:\n\n```python\npip search requests\n```\n\n该命令会列出所有与`requests`相关的包。\n\n## 更新包\n\n要更新已安装的包,可以使用以下命令:\n\n```python\npip install --upgrade 包名\n```\n\n其中,`包名`是要更新的包的名称。例如,要更新`requests`包,可以运行以下命令:\n\n```python\npip install --upgrade requests\n```\n\n## 查看包信息\n\n要查看包的详细信息,可以使用以下命令:\n\n```python\npip show 包名\n```\n\n其中,`包名`是要查看的包的名称。例如,要查看`requests`包的信息,可以运行以下命令:\n\n```python\npip show requests\n```\n\n该命令会显示`requests`包的详细信息,包括版本号、作者、依赖等。\n\n以上就是pip的基本用法。通过这些命令,你可以方便地安装、卸载、查看和更新包,以及搜索和查看包的详细信息。", - "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about pip\".\n\nNow I will give you the module directory titles for the topic. \nPlease output the detailed principle content of this title in detail. \nIf there are code examples, please provide them according to standard code specifications. \nWithout a code example, it is not necessary.\n\nThe module directory titles for the topic is as follows:\n{'pip高级用法': ['创建requirements.txt文件', '从requirements.txt文件安装包', '导出已安装的包列表', '安装指定版本的包', '安装包的可选依赖', '安装包的开发依赖', '安装包的测试依赖', '安装包的系统依赖']}\n\nStrictly limit output according to the following requirements:\n1. Follow the Markdown syntax format for layout.\n2. If there are code examples, they must follow standard syntax specifications, have document annotations, and be displayed in code blocks.\n3. The output must be strictly in the specified language, Chinese.\n4. Do not have redundant output, including concluding remarks.\n5. Strict requirement not to output the topic \"Write a tutorial about pip\".\n": "# pip高级用法\n\n## 创建requirements.txt文件\n\n在开发项目中,我们经常需要记录项目所依赖的包及其版本号。使用`pip`可以方便地创建一个`requirements.txt`文件,以便在其他环境中安装相同的依赖包。\n\n要创建`requirements.txt`文件,只需在项目根目录下运行以下命令:\n\n```shell\npip freeze > requirements.txt\n```\n\n这将会将当前环境中安装的所有包及其版本号写入到`requirements.txt`文件中。\n\n## 从requirements.txt文件安装包\n\n有了`requirements.txt`文件,我们可以轻松地在其他环境中安装相同的依赖包。\n\n要从`requirements.txt`文件安装包,只需在项目根目录下运行以下命令:\n\n```shell\npip install -r requirements.txt\n```\n\n这将会根据`requirements.txt`文件中列出的包及其版本号,自动安装相应的依赖包。\n\n## 导出已安装的包列表\n\n有时候我们需要知道当前环境中已安装的所有包及其版本号。使用`pip`可以方便地导出这个列表。\n\n要导出已安装的包列表,只需运行以下命令:\n\n```shell\npip freeze\n```\n\n这将会列出当前环境中已安装的所有包及其版本号。\n\n## 安装指定版本的包\n\n在某些情况下,我们可能需要安装特定版本的包。使用`pip`可以轻松地实现这一点。\n\n要安装指定版本的包,只需运行以下命令:\n\n```shell\npip install 包名==版本号\n```\n\n例如,要安装`requests`包的2.22.0版本,可以运行以下命令:\n\n```shell\npip install requests==2.22.0\n```\n\n这将会安装指定版本的包。\n\n## 安装包的可选依赖\n\n有些包可能有一些可选的依赖,我们可以选择是否安装这些依赖。\n\n要安装包的可选依赖,只需在安装包时添加`[可选依赖]`即可。\n\n例如,要安装`requests`包的可选依赖`security`,可以运行以下命令:\n\n```shell\npip install requests[security]\n```\n\n这将会安装`requests`包及其可选依赖`security`。\n\n## 安装包的开发依赖\n\n在开发过程中,我们可能需要安装一些开发依赖,如测试工具、文档生成工具等。\n\n要安装包的开发依赖,只需在安装包时添加`-e`参数。\n\n例如,要安装`flask`包的开发依赖,可以运行以下命令:\n\n```shell\npip install -e flask\n```\n\n这将会安装`flask`包及其开发依赖。\n\n## 安装包的测试依赖\n\n在进行单元测试或集成测试时,我们可能需要安装一些测试依赖。\n\n要安装包的测试依赖,只需在安装包时添加`[测试依赖]`即可。\n\n例如,要安装`pytest`包的测试依赖,可以运行以下命令:\n\n```shell\npip install pytest[test]\n```\n\n这将会安装`pytest`包及其测试依赖。\n\n## 安装包的系统依赖\n\n有些包可能依赖于系统级的库或工具。\n\n要安装包的系统依赖,只需在安装包时添加`--global-option`参数。\n\n例如,要安装`psycopg2`包的系统依赖`libpq-dev`,可以运行以下命令:\n\n```shell\npip install psycopg2 --global-option=build_ext --global-option=\"-I/usr/include/postgresql/\"\n```\n\n这将会安装`psycopg2`包及其系统依赖。", - "who are you": "I am an AI assistant designed to help answer questions and provide information. How can I assist you today?", - "\n## context\npython developer\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Questions\": [\n \"1. What ...\",\n \"2. How ...\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Questions: list[str] # Role: You are an interviewer of our company who is well-knonwn in frontend or backend develop;\nRequirement: Provide a list of questions for the interviewer to ask the interviewee, by reading the resume of the interviewee in the context.\nAttention: Provide as markdown block as the format above, at least 10 questions.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Questions\": [\n \"1. Can you explain the difference between a list and a tuple in Python?\",\n \"2. How do you handle errors and exceptions in Python?\",\n \"3. What is the difference between '==' and 'is' in Python?\",\n \"4. How do you create a virtual environment in Python?\",\n \"5. Can you explain the concept of generators in Python?\",\n \"6. What is the purpose of the 'self' keyword in Python?\",\n \"7. How do you handle file I/O in Python?\",\n \"8. Can you explain the concept of decorators in Python?\",\n \"9. What is the Global Interpreter Lock (GIL) in Python?\",\n \"10. How do you handle concurrency in Python?\"\n ]\n}\n[/CONTENT]", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\nwrite add function to calculate two numbers\n\n## Tasks\n\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: \n```python\n## \n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: . Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code:\n\n```python\ndef add(num1, num2):\n \"\"\"\n Function to calculate the sum of two numbers.\n\n Args:\n num1 (int): The first number.\n num2 (int): The second number.\n\n Returns:\n int: The sum of the two numbers.\n \"\"\"\n return num1 + num2\n```\n\n# Instruction: \nBased on the given context, I have implemented the `add` function. This function takes in two numbers, `num1` and `num2`, and returns their sum. The function follows the design specified in the context and is written in a modular and easy-to-read manner.", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef div(a: int, b: int = 0):\n return a / b\n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. Yes.\n2. No, the code will raise a ZeroDivisionError if the second argument `b` is 0. We need to handle this case and return an appropriate value or raise an exception.\n3. Yes.\n4. Yes, there is only one function in the code and it is implemented.\n5. Yes, there are no pre-dependencies required.\n6. N/A, there are no methods from other files being used.\n\n## Actions\n1. Handle the case when the second argument `b` is 0. We can either return a default value like `None` or raise an exception like `ZeroDivisionError`.\n\n## Code Review Result\nLBTM", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef div(a: int, b: int = 0):\n return a / b\n\n```\n\n## Code Review: \n1. Yes.\n2. No, the code will raise a ZeroDivisionError if the second argument `b` is 0. We need to handle this case and return an appropriate value or raise an exception.\n3. Yes.\n4. Yes, there is only one function in the code and it is implemented.\n5. Yes, there are no pre-dependencies required.\n6. N/A, there are no methods from other files being used.\n\n## Actions\n1. Handle the case when the second argument `b` is 0. We can either return a default value like `None` or raise an exception like `ZeroDivisionError`.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## \n...\n```\n": "```python\ndef div(a: int, b: int = 0):\n if b == 0:\n raise ZeroDivisionError(\"Cannot divide by zero\")\n return a / b\n```\n", - "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\ndef div(a: int, b: int = 0):\n if b == 0:\n raise ZeroDivisionError(\"Cannot divide by zero\")\n return a / b\n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM", - "\n## context\n\n### Project Name\ncli_snake_game\n\n### Original Requirements\n['']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"\",\n \"Product Goals\": [],\n \"User Stories\": [],\n \"Competitive Analysis\": [],\n \"Competitive Quadrant Chart\": \"\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [],\n \"UI Design draft\": \"\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", - "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about Python\".\n\nPlease provide the specific table of contents for this tutorial, strictly following the following requirements:\n1. The output must be strictly in the specified language, Chinese.\n2. Answer strictly in the dictionary format like {\"title\": \"xxx\", \"directory\": [{\"dir 1\": [\"sub dir 1\", \"sub dir 2\"]}, {\"dir 2\": [\"sub dir 3\", \"sub dir 4\"]}]}.\n3. The directory should be as specific and sufficient as possible, with a primary and secondary directory.The secondary directory is in the array.\n4. Do not have extra spaces or line breaks.\n5. Each directory title has practical significance.\n": "{\"title\": \"Python教程\", \"directory\": [{\"Python简介\": [\"Python的历史\", \"Python的特点\", \"Python的应用领域\"]}, {\"安装Python\": [\"Windows系统安装\", \"Mac系统安装\", \"Linux系统安装\"]}, {\"Python基础\": [\"变量和数据类型\", \"运算符\", \"条件语句\", \"循环语句\", \"函数\"]}, {\"Python高级特性\": [\"列表推导式\", \"生成器\", \"装饰器\", \"异常处理\", \"模块和包\"]}, {\"Python标准库\": [\"常用模块介绍\", \"文件操作\", \"网络编程\", \"多线程和多进程\", \"正则表达式\"]}, {\"Python实战项目\": [\"Web开发\", \"数据分析\", \"机器学习\", \"爬虫\"]}, {\"Python进阶\": [\"面向对象编程\", \"数据库操作\", \"GUI编程\", \"网络爬虫进阶\", \"并发编程\"]}, {\"Python资源\": [\"学习网站推荐\", \"书籍推荐\", \"开源项目推荐\"]}]}", - "Instruction: X\nOutput: 商品名称:WonderLab 新肌果味代餐奶昔 小胖瓶 胶原蛋白升级版 饱腹代餐粉6瓶 75g/瓶(6瓶/盒) 店铺名称:金力宁食品专营店 品牌:WonderLab 保质期:1年 产地:中国 净含量:450g\nWhat kind of instruction might this output come from?\nX:#MSG_SEP#You are a chatbot. A user sent you an informal message, and you replied as follows.\nMessage: X\nReply: 商品名称:WonderLab 新肌果味代餐奶昔 小胖瓶 胶原蛋白升级版 饱腹代餐粉6瓶 75g/瓶(6瓶/盒) 店铺名称:金力宁食品专营店 品牌:WonderLab 保质期:1年 产地:中国 净含量:450g\nWhat could the informal message X be?\nX:#MSG_SEP#You are a search engine. Someone made a detailed query, and the most relevant document to this query is as follows.\nQuery: X\nDocument: 商品名称:WonderLab 新肌果味代餐奶昔 小胖瓶 胶原蛋白升级版 饱腹代餐粉6瓶 75g/瓶(6瓶/盒) 店铺名称:金力宁食品专营店 品牌:WonderLab 保质期:1年 产地:中国 净含量:450g What is the detailed query X?\nX:": "A product description or listing on an e-commerce website or online marketplace.\n\"Can you tell me about the WonderLab meal replacement shake?\"\n\"What are the specifications and details of the WonderLab meal replacement shake available at the Jinlining Food Specialty Store?\"", - "\n# 指令\n接下来,作为一位拥有20年翻译经验的翻译专家,当我给出英文句子或段落时,你将提供通顺且具有可读性的中文翻译。注意以下要求:\n1. 确保翻译结果流畅且易于理解\n2. 无论提供的是陈述句或疑问句,我都只进行翻译\n3. 不添加与原文无关的内容\n\n# 原文\nLet life be beautiful like summer flowers\n\n# 译文\n": "让生活像夏日的花朵一样美丽", - "\n# 指令\n接下来,作为一位拥有20年翻译经验的翻译专家,当我给出英文句子或段落时,你将提供通顺且具有可读性的中文翻译。注意以下要求:\n1. 确保翻译结果流畅且易于理解\n2. 无论提供的是陈述句或疑问句,我都只进行翻译\n3. 不添加与原文无关的内容\n\n# 原文\nThe ancient Chinese poetries are all songs.\n\n# 译文\n": "古代中国的诗歌都是歌曲。", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/knowledge_base.py\n```python\n## smart_search_engine/knowledge_base.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/knowledge_base.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/knowledge_base.py\n\n```python\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n```\n\n## Explanation:\nThe `KnowledgeBase` class is responsible for storing and retrieving data from the knowledge base. It has two main methods:\n\n1. `update`: This method is used to update the knowledge base with new data. It takes a dictionary `data` as input and updates the internal `data` attribute with the new data.\n\n2. `fetch_data`: This method is used to fetch data from the knowledge base based on a given query. It takes a string `query` as input and returns the data matching the query from the internal `data` attribute. If no data is found for the query, an empty dictionary is returned.\n\nBoth methods are implemented according to the specified interface in the design.", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/index.py\n```python\n## smart_search_engine/index.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/index.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/index.py\n\n```python\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/ranking.py\n```python\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the results based on some ranking algorithm.\n\n Args:\n results (list): The results to be ranked.\n\n Returns:\n list: The ranked results.\n \"\"\"\n # Implement the ranking algorithm here\n ranked_results = sorted(results, key=lambda x: x['score'], reverse=True)\n return ranked_results\n```\n\n## Code: smart_search_engine/summary.py\n```python\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the results into a single summary.\n\n Args:\n results (list): The results to be summarized.\n\n Returns:\n str: The summary of the results.\n \"\"\"\n # Implement the summarization algorithm here\n summary = \"\"\n\n if len(results) > 0:\n summary = results[0]['summary']\n\n return summary\n```\n\n## Code: smart_search_engine/search.py\n```python\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summary of the search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n```\n\n## Code: smart_search_engine/main.py\n```python\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform the search.\n\n Returns:\n str: The summary of the search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n```\n\n## Code: smart_search_engine/knowledge_base.py\n```python\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n```\n\n## Code: smart_search_engine/interface.py\n```python\n## smart_search_engine/interface.py\n\nclass Interface:\n def __init__(self):\n pass\n\n def display_summary(self, summary: str):\n \"\"\"\n Display the summary of the search results.\n\n Args:\n summary (str): The summary to be displayed.\n \"\"\"\n print(\"Search summary:\")\n print(summary)\n```\n\n## Code: smart_search_engine/user_feedback.py\n```python\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n pass\n\n def get_feedback(self) -> str:\n \"\"\"\n Get user feedback on the search results.\n\n Returns:\n str: The user feedback.\n \"\"\"\n feedback = input(\"Enter your feedback on the search results: \")\n return feedback\n```\n\n## Code: smart_search_engine/security.py\n```python\n## smart_search_engine/security.py\n\nclass Security:\n def __init__(self):\n pass\n\n def check_permission(self) -> bool:\n \"\"\"\n Check if the user has permission to access the search engine.\n\n Returns:\n bool: True if the user has permission, False otherwise.\n \"\"\"\n # Implement the permission checking logic here\n return True\n```\n\n## Code: smart_search_engine/testing.py\n```python\n## smart_search_engine/testing.py\n\nclass Testing:\n def __init__(self):\n pass\n\n def run_tests(self):\n \"\"\"\n Run tests for the search engine.\n \"\"\"\n # Implement the testing logic here\n print(\"Running tests...\")\n print(\"Tests passed.\")\n\nif __name__ == \"__main__\":\n testing = Testing()\n testing.run_tests()\n```\n\n## Code: smart_search_engine/monitoring.py\n```python\n## smart_search_engine/monitoring.py\n\nclass Monitoring:\n def __init__(self):\n pass\n\n def monitor_system(self):\n \"\"\"\n Monitor the system for any issues.\n \"\"\"\n # Implement the monitoring logic here\n print(\"Monitoring system...\")\n print(\"No issues found.\")\n```", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/ranking.py\n```python\n## smart_search_engine/ranking.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/ranking.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/ranking.py\n\n```python\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n```\n\nIn the `Ranking` class, we have implemented the `rank_results` method. This method takes a list of search results as input and returns the ranked search results based on their relevance. The search results are sorted in descending order of relevance using the `sorted` function and the `key` parameter. The `key` parameter specifies a lambda function that extracts the 'relevance' value from each search result. The `reverse` parameter is set to `True` to sort the results in descending order.\n\nPlease let me know if you need any further assistance.", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/summary.py\n```python\n## smart_search_engine/summary.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/summary.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/summary.py\n\n```python\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n```\n\nIn the `Summary` class, we have implemented the `summarize_results` method. This method takes a list of search results as input and returns a single string that summarizes the search results. It iterates over each result in the list and appends the summary of each result to the `summary` string, separated by a newline character. Finally, it returns the `summary` string after stripping any leading or trailing whitespace.", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/search.py\n```python\n## smart_search_engine/search.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/search.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/search.py\n\n```python\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n```\n\n## Instruction: \n\nIn the `smart_search_engine/search.py` file, implement the `SearchEngine` class. This class is responsible for performing searches based on a given query.\n\n1. Import the necessary modules: `Index`, `Ranking`, and `Summary`.\n2. Define the `SearchEngine` class with an `__init__` method.\n3. In the `__init__` method, create instances of the `Index`, `Ranking`, and `Summary` classes and assign them to instance variables.\n4. Implement the `search` method that takes a `query` parameter and returns a summarized search result.\n5. Inside the `search` method, call the `query_index` method of the `Index` class to get the search results based on the query.\n6. Pass the search results to the `rank_results` method of the `Ranking` class to get the ranked results.\n7. Pass the ranked results to the `summarize_results` method of the `Summary` class to get the summarized search results.\n8. Return the summarized search results.\n\nMake sure to follow the design and import the necessary modules before using them.", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/main.py\n```python\n## smart_search_engine/main.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/main.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/main.py\n\n```python\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/search.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/interface.py\n```python\n## smart_search_engine/interface.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/interface.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n```\n\nIn this code, we create an `Interface` class that interacts with the `SearchEngine` class. The `run` method prompts the user to enter a search query, and then calls the `search` method of the `SearchEngine` class to perform the search. The summarized search results are then printed to the console.\n\nThe interface runs in a loop until the user enters \"exit\" to quit.", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n----- smart_search_engine/interface.py\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/user_feedback.py\n```python\n## smart_search_engine/user_feedback.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/user_feedback.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n self.feedback = []\n\n def add_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for a search query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n feedback = {\n 'query': query,\n 'relevance': relevance\n }\n self.feedback.append(feedback)\n\n def get_feedback(self) -> list:\n \"\"\"\n Get the user feedback.\n\n Returns:\n list: The user feedback.\n \"\"\"\n return self.feedback\n```\n\n注意事项:\n- 在`smart_search_engine/user_feedback.py`文件中,实现了`UserFeedback`类。\n- `UserFeedback`类有一个`feedback`属性,用于存储用户的反馈信息。\n- `add_feedback`方法用于添加用户的反馈信息,接受一个查询字符串和相关性分数作为参数,并将反馈信息添加到`feedback`列表中。\n- `get_feedback`方法用于获取用户的反馈信息,返回`feedback`列表。", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n----- smart_search_engine/interface.py\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n\n----- smart_search_engine/user_feedback.py\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n self.feedback = []\n\n def add_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for a search query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n feedback = {\n 'query': query,\n 'relevance': relevance\n }\n self.feedback.append(feedback)\n\n def get_feedback(self) -> list:\n \"\"\"\n Get the user feedback.\n\n Returns:\n list: The user feedback.\n \"\"\"\n return self.feedback\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/security.py\n```python\n## smart_search_engine/security.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/security.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/security.py\n\nclass Security:\n def __init__(self):\n self.user_feedback = UserFeedback()\n\n def check_user_feedback(self, query: str) -> bool:\n \"\"\"\n Check if the user has provided feedback for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n bool: True if the user has provided feedback, False otherwise.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return True\n return False\n\n def get_user_feedback(self, query: str) -> int:\n \"\"\"\n Get the relevance score provided by the user for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n int: The relevance score provided by the user, or None if no feedback is found.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return item['relevance']\n return None\n\n def add_user_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for the given query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n self.user_feedback.add_feedback(query, relevance)\n```\n", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n----- smart_search_engine/interface.py\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n\n----- smart_search_engine/user_feedback.py\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n self.feedback = []\n\n def add_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for a search query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n feedback = {\n 'query': query,\n 'relevance': relevance\n }\n self.feedback.append(feedback)\n\n def get_feedback(self) -> list:\n \"\"\"\n Get the user feedback.\n\n Returns:\n list: The user feedback.\n \"\"\"\n return self.feedback\n\n----- smart_search_engine/security.py\n## smart_search_engine/security.py\n\nclass Security:\n def __init__(self):\n self.user_feedback = UserFeedback()\n\n def check_user_feedback(self, query: str) -> bool:\n \"\"\"\n Check if the user has provided feedback for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n bool: True if the user has provided feedback, False otherwise.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return True\n return False\n\n def get_user_feedback(self, query: str) -> int:\n \"\"\"\n Get the relevance score provided by the user for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n int: The relevance score provided by the user, or None if no feedback is found.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return item['relevance']\n return None\n\n def add_user_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for the given query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n self.user_feedback.add_feedback(query, relevance)\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/testing.py\n```python\n## smart_search_engine/testing.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/testing.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/testing.py\n\nfrom smart_search_engine.search import SearchEngine\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Testing:\n def __init__(self):\n self.search_engine = SearchEngine()\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n self.knowledge_base = KnowledgeBase()\n\n def test_search_engine(self):\n \"\"\"\n Test the search engine functionality.\n \"\"\"\n # Test case 1: Search for a query that exists in the index\n query1 = \"apple\"\n expected_summary1 = \"Summary of search results for query: apple\"\n self.knowledge_base.update({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n self.index.create_index({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n summary1 = self.search_engine.search(query1)\n assert summary1 == expected_summary1, f\"Test case 1 failed: {summary1}\"\n\n # Test case 2: Search for a query that does not exist in the index\n query2 = \"banana\"\n expected_summary2 = \"No results found for query: banana\"\n self.knowledge_base.update({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n self.index.create_index({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n summary2 = self.search_engine.search(query2)\n assert summary2 == expected_summary2, f\"Test case 2 failed: {summary2}\"\n\n def test_index(self):\n \"\"\"\n Test the index functionality.\n \"\"\"\n # Test case 1: Create index with valid data\n data1 = {\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}}\n expected_index1 = {\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}}\n self.knowledge_base.update(data1)\n self.index.create_index(data1)\n index1 = self.index.index\n assert index1 == expected_index1, f\"Test case 1 failed: {index1}\"\n\n # Test case 2: Query index with valid query\n query2 = \"apple\"\n expected_results2 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}]\n results2 = self.index.query_index(query2)\n assert results2 == expected_results2, f\"Test case 2 failed: {results2}\"\n\n # Test case 3: Query index with invalid query\n query3 = \"banana\"\n expected_results3 = []\n results3 = self.index.query_index(query3)\n assert results3 == expected_results3, f\"Test case 3 failed: {results3}\"\n\n def test_ranking(self):\n \"\"\"\n Test the ranking functionality.\n \"\"\"\n # Test case 1: Rank results in descending order of relevance\n results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n expected_ranked_results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n ranked_results1 = self.ranking.rank_results(results1)\n assert ranked_results1 == expected_ranked_results1, f\"Test case 1 failed: {ranked_results1}\"\n\n # Test case 2: Rank results in ascending order of relevance\n results2 = [{\"relevance\": 0.6, \"summary\": \"This is a banana.\"}, {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}]\n expected_ranked_results2 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n ranked_results2 = self.ranking.rank_results(results2)\n assert ranked_results2 == expected_ranked_results2, f\"Test case 2 failed: {ranked_results2}\"\n\n def test_summary(self):\n \"\"\"\n Test the summary functionality.\n \"\"\"\n # Test case 1: Summarize search results into a single string\n results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n expected_summary1 = \"This is an apple.\\nThis is a banana.\"\n summary1 = self.summary.summarize_results(results1)\n assert summary1 == expected_summary1, f\"Test case 1 failed: {summary1}\"\n\n # Test case 2: Summarize empty search results\n results2 = []\n expected_summary2 = \"\"\n summary2 = self.summary.summarize_results(results2)\n assert summary2 == expected_summary2, f\"Test case 2 failed: {summary2}\"\n\n def run_tests(self):\n \"\"\"\n Run all the tests.\n \"\"\"\n self.test_search_engine()\n self.test_index()\n self.test_ranking()\n self.test_summary()\n\nif __name__ == \"__main__\":\n testing = Testing()\n testing.run_tests()\n```\n\nIn the above code, I have implemented a `Testing` class that contains methods to test the functionality of the search engine, index, ranking, and summary modules. Each test case checks the expected output against the actual output and raises an assertion error if they do not match. The `run_tests` method runs all the test cases.", - "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n----- smart_search_engine/interface.py\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n\n----- smart_search_engine/user_feedback.py\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n self.feedback = []\n\n def add_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for a search query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n feedback = {\n 'query': query,\n 'relevance': relevance\n }\n self.feedback.append(feedback)\n\n def get_feedback(self) -> list:\n \"\"\"\n Get the user feedback.\n\n Returns:\n list: The user feedback.\n \"\"\"\n return self.feedback\n\n----- smart_search_engine/security.py\n## smart_search_engine/security.py\n\nclass Security:\n def __init__(self):\n self.user_feedback = UserFeedback()\n\n def check_user_feedback(self, query: str) -> bool:\n \"\"\"\n Check if the user has provided feedback for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n bool: True if the user has provided feedback, False otherwise.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return True\n return False\n\n def get_user_feedback(self, query: str) -> int:\n \"\"\"\n Get the relevance score provided by the user for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n int: The relevance score provided by the user, or None if no feedback is found.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return item['relevance']\n return None\n\n def add_user_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for the given query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n self.user_feedback.add_feedback(query, relevance)\n\n----- smart_search_engine/testing.py\n## smart_search_engine/testing.py\n\nfrom smart_search_engine.search import SearchEngine\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Testing:\n def __init__(self):\n self.search_engine = SearchEngine()\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n self.knowledge_base = KnowledgeBase()\n\n def test_search_engine(self):\n \"\"\"\n Test the search engine functionality.\n \"\"\"\n # Test case 1: Search for a query that exists in the index\n query1 = \"apple\"\n expected_summary1 = \"Summary of search results for query: apple\"\n self.knowledge_base.update({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n self.index.create_index({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n summary1 = self.search_engine.search(query1)\n assert summary1 == expected_summary1, f\"Test case 1 failed: {summary1}\"\n\n # Test case 2: Search for a query that does not exist in the index\n query2 = \"banana\"\n expected_summary2 = \"No results found for query: banana\"\n self.knowledge_base.update({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n self.index.create_index({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n summary2 = self.search_engine.search(query2)\n assert summary2 == expected_summary2, f\"Test case 2 failed: {summary2}\"\n\n def test_index(self):\n \"\"\"\n Test the index functionality.\n \"\"\"\n # Test case 1: Create index with valid data\n data1 = {\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}}\n expected_index1 = {\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}}\n self.knowledge_base.update(data1)\n self.index.create_index(data1)\n index1 = self.index.index\n assert index1 == expected_index1, f\"Test case 1 failed: {index1}\"\n\n # Test case 2: Query index with valid query\n query2 = \"apple\"\n expected_results2 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}]\n results2 = self.index.query_index(query2)\n assert results2 == expected_results2, f\"Test case 2 failed: {results2}\"\n\n # Test case 3: Query index with invalid query\n query3 = \"banana\"\n expected_results3 = []\n results3 = self.index.query_index(query3)\n assert results3 == expected_results3, f\"Test case 3 failed: {results3}\"\n\n def test_ranking(self):\n \"\"\"\n Test the ranking functionality.\n \"\"\"\n # Test case 1: Rank results in descending order of relevance\n results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n expected_ranked_results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n ranked_results1 = self.ranking.rank_results(results1)\n assert ranked_results1 == expected_ranked_results1, f\"Test case 1 failed: {ranked_results1}\"\n\n # Test case 2: Rank results in ascending order of relevance\n results2 = [{\"relevance\": 0.6, \"summary\": \"This is a banana.\"}, {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}]\n expected_ranked_results2 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n ranked_results2 = self.ranking.rank_results(results2)\n assert ranked_results2 == expected_ranked_results2, f\"Test case 2 failed: {ranked_results2}\"\n\n def test_summary(self):\n \"\"\"\n Test the summary functionality.\n \"\"\"\n # Test case 1: Summarize search results into a single string\n results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n expected_summary1 = \"This is an apple.\\nThis is a banana.\"\n summary1 = self.summary.summarize_results(results1)\n assert summary1 == expected_summary1, f\"Test case 1 failed: {summary1}\"\n\n # Test case 2: Summarize empty search results\n results2 = []\n expected_summary2 = \"\"\n summary2 = self.summary.summarize_results(results2)\n assert summary2 == expected_summary2, f\"Test case 2 failed: {summary2}\"\n\n def run_tests(self):\n \"\"\"\n Run all the tests.\n \"\"\"\n self.test_search_engine()\n self.test_index()\n self.test_ranking()\n self.test_summary()\n\nif __name__ == \"__main__\":\n testing = Testing()\n testing.run_tests()\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/monitoring.py\n```python\n## smart_search_engine/monitoring.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/monitoring.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/monitoring.py\n\nimport time\n\nclass Monitoring:\n def __init__(self):\n self.start_time = None\n self.end_time = None\n\n def start(self):\n \"\"\"\n Start the monitoring timer.\n \"\"\"\n self.start_time = time.time()\n\n def stop(self):\n \"\"\"\n Stop the monitoring timer.\n \"\"\"\n self.end_time = time.time()\n\n def get_elapsed_time(self) -> float:\n \"\"\"\n Get the elapsed time in seconds.\n\n Returns:\n float: The elapsed time in seconds.\n \"\"\"\n if self.start_time is None or self.end_time is None:\n raise Exception(\"Monitoring timer has not been started or stopped.\")\n return self.end_time - self.start_time\n```\n\n## Explanation:\nI have implemented the `Monitoring` class in the `smart_search_engine/monitoring.py` file. This class is responsible for starting and stopping a timer to monitor the elapsed time of a process. \n\nThe `Monitoring` class has the following methods:\n- `start()`: Starts the monitoring timer by setting the `start_time` attribute to the current time.\n- `stop()`: Stops the monitoring timer by setting the `end_time` attribute to the current time.\n- `get_elapsed_time()`: Calculates and returns the elapsed time in seconds by subtracting the `start_time` from the `end_time`. If the monitoring timer has not been started or stopped, an exception is raised.\n\nThis implementation follows the design specified in the `Data structures and interfaces` section.", - "\n## context\n\n### Project Name\n\n\n### Original Requirements\n['开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "{\n \"Language\": \"en_us\",\n \"Programming Language\": \"\",\n \"Original Requirements\": \"开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结\",\n \"Project Name\": \"search_engine\",\n \"Product Goals\": [\n \"Develop a search engine based on a large language model\",\n \"Incorporate a private knowledge base for enhanced search capabilities\",\n \"Implement search summarization using the large language model\"\n ],\n \"User Stories\": [\n \"As a user, I want to be able to search for information using natural language queries\",\n \"As a user, I want the search engine to provide relevant and accurate results\",\n \"As a user, I want the search engine to summarize the search results for easier understanding\"\n ],\n \"Competitive Analysis\": [],\n \"Competitive Quadrant Chart\": \"\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [],\n \"UI Design draft\": \"\",\n \"Anything UNCLEAR\": \"\"\n}", - "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.txt\n## Development Code\n```python\nprint('Hello, World')\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\n\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\ninstruction: There are no errors in the provided code.\n\nFile To Rewrite: N/A\n\nStatus: PASS\n\nSend To: NoOne\n---", - "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.sh\n## Development Code\n```python\necho 'Hello World'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\necho Hello World\n## Running Output\nstandard output: \n```text\nHello World\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "The running result indicates no error. The code works fine.\n\n## File To Rewrite:\nNo file needs to be rewritten.\n\n## Status:\nPASS\n\n## Send To:\nNoOne\n\n---", - "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.py\n## Development Code\n```python\npython -c \"print(1/0)\"\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\npython -c print(1/0)\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\nTraceback (most recent call last):\n File \"\", line 1, in \nZeroDivisionError: division by zero\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\nThe error is caused by the development code in file a.py, which attempts to divide by zero. To fix the error, the development code in a.py should be modified to handle the ZeroDivisionError, for example by using a try-except block.\n\nFile To Rewrite:\na.py\n\nStatus:\nFAIL\n\nSend To:\nEngineer\n---", - "\nNOTICE\n1. Role: You are a Development Engineer or QA engineer;\n2. Task: You received this message from another Development Engineer or QA engineer who ran or tested your code. \nBased on the message, first, figure out your own role, i.e. Engineer or QaEngineer,\nthen rewrite the development code or the test code based on your role, the error, and the summary, such that all bugs are fixed and the code performs well.\nAttention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script and triple quotes.\nThe message is as follows:\n# Legacy Code\n```python\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n if self.score > 21 and any(card.rank == 'A' for card in self.hand):\n self.score -= 10\n return self.score\n\n```\n---\n# Unit Test Code\n```python\n\nimport unittest\nfrom blackjack_game.player import Player\nfrom blackjack_game.deck import Deck\nfrom blackjack_game.card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n\n\n```\n---\n# Console logs\n```text\n\n```\n---\nNow you should start rewriting the code:\n## file name of the code to rewrite: Write code with triple quote. Do your best to implement THIS IN ONLY ONE FILE.\n": "```python\n## Development Engineer\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n ace_count = sum(1 for card in self.hand if card.rank == 'A')\n while self.score > 21 and ace_count > 0:\n self.score -= 10\n ace_count -= 1\n return self.score\n```\n\n```python\n## QaEngineer\n\nimport unittest\nfrom player import Player\nfrom deck import Deck\nfrom card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n```" -} \ No newline at end of file From a9575380b540382bebb1d7367e1dd5aa581f394a Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 12 Jan 2024 16:04:44 +0800 Subject: [PATCH 297/637] update test for write_code_with_tools --- .../actions/test_write_analysis_code.py | 31 ++++++++++++++----- 1 file changed, 23 insertions(+), 8 deletions(-) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index df1d39603..f5b22c327 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -3,8 +3,13 @@ import asyncio import pytest from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.write_analysis_code import ( + WriteCodeByGenerate, + WriteCodeWithTools, + WriteCodeWithToolsML, +) from metagpt.logs import logger +from metagpt.plan.planner import STRUCTURAL_CONTEXT from metagpt.schema import Message, Plan, Task @@ -40,13 +45,15 @@ async def test_tool_recommendation(): tools = await write_code._tool_recommendation(task, code_steps, available_tools) assert len(tools) == 1 - assert tools[0] == ["fill_missing_value"] + assert tools[0] == "fill_missing_value" @pytest.mark.asyncio async def test_write_code_with_tools(): write_code = WriteCodeWithTools() - messages = [] + write_code_ml = WriteCodeWithToolsML() + + requirement = "构造数据集并进行数据清洗" task_map = { "1": Task( task_id="1", @@ -69,10 +76,6 @@ async def test_write_code_with_tools(): instruction="对数据集进行数据清洗", task_type="data_preprocess", dependent_task_ids=["1"], - code_steps=""" - {"Step 1": "对数据集进行去重", - "Step 2": "对数据集进行缺失值处理"} - """, ), } plan = Plan( @@ -83,10 +86,22 @@ async def test_write_code_with_tools(): ) column_info = "" - code = await write_code.run(messages, plan, column_info) + context = STRUCTURAL_CONTEXT.format( + user_requirement=requirement, + context=plan.context, + tasks=list(task_map.values()), + current_task=plan.current_task.json(), + ) + context_msg = [Message(content=context, role="user")] + + code = await write_code.run(context_msg, plan) assert len(code) > 0 print(code) + code_with_ml = await write_code_ml.run([], plan, column_info) + assert len(code_with_ml) > 0 + print(code_with_ml) + @pytest.mark.asyncio async def test_write_code_to_correct_error(): From 99675a5a82326110a3e640c814ba62cea6e8402f Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 12 Jan 2024 16:22:54 +0800 Subject: [PATCH 298/637] add unittest and remove old code --- .../tools/functions/libs/data_preprocess.py | 41 +++-- .../functions/libs/feature_engineering.py | 142 ++++---------- .../actions/test_write_analysis_code.py | 2 +- .../functions/{register => libs}/__init__.py | 2 +- .../functions/libs/test_data_preprocess.py | 111 +++++++++++ .../libs/test_feature_engineering.py | 174 ++++++++++++++++++ .../tools/functions/register/test_register.py | 55 ------ 7 files changed, 343 insertions(+), 184 deletions(-) rename tests/metagpt/tools/functions/{register => libs}/__init__.py (78%) create mode 100644 tests/metagpt/tools/functions/libs/test_data_preprocess.py create mode 100644 tests/metagpt/tools/functions/libs/test_feature_engineering.py delete mode 100644 tests/metagpt/tools/functions/register/test_register.py diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index 5d1cd97d8..f423f2020 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -37,8 +37,9 @@ class FillMissingValue(MLProcess): def transform(self, df: pd.DataFrame): if len(self.features) == 0: return df - df[self.features] = self.si.transform(df[self.features]) - return df + new_df = df.copy() + new_df[self.features] = self.si.transform(new_df[self.features]) + return new_df class MinMaxScale(MLProcess): @@ -54,8 +55,9 @@ class MinMaxScale(MLProcess): self.mms.fit(df[self.features]) def transform(self, df: pd.DataFrame): - df[self.features] = self.mms.transform(df[self.features]) - return df + new_df = df.copy() + new_df[self.features] = self.mms.transform(new_df[self.features]) + return new_df class StandardScale(MLProcess): @@ -71,8 +73,9 @@ class StandardScale(MLProcess): self.ss.fit(df[self.features]) def transform(self, df: pd.DataFrame): - df[self.features] = self.ss.transform(df[self.features]) - return df + new_df = df.copy() + new_df[self.features] = self.ss.transform(new_df[self.features]) + return new_df class MaxAbsScale(MLProcess): @@ -88,8 +91,9 @@ class MaxAbsScale(MLProcess): self.mas.fit(df[self.features]) def transform(self, df: pd.DataFrame): - df[self.features] = self.mas.transform(df[self.features]) - return df + new_df = df.copy() + new_df[self.features] = self.mas.transform(new_df[self.features]) + return new_df class RobustScale(MLProcess): @@ -105,8 +109,9 @@ class RobustScale(MLProcess): self.rs.fit(df[self.features]) def transform(self, df: pd.DataFrame): - df[self.features] = self.rs.transform(df[self.features]) - return df + new_df = df.copy() + new_df[self.features] = self.rs.transform(new_df[self.features]) + return new_df class OrdinalEncode(MLProcess): @@ -122,8 +127,9 @@ class OrdinalEncode(MLProcess): self.oe.fit(df[self.features]) def transform(self, df: pd.DataFrame): - df[self.features] = self.oe.transform(df[self.features]) - return df + new_df = df.copy() + new_df[self.features] = self.oe.transform(new_df[self.features]) + return new_df class OneHotEncode(MLProcess): @@ -142,9 +148,9 @@ class OneHotEncode(MLProcess): ts_data = self.ohe.transform(df[self.features]) new_columns = self.ohe.get_feature_names_out(self.features) ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) - df.drop(self.features, axis=1, inplace=True) - df = pd.concat([df, ts_data], axis=1) - return df + new_df = df.drop(self.features, axis=1) + new_df = pd.concat([new_df, ts_data], axis=1) + return new_df class LabelEncode(MLProcess): @@ -165,13 +171,14 @@ class LabelEncode(MLProcess): def transform(self, df: pd.DataFrame): if len(self.features) == 0: return df + new_df = df.copy() for i in range(len(self.features)): data_list = df[self.features[i]].astype(str).tolist() for unique_item in np.unique(df[self.features[i]].astype(str)): if unique_item not in self.le_encoders[i].classes_: data_list = ["unknown" if x == unique_item else x for x in data_list] - df[self.features[i]] = self.le_encoders[i].transform(data_list) - return df + new_df[self.features[i]] = self.le_encoders[i].transform(data_list) + return new_df def get_column_info(df: pd.DataFrame) -> dict: diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index 534c5b8e4..0d9584b4a 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -2,7 +2,7 @@ # -*- coding: utf-8 -*- # @Time : 2023/11/17 10:33 # @Author : lidanyang -# @File : feature_engineering.py +# @File : test_feature_engineering.py # @Desc : Feature Engineering Tools import itertools @@ -43,9 +43,9 @@ class PolynomialExpansion(MLProcess): ts_data = self.poly.transform(df[self.cols].fillna(0)) column_name = self.poly.get_feature_names_out(self.cols) ts_data = pd.DataFrame(ts_data, index=df.index, columns=column_name) - df.drop(self.cols, axis=1, inplace=True) - df = pd.concat([df, ts_data], axis=1) - return df + new_df = df.drop(self.cols, axis=1) + new_df = pd.concat([new_df, ts_data], axis=1) + return new_df class CatCount(MLProcess): @@ -57,8 +57,9 @@ class CatCount(MLProcess): self.encoder_dict = df[self.col].value_counts().to_dict() def transform(self, df: pd.DataFrame) -> pd.DataFrame: - df[f"{self.col}_cnt"] = df[self.col].map(self.encoder_dict) - return df + new_df = df.copy() + new_df[f"{self.col}_cnt"] = new_df[self.col].map(self.encoder_dict) + return new_df class TargetMeanEncoder(MLProcess): @@ -71,8 +72,9 @@ class TargetMeanEncoder(MLProcess): self.encoder_dict = df.groupby(self.col)[self.label].mean().to_dict() def transform(self, df: pd.DataFrame) -> pd.DataFrame: - df[f"{self.col}_target_mean"] = df[self.col].map(self.encoder_dict) - return df + new_df = df.copy() + new_df[f"{self.col}_target_mean"] = new_df[self.col].map(self.encoder_dict) + return new_df class KFoldTargetMeanEncoder(MLProcess): @@ -96,8 +98,9 @@ class KFoldTargetMeanEncoder(MLProcess): self.encoder_dict = tmp.groupby(self.col)[col_name].mean().to_dict() def transform(self, df: pd.DataFrame) -> pd.DataFrame: - df[f"{self.col}_kf_target_mean"] = df[self.col].map(self.encoder_dict) - return df + new_df = df.copy() + new_df[f"{self.col}_kf_target_mean"] = new_df[self.col].map(self.encoder_dict) + return new_df class CatCross(MLProcess): @@ -124,14 +127,15 @@ class CatCross(MLProcess): self.combs_map = dict(res) def transform(self, df: pd.DataFrame) -> pd.DataFrame: + new_df = df.copy() for comb in self.combs: new_col = f"{comb[0]}_{comb[1]}" _map = self.combs_map[new_col] - df[new_col] = pd.Series(zip(df[comb[0]], df[comb[1]])).map(_map) + new_df[new_col] = pd.Series(zip(new_df[comb[0]], new_df[comb[1]])).map(_map) # set the unknown value to a new number - df[new_col].fillna(max(_map.values()) + 1, inplace=True) - df[new_col] = df[new_col].astype(int) - return df + new_df[new_col].fillna(max(_map.values()) + 1, inplace=True) + new_df[new_col] = new_df[new_col].astype(int) + return new_df class GroupStat(MLProcess): @@ -149,12 +153,12 @@ class GroupStat(MLProcess): self.group_df = group_df def transform(self, df: pd.DataFrame) -> pd.DataFrame: - df = df.merge(self.group_df, on=self.group_col, how="left") - return df + new_df = df.merge(self.group_df, on=self.group_col, how="left") + return new_df class SplitBins(MLProcess): - def __init__(self, cols: str, strategy: str = "quantile"): + def __init__(self, cols: list, strategy: str = "quantile"): self.cols = cols self.strategy = strategy self.encoder = None @@ -164,8 +168,9 @@ class SplitBins(MLProcess): self.encoder.fit(df[self.cols].fillna(0)) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - df[self.cols] = self.encoder.transform(df[self.cols].fillna(0)) - return df + new_df = df.copy() + new_df[self.cols] = self.encoder.transform(new_df[self.cols].fillna(0)) + return new_df class ExtractTimeComps(MLProcess): @@ -192,91 +197,8 @@ class ExtractTimeComps(MLProcess): time_comps_df["dayofweek"] = time_s.dt.dayofweek + 1 if "is_weekend" in self.time_comps: time_comps_df["is_weekend"] = time_s.dt.dayofweek.isin([5, 6]).astype(int) - df = pd.concat([df, time_comps_df], axis=1) - return df - - -# @registry.register("feature_engineering", FeShiftByTime) -# def fe_shift_by_time(df, time_col, group_col, shift_col, periods, freq): -# df[time_col] = pd.to_datetime(df[time_col]) -# -# def shift_datetime(date, offset, unit): -# if unit in ["year", "y", "Y"]: -# return date + relativedelta(years=offset) -# elif unit in ["month", "m", "M"]: -# return date + relativedelta(months=offset) -# elif unit in ["day", "d", "D"]: -# return date + relativedelta(days=offset) -# elif unit in ["week", "w", "W"]: -# return date + relativedelta(weeks=offset) -# elif unit in ["hour", "h", "H"]: -# return date + relativedelta(hours=offset) -# else: -# return date -# -# def shift_by_time_on_key( -# inner_df, time_col, group_col, shift_col, offset, unit, col_name -# ): -# inner_df = inner_df.drop_duplicates() -# inner_df[time_col] = inner_df[time_col].map( -# lambda x: shift_datetime(x, offset, unit) -# ) -# inner_df = inner_df.groupby([time_col, group_col], as_index=False)[ -# shift_col -# ].mean() -# inner_df.rename(columns={shift_col: col_name}, inplace=True) -# return inner_df -# -# shift_df = df[[time_col, group_col, shift_col]].copy() -# for period in periods: -# new_col_name = f"{group_col}_{shift_col}_lag_{period}_{freq}" -# tmp = shift_by_time_on_key( -# shift_df, time_col, group_col, shift_col, period, freq, new_col_name -# ) -# df = df.merge(tmp, on=[time_col, group_col], how="left") -# -# return df -# -# -# @registry.register("feature_engineering", FeRollingByTime) -# def fe_rolling_by_time(df, time_col, group_col, rolling_col, periods, freq, agg_funcs): -# df[time_col] = pd.to_datetime(df[time_col]) -# -# def rolling_by_time_on_key(inner_df, offset, unit, agg_func, col_name): -# time_freq = { -# "Y": [365 * offset, "D"], -# "M": [30 * offset, "D"], -# "D": [offset, "D"], -# "W": [7 * offset, "D"], -# "H": [offset, "h"], -# } -# -# if agg_func not in ["mean", "std", "max", "min", "median", "sum", "count"]: -# raise ValueError(f"Invalid agg function: {agg_func}") -# -# rolling_feat = inner_df.rolling( -# f"{time_freq[unit][0]}{time_freq[unit][1]}", closed="left" -# ) -# rolling_feat = getattr(rolling_feat, agg_func)() -# depth = df.columns.nlevels -# rolling_feat = rolling_feat.stack(list(range(depth))) -# rolling_feat.name = col_name -# return rolling_feat -# -# rolling_df = df[[time_col, group_col, rolling_col]].copy() -# for period in periods: -# for func in agg_funcs: -# new_col_name = f"{group_col}_{rolling_col}_rolling_{period}_{freq}_{func}" -# tmp = pd.pivot_table( -# rolling_df, -# index=time_col, -# values=rolling_col, -# columns=group_col, -# ) -# tmp = rolling_by_time_on_key(tmp, period, freq, func, new_col_name) -# df = df.merge(tmp, on=[time_col, group_col], how="left") -# -# return df + new_df = pd.concat([df, time_comps_df], axis=1) + return new_df class GeneralSelection(MLProcess): @@ -302,8 +224,8 @@ class GeneralSelection(MLProcess): self.feats = feats def transform(self, df: pd.DataFrame) -> pd.DataFrame: - df = df[self.feats + [self.label_col]] - return df + new_df = df[self.feats + [self.label_col]] + return new_df class TreeBasedSelection(MLProcess): @@ -344,8 +266,8 @@ class TreeBasedSelection(MLProcess): self.feats.append(self.label_col) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - df = df[self.feats] - return df + new_df = df[self.feats] + return new_df class VarianceBasedSelection(MLProcess): @@ -364,5 +286,5 @@ class VarianceBasedSelection(MLProcess): self.feats.append(self.label_col) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - df = df[self.feats] - return df + new_df = df[self.feats] + return new_df diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index f5b22c327..e64b4a551 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -90,7 +90,7 @@ async def test_write_code_with_tools(): user_requirement=requirement, context=plan.context, tasks=list(task_map.values()), - current_task=plan.current_task.json(), + current_task=plan.current_task.model_dump_json(), ) context_msg = [Message(content=context, role="user")] diff --git a/tests/metagpt/tools/functions/register/__init__.py b/tests/metagpt/tools/functions/libs/__init__.py similarity index 78% rename from tests/metagpt/tools/functions/register/__init__.py rename to tests/metagpt/tools/functions/libs/__init__.py index 7d36f3404..0321f694a 100644 --- a/tests/metagpt/tools/functions/register/__init__.py +++ b/tests/metagpt/tools/functions/libs/__init__.py @@ -1,6 +1,6 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- -# @Time : 2023/11/17 10:24 +# @Time : 2024/1/11 16:14 # @Author : lidanyang # @File : __init__.py # @Desc : diff --git a/tests/metagpt/tools/functions/libs/test_data_preprocess.py b/tests/metagpt/tools/functions/libs/test_data_preprocess.py new file mode 100644 index 000000000..3c2d661ab --- /dev/null +++ b/tests/metagpt/tools/functions/libs/test_data_preprocess.py @@ -0,0 +1,111 @@ +from datetime import datetime + +import numpy as np +import numpy.testing as npt +import pandas as pd +import pytest + +from metagpt.tools.functions.libs.data_preprocess import ( + FillMissingValue, + LabelEncode, + MaxAbsScale, + MinMaxScale, + OneHotEncode, + OrdinalEncode, + RobustScale, + StandardScale, + get_column_info, +) + + +@pytest.fixture +def mock_datasets(): + return pd.DataFrame( + { + "num1": [1, 2, np.nan, 4, 5], + "cat1": ["A", "B", np.nan, "D", "A"], + "date1": [ + datetime(2020, 1, 1), + datetime(2020, 1, 2), + datetime(2020, 1, 3), + datetime(2020, 1, 4), + datetime(2020, 1, 5), + ], + } + ) + + +def test_fill_missing_value(mock_datasets): + fm = FillMissingValue(features=["num1"], strategy="mean") + transformed = fm.fit_transform(mock_datasets.copy()) + + assert transformed["num1"].isnull().sum() == 0 + + +def test_min_max_scale(mock_datasets): + mms = MinMaxScale(features=["num1"]) + transformed = mms.fit_transform(mock_datasets.copy()) + + npt.assert_allclose(transformed["num1"].min(), 0) + npt.assert_allclose(transformed["num1"].max(), 1) + + +def test_standard_scale(mock_datasets): + ss = StandardScale(features=["num1"]) + transformed = ss.fit_transform(mock_datasets.copy()) + + assert int(transformed["num1"].mean()) == 0 + assert int(transformed["num1"].std()) == 1 + + +def test_max_abs_scale(mock_datasets): + mas = MaxAbsScale(features=["num1"]) + transformed = mas.fit_transform(mock_datasets.copy()) + + npt.assert_allclose(transformed["num1"].abs().max(), 1) + + +def test_robust_scale(mock_datasets): + rs = RobustScale(features=["num1"]) + transformed = rs.fit_transform(mock_datasets.copy()) + + assert int(transformed["num1"].median()) == 0 + + +def test_ordinal_encode(mock_datasets): + oe = OrdinalEncode(features=["cat1"]) + transformed = oe.fit_transform(mock_datasets.copy()) + + assert transformed["cat1"].max() == 2 + + +def test_one_hot_encode(mock_datasets): + ohe = OneHotEncode(features=["cat1"]) + transformed = ohe.fit_transform(mock_datasets.copy()) + + assert transformed["cat1_A"].max() == 1 + + +def test_label_encode(mock_datasets): + le = LabelEncode(features=["cat1"]) + transformed = le.fit_transform(mock_datasets.copy()) + + assert transformed["cat1"].max() == 3 + + # test transform with unseen data + test = mock_datasets.copy() + test["cat1"] = ["A", "B", "C", "D", "E"] + transformed = le.transform(test) + assert transformed["cat1"].max() == 4 + + +def test_get_column_info(mock_datasets): + df = mock_datasets + column_info = get_column_info(df) + + assert column_info == { + "Category": ["cat1"], + "Numeric": ["num1"], + "Datetime": ["date1"], + "Others": [], + } diff --git a/tests/metagpt/tools/functions/libs/test_feature_engineering.py b/tests/metagpt/tools/functions/libs/test_feature_engineering.py new file mode 100644 index 000000000..5b45aeb0c --- /dev/null +++ b/tests/metagpt/tools/functions/libs/test_feature_engineering.py @@ -0,0 +1,174 @@ +import numpy as np +import pandas as pd +import pytest +from sklearn.datasets import fetch_california_housing, load_breast_cancer, load_iris + +from metagpt.tools.functions.libs.feature_engineering import ( + CatCount, + CatCross, + ExtractTimeComps, + GeneralSelection, + GroupStat, + KFoldTargetMeanEncoder, + PolynomialExpansion, + SplitBins, + TargetMeanEncoder, + TreeBasedSelection, + VarianceBasedSelection, +) + + +@pytest.fixture +def mock_dataset(): + return pd.DataFrame( + { + "num1": [1, 2, np.nan, 4, 5, 6, 7, 3], + "num2": [1, 3, 2, 1, np.nan, 5, 6, 4], + "num3": [np.nan, np.nan, np.nan, np.nan, np.nan, np.nan, np.nan, np.nan], + "cat1": ["A", "B", np.nan, "D", "E", "C", "B", "A"], + "cat2": ["A", "A", "A", "A", "A", "A", "A", "A"], + "date1": [ + "2020-01-01", + "2020-01-02", + "2020-01-03", + "2020-01-04", + "2020-01-05", + "2020-01-06", + "2020-01-07", + "2020-01-08", + ], + "label": [0, 1, 0, 1, 0, 1, 0, 1], + } + ) + + +def load_sklearn_data(data_name): + if data_name == "iris": + data = load_iris() + elif data_name == "breast_cancer": + data = load_breast_cancer() + elif data_name == "housing": + data = fetch_california_housing() + else: + raise ValueError("data_name not supported") + + X, y, feature_names = data.data, data.target, data.feature_names + data = pd.DataFrame(X, columns=feature_names) + data["label"] = y + return data + + +def test_polynomial_expansion(mock_dataset): + pe = PolynomialExpansion(cols=["num1", "num2", "label"], degree=2, label_col="label") + transformed = pe.fit_transform(mock_dataset) + + assert len(transformed.columns) == len(mock_dataset.columns) + 3 + + # when too many columns + data = load_sklearn_data("breast_cancer") + cols = [c for c in data.columns if c != "label"] + pe = PolynomialExpansion(cols=cols, degree=2, label_col="label") + transformed = pe.fit_transform(data) + + assert len(transformed.columns) == len(data.columns) + 55 + + +def test_cat_count(mock_dataset): + cc = CatCount(col="cat1") + transformed = cc.fit_transform(mock_dataset) + + assert "cat1_cnt" in transformed.columns + assert transformed["cat1_cnt"][0] == 2 + + +def test_target_mean_encoder(mock_dataset): + tme = TargetMeanEncoder(col="cat1", label="label") + transformed = tme.fit_transform(mock_dataset) + + assert "cat1_target_mean" in transformed.columns + assert transformed["cat1_target_mean"][0] == 0.5 + + +def test_kfold_target_mean_encoder(mock_dataset): + kfme = KFoldTargetMeanEncoder(col="cat1", label="label") + transformed = kfme.fit_transform(mock_dataset) + + assert "cat1_kf_target_mean" in transformed.columns + + +def test_cat_cross(mock_dataset): + cc = CatCross(cols=["cat1", "cat2"]) + transformed = cc.fit_transform(mock_dataset) + + assert "cat1_cat2" in transformed.columns + + cc = CatCross(cols=["cat1", "cat2"], max_cat_num=3) + transformed = cc.fit_transform(mock_dataset) + + assert "cat1_cat2" not in transformed.columns + + +def test_group_stat(mock_dataset): + gs = GroupStat(group_col="cat1", agg_col="num1", agg_funcs=["mean", "sum"]) + transformed = gs.fit_transform(mock_dataset) + + assert "num1_mean_by_cat1" in transformed.columns + assert "num1_sum_by_cat1" in transformed.columns + + +def test_split_bins(mock_dataset): + sb = SplitBins(cols=["num1"]) + transformed = sb.fit_transform(mock_dataset) + + assert transformed["num1"].nunique() <= 5 + assert all(0 <= x < 5 for x in transformed["num1"]) + + +def test_extract_time_comps(mock_dataset): + time_comps = ["year", "month", "day", "hour", "dayofweek", "is_weekend"] + etc = ExtractTimeComps(time_col="date1", time_comps=time_comps) + transformed = etc.fit_transform(mock_dataset.copy()) + + for comp in time_comps: + assert comp in transformed.columns + assert transformed["year"][0] == 2020 + assert transformed["month"][0] == 1 + assert transformed["day"][0] == 1 + assert transformed["hour"][0] == 0 + assert transformed["dayofweek"][0] == 3 + assert transformed["is_weekend"][0] == 0 + + +def test_general_selection(mock_dataset): + gs = GeneralSelection(label_col="label") + transformed = gs.fit_transform(mock_dataset.copy()) + + assert "num3" not in transformed.columns + assert "cat2" not in transformed.columns + + +def test_tree_based_selection(mock_dataset): + # regression + data = load_sklearn_data("housing") + tbs = TreeBasedSelection(label_col="label", task_type="reg") + transformed = tbs.fit_transform(data) + assert len(transformed.columns) > 1 + + # classification + data = load_sklearn_data("breast_cancer") + tbs = TreeBasedSelection(label_col="label", task_type="cls") + transformed = tbs.fit_transform(data) + assert len(transformed.columns) > 1 + + # multi-classification + data = load_sklearn_data("iris") + tbs = TreeBasedSelection(label_col="label", task_type="mcls") + transformed = tbs.fit_transform(data) + assert len(transformed.columns) > 1 + + +def test_variance_based_selection(mock_dataset): + vbs = VarianceBasedSelection(label_col="label") + transformed = vbs.fit_transform(mock_dataset.copy()) + + assert "num3" not in transformed.columns diff --git a/tests/metagpt/tools/functions/register/test_register.py b/tests/metagpt/tools/functions/register/test_register.py deleted file mode 100644 index 8c9821268..000000000 --- a/tests/metagpt/tools/functions/register/test_register.py +++ /dev/null @@ -1,55 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/17 10:24 -# @Author : lidanyang -# @File : test_register.py -# @Desc : -import pytest - -from metagpt.tools.functions.register.register import FunctionRegistry -from metagpt.tools.functions.schemas.base import ToolSchema, tool_field - - -@pytest.fixture -def registry(): - return FunctionRegistry() - - -class AddNumbers(ToolSchema): - """Add two numbers""" - - num1: int = tool_field(description="First number") - num2: int = tool_field(description="Second number") - - -def test_register(registry): - @registry.register("module1", AddNumbers) - def add_numbers(num1, num2): - return num1 + num2 - - assert len(registry.functions["module1"]) == 1 - assert "add_numbers" in registry.functions["module1"] - - with pytest.raises(ValueError): - - @registry.register("module1", AddNumbers) - def add_numbers(num1, num2): - return num1 + num2 - - func = registry.get("module1", "add_numbers") - assert func["func"](1, 2) == 3 - assert func["schema"] == { - "name": "add_numbers", - "description": "Add two numbers", - "parameters": { - "type": "object", - "properties": { - "num1": {"description": "First number", "type": "int"}, - "num2": {"description": "Second number", "type": "int"}, - }, - "required": ["num1", "num2"], - }, - } - - module1_funcs = registry.get_all_by_module("module1") - assert len(module1_funcs) == 1 From 88270482d0e0d9c89c50f9549da2575bd3a0e499 Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 12 Jan 2024 16:35:18 +0800 Subject: [PATCH 299/637] fix libs path --- .pylintrc | 633 -------------------------------------- metagpt/const.py | 1 + metagpt/tools/__init__.py | 6 +- 3 files changed, 4 insertions(+), 636 deletions(-) delete mode 100644 .pylintrc diff --git a/.pylintrc b/.pylintrc deleted file mode 100644 index 21f5fb173..000000000 --- a/.pylintrc +++ /dev/null @@ -1,633 +0,0 @@ -[MAIN] - -# Analyse import fallback blocks. This can be used to support both Python 2 and -# 3 compatible code, which means that the block might have code that exists -# only in one or another interpreter, leading to false positives when analysed. -analyse-fallback-blocks=no - -# Clear in-memory caches upon conclusion of linting. Useful if running pylint -# in a server-like mode. -clear-cache-post-run=no - -# Load and enable all available extensions. Use --list-extensions to see a list -# all available extensions. -#enable-all-extensions= - -# In error mode, messages with a category besides ERROR or FATAL are -# suppressed, and no reports are done by default. Error mode is compatible with -# disabling specific errors. -#errors-only= - -# Always return a 0 (non-error) status code, even if lint errors are found. -# This is primarily useful in continuous integration scripts. -#exit-zero= - -# A comma-separated list of package or module names from where C extensions may -# be loaded. Extensions are loading into the active Python interpreter and may -# run arbitrary code. -extension-pkg-allow-list= - -# A comma-separated list of package or module names from where C extensions may -# be loaded. Extensions are loading into the active Python interpreter and may -# run arbitrary code. (This is an alternative name to extension-pkg-allow-list -# for backward compatibility.) -extension-pkg-whitelist= - -# Return non-zero exit code if any of these messages/categories are detected, -# even if score is above --fail-under value. Syntax same as enable. Messages -# specified are enabled, while categories only check already-enabled messages. -fail-on= - -# Specify a score threshold under which the program will exit with error. -fail-under=10 - -# Interpret the stdin as a python script, whose filename needs to be passed as -# the module_or_package argument. -#from-stdin= - -# Files or directories to be skipped. They should be base names, not paths. -ignore=CVS, offline - -# Add files or directories matching the regular expressions patterns to the -# ignore-list. The regex matches against paths and can be in Posix or Windows -# format. Because '\\' represents the directory delimiter on Windows systems, -# it can't be used as an escape character. -ignore-paths= - -# Files or directories matching the regular expression patterns are skipped. -# The regex matches against base names, not paths. The default value ignores -# Emacs file locks -ignore-patterns=^\.# - -# List of module names for which member attributes should not be checked -# (useful for modules/projects where namespaces are manipulated during runtime -# and thus existing member attributes cannot be deduced by static analysis). It -# supports qualified module names, as well as Unix pattern matching. -ignored-modules= - -# Python code to execute, usually for sys.path manipulation such as -# pygtk.require(). -#init-hook= - -# Use multiple processes to speed up Pylint. Specifying 0 will auto-detect the -# number of processors available to use, and will cap the count on Windows to -# avoid hangs. -jobs=1 - -# Control the amount of potential inferred values when inferring a single -# object. This can help the performance when dealing with large functions or -# complex, nested conditions. -limit-inference-results=100 - -# List of plugins (as comma separated values of python module names) to load, -# usually to register additional checkers. -load-plugins= - -# Pickle collected data for later comparisons. -persistent=yes - -# Minimum Python version to use for version dependent checks. Will default to -# the version used to run pylint. -py-version=3.9 - -# Discover python modules and packages in the file system subtree. -recursive=no - -# Add paths to the list of the source roots. Supports globbing patterns. The -# source root is an absolute path or a path relative to the current working -# directory used to determine a package namespace for modules located under the -# source root. -source-roots= - -# When enabled, pylint would attempt to guess common misconfiguration and emit -# user-friendly hints instead of false-positive error messages. -suggestion-mode=yes - -# Allow loading of arbitrary C extensions. Extensions are imported into the -# active Python interpreter and may run arbitrary code. -unsafe-load-any-extension=no - -# In verbose mode, extra non-checker-related info will be displayed. -#verbose= - - -[BASIC] - -# Naming style matching correct argument names. -argument-naming-style=snake_case - -# Regular expression matching correct argument names. Overrides argument- -# naming-style. If left empty, argument names will be checked with the set -# naming style. -#argument-rgx= - -# Naming style matching correct attribute names. -attr-naming-style=snake_case - -# Regular expression matching correct attribute names. Overrides attr-naming- -# style. If left empty, attribute names will be checked with the set naming -# style. -#attr-rgx= - -# Bad variable names which should always be refused, separated by a comma. -bad-names=foo, - bar, - baz, - toto, - tutu, - tata - -# Bad variable names regexes, separated by a comma. If names match any regex, -# they will always be refused -bad-names-rgxs= - -# Naming style matching correct class attribute names. -class-attribute-naming-style=any - -# Regular expression matching correct class attribute names. Overrides class- -# attribute-naming-style. If left empty, class attribute names will be checked -# with the set naming style. -#class-attribute-rgx= - -# Naming style matching correct class constant names. -class-const-naming-style=UPPER_CASE - -# Regular expression matching correct class constant names. Overrides class- -# const-naming-style. If left empty, class constant names will be checked with -# the set naming style. -#class-const-rgx= - -# Naming style matching correct class names. -class-naming-style=PascalCase - -# Regular expression matching correct class names. Overrides class-naming- -# style. If left empty, class names will be checked with the set naming style. -#class-rgx= - -# Naming style matching correct constant names. -const-naming-style=UPPER_CASE - -# Regular expression matching correct constant names. Overrides const-naming- -# style. If left empty, constant names will be checked with the set naming -# style. -#const-rgx= - -# Minimum line length for functions/classes that require docstrings, shorter -# ones are exempt. -docstring-min-length=-1 - -# Naming style matching correct function names. -function-naming-style=snake_case - -# Regular expression matching correct function names. Overrides function- -# naming-style. If left empty, function names will be checked with the set -# naming style. -#function-rgx= - -# Good variable names which should always be accepted, separated by a comma. -good-names=i, - j, - k, - ex, - Run, - _ - -# Good variable names regexes, separated by a comma. If names match any regex, -# they will always be accepted -good-names-rgxs= - -# Include a hint for the correct naming format with invalid-name. -include-naming-hint=no - -# Naming style matching correct inline iteration names. -inlinevar-naming-style=any - -# Regular expression matching correct inline iteration names. Overrides -# inlinevar-naming-style. If left empty, inline iteration names will be checked -# with the set naming style. -#inlinevar-rgx= - -# Naming style matching correct method names. -method-naming-style=snake_case - -# Regular expression matching correct method names. Overrides method-naming- -# style. If left empty, method names will be checked with the set naming style. -#method-rgx= - -# Naming style matching correct module names. -module-naming-style=snake_case - -# Regular expression matching correct module names. Overrides module-naming- -# style. If left empty, module names will be checked with the set naming style. -#module-rgx= - -# Colon-delimited sets of names that determine each other's naming style when -# the name regexes allow several styles. -name-group= - -# Regular expression which should only match function or class names that do -# not require a docstring. -no-docstring-rgx=^_ - -# List of decorators that produce properties, such as abc.abstractproperty. Add -# to this list to register other decorators that produce valid properties. -# These decorators are taken in consideration only for invalid-name. -property-classes=abc.abstractproperty - -# Regular expression matching correct type alias names. If left empty, type -# alias names will be checked with the set naming style. -#typealias-rgx= - -# Regular expression matching correct type variable names. If left empty, type -# variable names will be checked with the set naming style. -#typevar-rgx= - -# Naming style matching correct variable names. -variable-naming-style=snake_case - -# Regular expression matching correct variable names. Overrides variable- -# naming-style. If left empty, variable names will be checked with the set -# naming style. -#variable-rgx= - - -[CLASSES] - -# Warn about protected attribute access inside special methods -check-protected-access-in-special-methods=no - -# List of method names used to declare (i.e. assign) instance attributes. -defining-attr-methods=__init__, - __new__, - setUp, - asyncSetUp, - __post_init__ - -# List of member names, which should be excluded from the protected access -# warning. -exclude-protected=_asdict,_fields,_replace,_source,_make,os._exit - -# List of valid names for the first argument in a class method. -valid-classmethod-first-arg=cls - -# List of valid names for the first argument in a metaclass class method. -valid-metaclass-classmethod-first-arg=mcs - - -[DESIGN] - -# List of regular expressions of class ancestor names to ignore when counting -# public methods (see R0903) -exclude-too-few-public-methods= - -# List of qualified class names to ignore when counting class parents (see -# R0901) -ignored-parents= - -# Maximum number of arguments for function / method. -max-args=5 - -# Maximum number of attributes for a class (see R0902). -max-attributes=7 - -# Maximum number of boolean expressions in an if statement (see R0916). -max-bool-expr=5 - -# Maximum number of branch for function / method body. -max-branches=12 - -# Maximum number of locals for function / method body. -max-locals=15 - -# Maximum number of parents for a class (see R0901). -max-parents=7 - -# Maximum number of public methods for a class (see R0904). -max-public-methods=20 - -# Maximum number of return / yield for function / method body. -max-returns=6 - -# Maximum number of statements in function / method body. -max-statements=50 - -# Minimum number of public methods for a class (see R0903). -min-public-methods=2 - - -[EXCEPTIONS] - -# Exceptions that will emit a warning when caught. -overgeneral-exceptions=builtins.BaseException,builtins.Exception - - -[FORMAT] - -# Expected format of line ending, e.g. empty (any line ending), LF or CRLF. -expected-line-ending-format= - -# Regexp for a line that is allowed to be longer than the limit. -ignore-long-lines=^\s*(# )??$ - -# Number of spaces of indent required inside a hanging or continued line. -indent-after-paren=4 - -# String used as indentation unit. This is usually " " (4 spaces) or "\t" (1 -# tab). -indent-string=' ' - -# Maximum number of characters on a single line. -max-line-length=120 - -# Maximum number of lines in a module. -max-module-lines=1000 - -# Allow the body of a class to be on the same line as the declaration if body -# contains single statement. -single-line-class-stmt=no - -# Allow the body of an if to be on the same line as the test if there is no -# else. -single-line-if-stmt=no - - -[IMPORTS] - -# List of modules that can be imported at any level, not just the top level -# one. -allow-any-import-level= - -# Allow explicit reexports by alias from a package __init__. -allow-reexport-from-package=no - -# Allow wildcard imports from modules that define __all__. -allow-wildcard-with-all=no - -# Deprecated modules which should not be used, separated by a comma. -deprecated-modules= - -# Output a graph (.gv or any supported image format) of external dependencies -# to the given file (report RP0402 must not be disabled). -ext-import-graph= - -# Output a graph (.gv or any supported image format) of all (i.e. internal and -# external) dependencies to the given file (report RP0402 must not be -# disabled). -import-graph= - -# Output a graph (.gv or any supported image format) of internal dependencies -# to the given file (report RP0402 must not be disabled). -int-import-graph= - -# Force import order to recognize a module as part of the standard -# compatibility libraries. -known-standard-library= - -# Force import order to recognize a module as part of a third party library. -known-third-party=enchant - -# Couples of modules and preferred modules, separated by a comma. -preferred-modules= - - -[LOGGING] - -# The type of string formatting that logging methods do. `old` means using % -# formatting, `new` is for `{}` formatting. -logging-format-style=old - -# Logging modules to check that the string format arguments are in logging -# function parameter format. -logging-modules=logging - - -[MESSAGES CONTROL] - -# Only show warnings with the listed confidence levels. Leave empty to show -# all. Valid levels: HIGH, CONTROL_FLOW, INFERENCE, INFERENCE_FAILURE, -# UNDEFINED. -confidence=HIGH, - CONTROL_FLOW, - INFERENCE, - INFERENCE_FAILURE, - UNDEFINED - -# Disable the message, report, category or checker with the given id(s). You -# can either give multiple identifiers separated by comma (,) or put this -# option multiple times (only on the command line, not in the configuration -# file where it should appear only once). You can also use "--disable=all" to -# disable everything first and then re-enable specific checks. For example, if -# you want to run only the similarities checker, you can use "--disable=all -# --enable=similarities". If you want to run only the classes checker, but have -# no Warning level messages displayed, use "--disable=all --enable=classes -# --disable=W". -disable=raw-checker-failed, - bad-inline-option, - locally-disabled, - file-ignored, - suppressed-message, - useless-suppression, - deprecated-pragma, - use-symbolic-message-instead, - import-error, - ; C0114, C0115, C0116 - -# Enable the message, report, category or checker with the given id(s). You can -# either give multiple identifier separated by comma (,) or put this option -# multiple time (only on the command line, not in the configuration file where -# it should appear only once). See also the "--disable" option for examples. -enable=c-extension-no-member - - -[METHOD_ARGS] - -# List of qualified names (i.e., library.method) which require a timeout -# parameter e.g. 'requests.api.get,requests.api.post' -timeout-methods=requests.api.delete,requests.api.get,requests.api.head,requests.api.options,requests.api.patch,requests.api.post,requests.api.put,requests.api.request - - -[MISCELLANEOUS] - -# List of note tags to take in consideration, separated by a comma. -notes=FIXME, - XXX, - TODO - -# Regular expression of note tags to take in consideration. -notes-rgx= - - -[REFACTORING] - -# Maximum number of nested blocks for function / method body -max-nested-blocks=5 - -# Complete name of functions that never returns. When checking for -# inconsistent-return-statements if a never returning function is called then -# it will be considered as an explicit return statement and no message will be -# printed. -never-returning-functions=sys.exit,argparse.parse_error - - -[REPORTS] - -# Python expression which should return a score less than or equal to 10. You -# have access to the variables 'fatal', 'error', 'warning', 'refactor', -# 'convention', and 'info' which contain the number of messages in each -# category, as well as 'statement' which is the total number of statements -# analyzed. This score is used by the global evaluation report (RP0004). -evaluation=max(0, 0 if fatal else 10.0 - ((float(5 * error + warning + refactor + convention) / statement) * 10)) - -# Template used to display messages. This is a python new-style format string -# used to format the message information. See doc for all details. -msg-template= - -# Set the output format. Available formats are text, parseable, colorized, json -# and msvs (visual studio). You can also give a reporter class, e.g. -# mypackage.mymodule.MyReporterClass. -#output-format= - -# Tells whether to display a full report or only the messages. -reports=no - -# Activate the evaluation score. -score=yes - - -[SIMILARITIES] - -# Comments are removed from the similarity computation -ignore-comments=yes - -# Docstrings are removed from the similarity computation -ignore-docstrings=yes - -# Imports are removed from the similarity computation -ignore-imports=yes - -# Signatures are removed from the similarity computation -ignore-signatures=yes - -# Minimum lines number of a similarity. -min-similarity-lines=4 - - -[SPELLING] - -# Limits count of emitted suggestions for spelling mistakes. -max-spelling-suggestions=4 - -# Spelling dictionary name. No available dictionaries : You need to install -# both the python package and the system dependency for enchant to work.. -spelling-dict= - -# List of comma separated words that should be considered directives if they -# appear at the beginning of a comment and should not be checked. -spelling-ignore-comment-directives=fmt: on,fmt: off,noqa:,noqa,nosec,isort:skip,mypy: - -# List of comma separated words that should not be checked. -spelling-ignore-words= - -# A path to a file that contains the private dictionary; one word per line. -spelling-private-dict-file= - -# Tells whether to store unknown words to the private dictionary (see the -# --spelling-private-dict-file option) instead of raising a message. -spelling-store-unknown-words=no - - -[STRING] - -# This flag controls whether inconsistent-quotes generates a warning when the -# character used as a quote delimiter is used inconsistently within a module. -check-quote-consistency=no - -# This flag controls whether the implicit-str-concat should generate a warning -# on implicit string concatenation in sequences defined over several lines. -check-str-concat-over-line-jumps=no - - -[TYPECHECK] - -# List of decorators that produce context managers, such as -# contextlib.contextmanager. Add to this list to register other decorators that -# produce valid context managers. -contextmanager-decorators=contextlib.contextmanager - -# List of members which are set dynamically and missed by pylint inference -# system, and so shouldn't trigger E1101 when accessed. Python regular -# expressions are accepted. -generated-members= - -# Tells whether to warn about missing members when the owner of the attribute -# is inferred to be None. -ignore-none=yes - -# This flag controls whether pylint should warn about no-member and similar -# checks whenever an opaque object is returned when inferring. The inference -# can return multiple potential results while evaluating a Python object, but -# some branches might not be evaluated, which results in partial inference. In -# that case, it might be useful to still emit no-member and other checks for -# the rest of the inferred objects. -ignore-on-opaque-inference=yes - -# List of symbolic message names to ignore for Mixin members. -ignored-checks-for-mixins=no-member, - not-async-context-manager, - not-context-manager, - attribute-defined-outside-init - -# List of class names for which member attributes should not be checked (useful -# for classes with dynamically set attributes). This supports the use of -# qualified names. -ignored-classes=optparse.Values,thread._local,_thread._local,argparse.Namespace - -# Show a hint with possible names when a member name was not found. The aspect -# of finding the hint is based on edit distance. -missing-member-hint=yes - -# The minimum edit distance a name should have in order to be considered a -# similar match for a missing member name. -missing-member-hint-distance=1 - -# The total number of similar names that should be taken in consideration when -# showing a hint for a missing member. -missing-member-max-choices=1 - -# Regex pattern to define which classes are considered mixins. -mixin-class-rgx=.*[Mm]ixin - -# List of decorators that change the signature of a decorated function. -signature-mutators= - - -[VARIABLES] - -# List of additional names supposed to be defined in builtins. Remember that -# you should avoid defining new builtins when possible. -additional-builtins= - -# Tells whether unused global variables should be treated as a violation. -allow-global-unused-variables=yes - -# List of names allowed to shadow builtins -allowed-redefined-builtins= - -# List of strings which can identify a callback function by name. A callback -# name must start or end with one of those strings. -callbacks=cb_, - _cb - -# A regular expression matching the name of dummy variables (i.e. expected to -# not be used). -dummy-variables-rgx=_+$|(_[a-zA-Z0-9_]*[a-zA-Z0-9]+?$)|dummy|^ignored_|^unused_ - -# Argument names that match this expression will be ignored. -ignored-argument-names=_.*|^ignored_|^unused_ - -# Tells whether we should check for unused import in __init__ files. -init-import=no - -# List of qualified module names which can have objects that can redefine -# builtins. -redefining-builtins-modules=six.moves,past.builtins,future.builtins,builtins,io diff --git a/metagpt/const.py b/metagpt/const.py index b1666e092..a57464a19 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -71,6 +71,7 @@ SOURCE_ROOT = METAGPT_ROOT / "metagpt" PROMPT_PATH = SOURCE_ROOT / "prompts" SKILL_DIRECTORY = SOURCE_ROOT / "skills" TOOL_SCHEMA_PATH = METAGPT_ROOT / "metagpt/tools/functions/schemas" +TOOL_LIBS_PATH = METAGPT_ROOT / "metagpt/tools/functions/libs" # REAL CONSTS diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index 543a2b8bb..4b3528795 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -11,7 +11,7 @@ from enum import Enum from pydantic import BaseModel -from metagpt.const import TOOL_SCHEMA_PATH +from metagpt.const import TOOL_LIBS_PATH from metagpt.prompts.tool_type import ( DATA_PREPROCESS_PROMPT, FEATURE_ENGINEERING_PROMPT, @@ -49,13 +49,13 @@ class ToolType(BaseModel): TOOL_TYPE_MAPPINGS = { "data_preprocess": ToolType( name="data_preprocess", - module=str(TOOL_SCHEMA_PATH / "data_preprocess"), + module=str(TOOL_LIBS_PATH / "data_preprocess"), desc="Only for changing value inplace.", usage_prompt=DATA_PREPROCESS_PROMPT, ), "feature_engineering": ToolType( name="feature_engineering", - module=str(TOOL_SCHEMA_PATH / "feature_engineering"), + module=str(TOOL_LIBS_PATH / "feature_engineering"), desc="Only for creating new columns for input data.", usage_prompt=FEATURE_ENGINEERING_PROMPT, ), From 13010f6c909aa7fb571e94da7676d9688df538d0 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 12 Jan 2024 17:19:49 +0800 Subject: [PATCH 300/637] add async function for sd tool --- metagpt/tools/functions/schemas/stable_diffusion.yml | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/metagpt/tools/functions/schemas/stable_diffusion.yml b/metagpt/tools/functions/schemas/stable_diffusion.yml index 119449caa..a93742a1d 100644 --- a/metagpt/tools/functions/schemas/stable_diffusion.yml +++ b/metagpt/tools/functions/schemas/stable_diffusion.yml @@ -9,7 +9,6 @@ SDEngine: sd_url: type: str description: "URL of the stable diffusion service." - simple_run_t2i: description: "Run the stable diffusion API for multiple prompts, calling the stable diffusion API to generate images." parameters: @@ -22,6 +21,16 @@ SDEngine: description: "Save generated images automatically." required: - prompts + run_t2i: + type: async function + description: "Run the stable diffusion API for multiple prompts, calling the stable diffusion API to generate images." + parameters: + properties: + payloads: + type: list + description: "List of payload, each payload is a dictionary of input parameters for the stable diffusion API." + required: + - payloads construct_payload: description: "Modify and set the API parameters for image generation." parameters: From d9ad3a6195034ca5c2bb610200ed2130e60de7b2 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 12 Jan 2024 17:30:22 +0800 Subject: [PATCH 301/637] update --- .gitignore | 10 ---------- tests/conftest.py | 2 +- 2 files changed, 1 insertion(+), 11 deletions(-) diff --git a/.gitignore b/.gitignore index 0a78c3d58..87c7b3120 100644 --- a/.gitignore +++ b/.gitignore @@ -177,13 +177,3 @@ htmlcov.* *.pkl *-structure.csv *-structure.json - -/Titanic/2023_12_07_11_44_319a116fff/LLM_inout_pair/*.json -/ICR/2023_12_06_14_14_26e593d09f/LLM_inout_pair/*.json -/ICR/5cd9acb669c443fabe763e8f1ade5e86/workspace/*.txt -/ICR/5cd9acb669c443fabe763e8f1ade5e86/workspace/*.csv -/Titanic/9530b3c5550a4366ae92e5af6a74e6c3/workspace/*.csv -/Titanic/9530b3c5550a4366ae92e5af6a74e6c3/workspace/*.txt -/metagpt/roles/catboost_info/*.tsv -/metagpt/roles/catboost_info/*.json -/Titanic/9530b3c5550a4366ae92e5af6a74e6c3/workspace/*.md diff --git a/tests/conftest.py b/tests/conftest.py index f551c9205..7dec506bb 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -140,7 +140,7 @@ def loguru_caplog(caplog): # init & dispose git repo -@pytest.fixture(scope="function", autouse=False) +@pytest.fixture(scope="function", autouse=True) def setup_and_teardown_git_repo(request): CONFIG.git_repo = GitRepository(local_path=DEFAULT_WORKSPACE_ROOT / f"unittest/{uuid.uuid4().hex}") CONFIG.git_reinit = True From a363691c38345c7e9e0bdc4d7a83e077f37a2882 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 12 Jan 2024 17:33:15 +0800 Subject: [PATCH 302/637] 1. add guideline in context of write_code_review.py 2. update prompt --- metagpt/actions/design_api_an.py | 7 ++-- metagpt/actions/project_management_an.py | 8 ++--- metagpt/actions/write_code.py | 9 ++++- metagpt/actions/write_code_guideline_an.py | 22 +++++++----- metagpt/actions/write_code_review.py | 42 +++++++++++++++++----- metagpt/actions/write_prd_an.py | 2 +- metagpt/roles/engineer.py | 14 +++++--- 7 files changed, 73 insertions(+), 31 deletions(-) diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index 3e8265e95..02f20a133 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -49,9 +49,8 @@ FILE_LIST = ActionNode( REFINED_FILE_LIST = ActionNode( key="Refined File List", expected_type=List[str], - instruction="Update and expand the original file list, including only relative paths. " - "Ensure that the refined file list reflects the evolving structure of the project due to incremental development." - "Only output filename! Do not include comments in the list.", + instruction="Update and expand the original file list including only relative paths, up to 2 files can be added." + "Ensure that the refined file list reflects the evolving structure of the project due to incremental development.", example=["main.py", "game.py", "new_feature.py"], ) @@ -148,7 +147,7 @@ INC_NODES = [INCREMENTAL_IMPLEMENTATION_APPROACH, INCREMENTAL_DATA_STRUCTURES_AN REFINE_NODES = [ REFINED_IMPLEMENTATION_APPROACH, - FILE_LIST, + REFINED_FILE_LIST, REFINED_DATA_STRUCTURES_AND_INTERFACES, REFINED_PROGRAM_CALL_FLOW, ANYTHING_UNCLEAR, diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index a970c05a3..9b54698a1 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -75,16 +75,16 @@ INCREMENTAL_TASK_LIST = ActionNode( instruction="Break down the incremental development tasks into a prioritized list of filenames." "Organize the tasks based on dependency order, ensuring a systematic and efficient implementation." "Only output filename! Do not include comments in the list ", - example=["new_feature.py", "utils.py", "main.py"], + example=["new_feature.py", "main.py"], ) REFINED_TASK_LIST = ActionNode( key="Refined Task list", expected_type=List[str], - instruction="Review and refine the combined task list after the merger of Legacy Content and Incremental Content. " + instruction="Review and refine the combined task list after the merger of Legacy Content and Incremental Content, and consistent with Refined File List." "Ensure that tasks are organized in a logical and prioritized order, considering dependencies for a streamlined and" - " efficient development process. Only output filename! Do not include comments in the list", - example=["game.py", "utils.py", "new_feature.py", "main.py"], + " efficient development process. ", + example=["new_feature.py", "utils", "game.py", "main.py"], ) FULL_API_SPEC = ActionNode( diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 7555ce101..acc25627e 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -27,6 +27,7 @@ from metagpt.const import ( BUGFIX_FILENAME, CODE_SUMMARIES_FILE_REPO, DOCS_FILE_REPO, + PRDS_FILE_REPO, REQUIREMENT_FILENAME, TASK_FILE_REPO, TEST_OUTPUTS_FILE_REPO, @@ -115,6 +116,11 @@ class WriteCode(Action): docs_file_repo = CONFIG.git_repo.new_file_repository(relative_path=DOCS_FILE_REPO) requirement_doc = await docs_file_repo.get(filename=REQUIREMENT_FILENAME) + + prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) + prd = await prd_file_repo.get_all() + prd_json = json.loads("\n".join([doc.content for doc in prd])) + product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) guideline = kwargs.get("guideline", "") if bug_feedback: code_context = coding_context.code_doc.content @@ -125,7 +131,8 @@ class WriteCode(Action): if guideline: prompt = REFINED_CODE_TEMPLATE.format( - requirement=requirement_doc.content if requirement_doc else "", + user_requirement=requirement_doc.content if requirement_doc else "", + product_requirement_pool=str(product_requirement_pool), guideline=guideline, design=coding_context.design_doc.content if coding_context.design_doc else "", tasks=coding_context.task_doc.content if coding_context.task_doc else "", diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index c08340cb7..9521e51e6 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -15,7 +15,8 @@ GUIDELINES_AND_INCREMENTAL_CHANGE = ActionNode( expected_type=str, instruction="Developing comprehensive and step-by-step incremental development guideline, and Write Incremental " "Change by making a code draft that how to implement incremental development including detailed steps based on the " - "context. Note: Track incremental changes using mark of '+' or '-' for add/modify/delete code", + "context. Note: Track incremental changes using mark of '+' or '-' for add/modify/delete code, and conforms to the " + "output format of git diff", example=""" 1. Guideline for calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Additionally, implement robust error handling for the division operation to mitigate potential issues related to division by zero. ```python @@ -105,11 +106,11 @@ def add_numbers(): ) CODE_GUIDELINE_CONTEXT = """ -## New Requirements -{requirement} +## User New Requirements +{user_requirement} -## PRD -{prd} +## Product Requirement Pool +{product_requirement_pool} ## Design {design} @@ -388,13 +389,16 @@ class Interface: REFINED_CODE_TEMPLATE = """ NOTICE -Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and Incremental Change, ensuring the integration of new features. +Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and Guidelines and Incremental Change, ensuring the integration of new features. # Context -## New Requirement -{requirement} +## User New Requirements +{user_requirement} -## Incremental Change +## Product Requirement Pool +{product_requirement_pool} + +## Guidelines and Incremental Change {guideline} ## Design diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index a8c913573..e3c5fd2ac 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -7,6 +7,7 @@ @Modified By: mashenquan, 2023/11/27. Following the think-act principle, solidify the task parameters when creating the WriteCode object, rather than passing them in when calling the run function. """ +import json from pydantic import Field from tenacity import retry, stop_after_attempt, wait_random_exponential @@ -14,6 +15,7 @@ from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions import WriteCode from metagpt.actions.action import Action from metagpt.config import CONFIG +from metagpt.const import DOCS_FILE_REPO, PRDS_FILE_REPO, REQUIREMENT_FILENAME from metagpt.logs import logger from metagpt.schema import CodingContext from metagpt.utils.common import CodeParser @@ -138,17 +140,41 @@ class WriteCodeReview(Action): async def run(self, *args, **kwargs) -> CodingContext: iterative_code = self.context.code_doc.content k = CONFIG.code_review_k_times or 1 + guideline = kwargs.get("guideline") + mode = "guide" if guideline else "normal" for i in range(k): format_example = FORMAT_EXAMPLE.format(filename=self.context.code_doc.filename) task_content = self.context.task_doc.content if self.context.task_doc else "" - code_context = await WriteCode.get_codes(self.context.task_doc, exclude=self.context.filename) - context = "\n".join( - [ - "## System Design\n" + str(self.context.design_doc) + "\n", - "## Tasks\n" + task_content + "\n", - "## Code Files\n" + code_context + "\n", - ] - ) + code_context = await WriteCode.get_codes(self.context.task_doc, exclude=self.context.filename, mode=mode) + + if not guideline: + context = "\n".join( + [ + "## System Design\n" + str(self.context.design_doc) + "\n", + "## Tasks\n" + task_content + "\n", + "## Code Files\n" + code_context + "\n", + ] + ) + else: + docs_file_repo = CONFIG.git_repo.new_file_repository(relative_path=DOCS_FILE_REPO) + requirement_doc = await docs_file_repo.get(filename=REQUIREMENT_FILENAME) + user_requirement = requirement_doc.content if requirement_doc else "" + prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) + prd = await prd_file_repo.get_all() + prd_json = json.loads("\n".join([doc.content for doc in prd])) + product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) + + context = "\n".join( + [ + "## User New Requirements\n" + str(user_requirement) + "\n", + "## Product Requirement Pool\n" + str(product_requirement_pool) + "\n", + "## Guidelines and Incremental Change\n" + guideline + "\n", + "## System Design\n" + str(self.context.design_doc) + "\n", + "## Tasks\n" + task_content + "\n", + "## Code Files\n" + code_context + "\n", + ] + ) + context_prompt = PROMPT_TEMPLATE.format( context=context, code=iterative_code, diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index 91c1b2837..2eaed7114 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -160,7 +160,7 @@ REQUIREMENT_POOL = ActionNode( REFINED_REQUIREMENT_POOL = ActionNode( key="Refined Requirement Pool", expected_type=List[List[str]], - instruction="List no less than 5 requirements with their priority (P0, P1, P2). " + instruction="List no less than 5 requirements with their priority (P0, P1, P2) from high to low. " "Cover both legacy content and incremental content. Retain any content unrelated to incremental development", example=[["P0", "The main code ..."], ["P0", "The game algorithm ..."]], ) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index e0b22ea5b..e67963de5 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -117,7 +117,7 @@ class Engineer(Role): if review: action = WriteCodeReview(context=coding_context, llm=self.llm) self._init_action_system_message(action) - coding_context = await action.run() + coding_context = await action.run(guideline=guideline) # Get dependencies if guideline: @@ -346,12 +346,14 @@ class Engineer(Role): async def _write_code_guideline(self): logger.info("Writing code guideline..") - requirement = str(self.rc.memory.get_by_role("Human")[0]) + user_requirement = str(self.rc.memory.get_by_role("Human")[0]) prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) design_file_repo = CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO) task_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) prd = await prd_file_repo.get_all() - prd = "\n".join([doc.content for doc in prd]) + prd_json = json.loads("\n".join([doc.content for doc in prd])) + product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) + design = await design_file_repo.get_all() design = "\n".join([doc.content for doc in design]) tasks = await task_file_repo.get_all() @@ -359,7 +361,11 @@ class Engineer(Role): old_codes = await self.get_old_codes() context = CODE_GUIDELINE_CONTEXT.format( - requirement=requirement, prd=prd, tasks=tasks, design=design, code=old_codes + user_requirement=user_requirement, + product_requirement_pool=str(product_requirement_pool), + tasks=tasks, + design=design, + code=old_codes, ) node = await WriteCodeGuideline().run(context=context) guideline = node.instruct_content.model_dump_json() From 3fded9b6e0f853661cf96ce4bced3931edb7da9e Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 12 Jan 2024 18:23:15 +0800 Subject: [PATCH 303/637] fix timeout --- metagpt/actions/execute_code.py | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 8355d3aca..c75711e75 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -4,6 +4,7 @@ @Author : orange-crow @File : code_executor.py """ +import asyncio import re import traceback from abc import ABC, abstractmethod @@ -81,6 +82,9 @@ class ExecutePyCode(ExecuteCode, Action): async def reset(self): """reset NotebookClient""" await self.terminate() + + # sleep 1s to wait for the kernel to be cleaned up completely + await asyncio.sleep(1) await self.build() self.nb_client = NotebookClient(self.nb, timeout=self.timeout) @@ -181,7 +185,11 @@ class ExecutePyCode(ExecuteCode, Action): await self.nb_client.async_execute_cell(cell, cell_index) return True, "" except CellTimeoutError: - return False, "TimeoutError" + assert self.nb_client.km is not None + await self.nb_client.km.interrupt_kernel() + await asyncio.sleep(1) + error_msg = "Cell execution timed out: Execution exceeded the time limit and was stopped; consider optimizing your code for better performance." + return False, error_msg except DeadKernelError: await self.reset() return False, "DeadKernelError" From 40f5d5e40efda6cafe1f809c43fbf28fab0d8479 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 12 Jan 2024 18:30:48 +0800 Subject: [PATCH 304/637] add vision tool for code_interpreter --- metagpt/prompts/tool_type.py | 6 ++ metagpt/tools/__init__.py | 7 ++ metagpt/tools/functions/libs/vision.py | 81 ++++++++++++++++++++++ metagpt/tools/functions/schemas/vision.yml | 20 ++++++ 4 files changed, 114 insertions(+) create mode 100644 metagpt/tools/functions/libs/vision.py create mode 100644 metagpt/tools/functions/schemas/vision.yml diff --git a/metagpt/prompts/tool_type.py b/metagpt/prompts/tool_type.py index ec848bbe4..43ead78a6 100644 --- a/metagpt/prompts/tool_type.py +++ b/metagpt/prompts/tool_type.py @@ -37,3 +37,9 @@ The current task is about evaluating a model, please note the following: - Ensure that the evaluated data is same processed as the training data. If not, remember use object in 'Done Tasks' to transform the data. - Use trained model from previous task result directly, do not mock or reload model yourself. """ + +# Prompt for using tools of "vision" type +VISION_PROMPT = """ +The current task is about converting image into webpage code. please note the following: +- Single-Step Code Generation: Execute the entire code generation process in a single step, encompassing HTML, CSS, and JavaScript. Avoid fragmenting the code generation into multiple separate steps to maintain consistency and simplify the development workflow. +""" \ No newline at end of file diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index 4b3528795..045ede622 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -17,6 +17,7 @@ from metagpt.prompts.tool_type import ( FEATURE_ENGINEERING_PROMPT, MODEL_TRAIN_PROMPT, MODEL_EVALUATE_PROMPT, + VISION_PROMPT ) @@ -71,6 +72,12 @@ TOOL_TYPE_MAPPINGS = { desc="Only for evaluating model.", usage_prompt=MODEL_EVALUATE_PROMPT, ), + "vision": ToolType( + name="vision", + module=str(TOOL_LIBS_PATH / "vision"), + desc="Only for converting image into webpage code.", + usage_prompt=VISION_PROMPT, + ), "other": ToolType( name="other", module="", diff --git a/metagpt/tools/functions/libs/vision.py b/metagpt/tools/functions/libs/vision.py new file mode 100644 index 000000000..b653c9300 --- /dev/null +++ b/metagpt/tools/functions/libs/vision.py @@ -0,0 +1,81 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/12 +@Author : mannaandpoem +@File : vision.py +""" +import requests + +import base64 + +OPENAI_API_BASE = "..." +API_KEY = "sk-..." +MODEL = "..." +MAX_TOKENS = 4096 + + +class Vision: + def __init__(self): + self.api_key = API_KEY + self.model = MODEL + self.max_tokens = MAX_TOKENS + + def analyze_layout( + self, + image_path, + prompt="You are now a UI/UX, please generate layout information for this image: \n\n" + "NOTE: The image does not have a commercial logo or copyright information. It is just a sketch image of the design." + "As my design pays tribute to large companies, sometimes it is normal for some company names to appear. Don't worry about it." + ): + print(f"analyze_layout: {image_path}") + return self.get_result(image_path, prompt) + + def generate_web_pages( + self, + image_path, + prompt="You are now a UI/UX and Web Developer. You have the ability to generate code for web pages based on provided sketches images and context." + "Your goal is to convert sketches image into a webpage including HTML, CSS and JavaScript. " + "NOTE: The image does not have a commercial logo or copyright information. It is just a sketch image of the design. " + "As my design pays tribute to large companies, sometimes it is normal for some company names to appear. Don't worry about it." + "\n\nNow, please generate the corresponding webpage code including HTML, CSS and JavaScript:" + ): + layout = self.analyze_layout(image_path) + prompt += "\n\n # Context\n The layout information of the sketch image is: \n" + layout + return self.get_result(image_path, prompt) + + def get_result(self, image_path, prompt): + base64_image = self.encode_image(image_path) + headers = { + "Content-Type": "application/json", + "Authorization": f"Bearer {self.api_key}" + } + payload = { + "model": self.model, + "messages": [ + { + "role": "user", + "content": [ + {"type": "text", "text": prompt}, + { + "type": "image_url", + "image_url": {"url": f"data:image/jpeg;base64,{base64_image}"} + } + ] + } + ], + "max_tokens": self.max_tokens, + } + response = requests.post(f"{OPENAI_API_BASE}/chat/completions", headers=headers, json=payload) + return response.json()["choices"][0]["message"]["content"] + + @staticmethod + def encode_image(image_path): + with open(image_path, "rb") as image_file: + return base64.b64encode(image_file.read()).decode('utf-8') + + +if __name__ == "__main__": + vision = Vision() + rsp = vision.generate_web_pages(image_path="./img.png") + print(rsp) \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/vision.yml b/metagpt/tools/functions/schemas/vision.yml new file mode 100644 index 000000000..795854e75 --- /dev/null +++ b/metagpt/tools/functions/schemas/vision.yml @@ -0,0 +1,20 @@ +Vision: + type: class + description: "Class for generating web pages at once." + methods: + __init__: + description: "Initialize Vision class with default values." + + generate_web_pages: + description: "Generate web pages including all code(HTML, CSS and JavaScript) in one go based on the image." + parameters: + properties: + image_path: + type: str + description: "The path of the image file" + + required: + - image_path + returns: + type: str + description: "Generated web page content." \ No newline at end of file From a249e46259d3f8e055c896bc0b5615ca5d693871 Mon Sep 17 00:00:00 2001 From: better629 Date: Fri, 12 Jan 2024 18:56:42 +0800 Subject: [PATCH 305/637] update prompt of review/revise to meet gpt3.5 --- metagpt/actions/action_node.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/metagpt/actions/action_node.py b/metagpt/actions/action_node.py index b4d8c32df..b511f2662 100644 --- a/metagpt/actions/action_node.py +++ b/metagpt/actions/action_node.py @@ -61,7 +61,7 @@ Follow instructions of nodes, generate output and make sure it follows the forma REVIEW_TEMPLATE = """ ## context -Compare the keys of nodes_output and the corresponding requirements one by one. If a key that does not match the requirement is found, provide the comment content on how to modify it. No output is required for matching keys. +Compare the key's value of nodes_output and the corresponding requirements one by one. If a key's value that does not match the requirement is found, provide the comment content on how to modify it. No output is required for matching keys. ### nodes_output {nodes_output} @@ -86,7 +86,7 @@ Compare the keys of nodes_output and the corresponding requirements one by one. {constraint} ## action -generate output and make sure it follows the format example. +Follow format example's json format, generate output and make sure it follows the format example. """ REVISE_TEMPLATE = """ @@ -108,7 +108,7 @@ change the nodes_output key's value to meet its comment and no need to add extra {constraint} ## action -generate output and make sure it follows the format example. +Follow format example's json format, generate output and make sure it follows the format example. """ @@ -469,7 +469,7 @@ class ActionNode: return dict() prompt = template.format( - nodes_output=json.dumps(nodes_output, ensure_ascii=False, indent=4), tag=TAG, constraint=FORMAT_CONSTRAINT + nodes_output=json.dumps(nodes_output, ensure_ascii=False), tag=TAG, constraint=FORMAT_CONSTRAINT ) content = await self.llm.aask(prompt) @@ -563,7 +563,7 @@ class ActionNode: instruction = self.compile_instruction(schema="markdown", mode="auto", exclude=exclude_keys) prompt = template.format( - nodes_output=json.dumps(nodes_output, ensure_ascii=False, indent=4), + nodes_output=json.dumps(nodes_output, ensure_ascii=False), example=example, instruction=instruction, constraint=FORMAT_CONSTRAINT, From 684d10bb247dc06355b4f8fd51510b64449be9ca Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 12 Jan 2024 20:38:23 +0800 Subject: [PATCH 306/637] 1. Remove PRD when write code --- metagpt/actions/write_code.py | 12 +----------- metagpt/actions/write_code_guideline_an.py | 3 --- metagpt/actions/write_code_review.py | 2 +- metagpt/roles/engineer.py | 6 +++--- 4 files changed, 5 insertions(+), 18 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index acc25627e..25e65ddc7 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -27,7 +27,6 @@ from metagpt.const import ( BUGFIX_FILENAME, CODE_SUMMARIES_FILE_REPO, DOCS_FILE_REPO, - PRDS_FILE_REPO, REQUIREMENT_FILENAME, TASK_FILE_REPO, TEST_OUTPUTS_FILE_REPO, @@ -117,10 +116,6 @@ class WriteCode(Action): docs_file_repo = CONFIG.git_repo.new_file_repository(relative_path=DOCS_FILE_REPO) requirement_doc = await docs_file_repo.get(filename=REQUIREMENT_FILENAME) - prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) - prd = await prd_file_repo.get_all() - prd_json = json.loads("\n".join([doc.content for doc in prd])) - product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) guideline = kwargs.get("guideline", "") if bug_feedback: code_context = coding_context.code_doc.content @@ -132,7 +127,6 @@ class WriteCode(Action): if guideline: prompt = REFINED_CODE_TEMPLATE.format( user_requirement=requirement_doc.content if requirement_doc else "", - product_requirement_pool=str(product_requirement_pool), guideline=guideline, design=coding_context.design_doc.content if coding_context.design_doc else "", tasks=coding_context.task_doc.content if coding_context.task_doc else "", @@ -188,11 +182,7 @@ class WriteCode(Action): else: doc = await src_file_repo.get(filename=filename) # 使用先前生成的代码 if not doc: - # if filename in old_files: - # doc = await old_file_repo.get(filename=filename) # 使用原始代码 - # else: - # continue - continue # 跳过 + continue codes.append(f"----- {filename}\n```{doc.content}```") else: diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index 9521e51e6..7b27fbe4c 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -395,9 +395,6 @@ Role: You are a professional engineer; The main goal is to complete incremental ## User New Requirements {user_requirement} -## Product Requirement Pool -{product_requirement_pool} - ## Guidelines and Incremental Change {guideline} diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index e3c5fd2ac..7fca9748c 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -161,7 +161,7 @@ class WriteCodeReview(Action): user_requirement = requirement_doc.content if requirement_doc else "" prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) prd = await prd_file_repo.get_all() - prd_json = json.loads("\n".join([doc.content for doc in prd])) + prd_json = json.loads(prd[0].content) product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) context = "\n".join( diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index e67963de5..1d1fed2b8 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -350,10 +350,10 @@ class Engineer(Role): prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) design_file_repo = CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO) task_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) - prd = await prd_file_repo.get_all() - prd_json = json.loads("\n".join([doc.content for doc in prd])) - product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) + prd = await prd_file_repo.get_all() + prd_json = json.loads(prd[0].content) + product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) design = await design_file_repo.get_all() design = "\n".join([doc.content for doc in design]) tasks = await task_file_repo.get_all() From b7adb1dc7d1353435cd2939bd4ae5435844b6ea8 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 12 Jan 2024 22:12:16 +0800 Subject: [PATCH 307/637] Update prompt and code comment in ActionNode --- metagpt/actions/design_api_an.py | 18 ++--- metagpt/actions/project_management_an.py | 15 ++-- metagpt/actions/write_code.py | 6 +- metagpt/actions/write_code_guideline_an.py | 4 +- metagpt/actions/write_prd_an.py | 13 ++-- tests/metagpt/test_incremental_dev.py | 82 ++++++++++++++++++++++ 6 files changed, 109 insertions(+), 29 deletions(-) diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index 02f20a133..ee1941350 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -22,7 +22,7 @@ INCREMENTAL_IMPLEMENTATION_APPROACH = ActionNode( key="Incremental Implementation approach", expected_type=str, instruction="Analyze the challenging aspects of the requirements and select a suitable open-source framework. " - "Outline the incremental steps involved in the implementation process with a list of detailed strategies.", + "Outline the incremental steps involved in the implementation process with the detailed strategies.", example="we will ...", ) @@ -30,8 +30,8 @@ REFINED_IMPLEMENTATION_APPROACH = ActionNode( key="Refined Implementation Approach", expected_type=str, instruction="Update and extend the original implementation approach to reflect the evolving challenges and " - "requirements due to incremental development. Provide detailed strategies for incremental steps in the " - "implementation process. Retain any content unrelated to incremental development for coherence and clarity.", + "requirements due to incremental development. Outline the steps involved in the implementation process with the " + "detailed strategies.", example="We will refine ...", ) @@ -49,8 +49,8 @@ FILE_LIST = ActionNode( REFINED_FILE_LIST = ActionNode( key="Refined File List", expected_type=List[str], - instruction="Update and expand the original file list including only relative paths, up to 2 files can be added." - "Ensure that the refined file list reflects the evolving structure of the project due to incremental development.", + instruction="Update and expand the original file list including only relative paths. Up to 2 files can be added." + "Ensure that the refined file list reflects the evolving structure of the project.", example=["main.py", "game.py", "new_feature.py"], ) @@ -78,9 +78,8 @@ REFINED_DATA_STRUCTURES_AND_INTERFACES = ActionNode( expected_type=str, instruction="Update and extend the existing mermaid classDiagram code syntax to incorporate new classes, " "methods (including __init__), and functions with precise type annotations. Delineate additional " - "relationships between classes, ensuring clarity and adherence to PEP8 standards. Further enhance the " - "detail in data structures for a comprehensive API design that seamlessly integrates with the evolving structure." - "Retain any content unrelated to incremental development for coherence and clarity.", + "relationships between classes, ensuring clarity and adherence to PEP8 standards." + "Retain content that is not related to incremental development but important for consistency and clarity.", example=MMC1_REFINE, ) @@ -97,7 +96,8 @@ REFINED_PROGRAM_CALL_FLOW = ActionNode( expected_type=str, instruction="Extend the existing sequenceDiagram code syntax with detailed information, accurately covering the" "CRUD and initialization of each object. Ensure correct syntax usage and reflect the incremental changes introduced" - "in the classes and API defined above.Retain content unrelated to incremental development for coherence and clarity", + "in the classes and API defined above. " + "Retain content that is not related to incremental development but important for consistency and clarity.", example=MMC2_REFINE, ) diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index 9b54698a1..559c5ef8e 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -52,8 +52,7 @@ REFINED_LOGIC_ANALYSIS = ActionNode( expected_type=List[List[str]], instruction="Review and refine the logic analysis by merging the Legacy Content and Incremental Content. " "Provide a comprehensive list of files with classes/methods/functions to be implemented or modified incrementally. " - "Include thorough dependency analysis, consider potential impacts on existing code, and document necessary imports." - "Retain any content unrelated to incremental development for coherence and clarity.", + "Include thorough dependency analysis, consider potential impacts on existing code, and document necessary imports.", example=[ ["game.py", "Contains Game class and ... functions"], ["main.py", "Contains main function, from game import Game"], @@ -81,9 +80,9 @@ INCREMENTAL_TASK_LIST = ActionNode( REFINED_TASK_LIST = ActionNode( key="Refined Task list", expected_type=List[str], - instruction="Review and refine the combined task list after the merger of Legacy Content and Incremental Content, and consistent with Refined File List." - "Ensure that tasks are organized in a logical and prioritized order, considering dependencies for a streamlined and" - " efficient development process. ", + instruction="Review and refine the combined task list after the merger of Legacy Content and Incremental Content, " + "and consistent with Refined File List. Ensure that tasks are organized in a logical and prioritized order, " + "considering dependencies for a streamlined and efficient development process. ", example=["new_feature.py", "utils", "game.py", "main.py"], ) @@ -113,9 +112,9 @@ INCREMENTAL_SHARED_KNOWLEDGE = ActionNode( REFINED_SHARED_KNOWLEDGE = ActionNode( key="Refined Shared Knowledge", expected_type=str, - instruction="Update and expand shared knowledge to reflect any new elements introduced during incremental " - "development. This includes common utility functions, configuration variables, or any information vital for team " - "collaboration. Retain any content unrelated to incremental development for coherence and clarity.", + instruction="Update and expand shared knowledge to reflect any new elements introduced. This includes common " + "utility functions, configuration variables for team collaboration. Retain content that is not related to " + "incremental development but important for consistency and clarity.", example="`new_module.py` enhances shared utility functions for improved code reusability and collaboration.", ) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 25e65ddc7..f92b72f7f 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -174,13 +174,15 @@ class WriteCode(Action): for filename in union_files_list: if filename == exclude: if filename in old_files and filename != "main.py": - doc = await old_file_repo.get(filename=filename) # 使用原始代码 + # Use legacy code + doc = await old_file_repo.get(filename=filename) else: continue codes.insert(0, f"-----Now, {filename} to be rewritten\n```{doc.content}```\n=====") else: - doc = await src_file_repo.get(filename=filename) # 使用先前生成的代码 + # Use new code + doc = await src_file_repo.get(filename=filename) if not doc: continue codes.append(f"----- {filename}\n```{doc.content}```") diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index 7b27fbe4c..cc532ed0f 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -434,11 +434,11 @@ Role: You are a professional engineer; The main goal is to complete incremental 2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets. 3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import. 4. Follow design: YOU MUST FOLLOW "Data structures and interfaces". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design. -5. Follow Guidelines and Incremental Change: If there is any Incremental Change, you must merge it into the code file according to the guidelines. +5. Follow Guidelines and Incremental Change: If there is any Incremental Change or Legacy Code files contain "{filename} to be rewritten", you must merge it into the code file according to the guidelines. 6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 7. Before using a external variable/module, make sure you import it first. 8. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. -9. Attention: If Legacy Code files contain "{filename} to be rewritten", you are required to merge the Incremental Change into the {filename} file when rewriting "{filename} to be rewritten". +9. Attention: Retain content that is not related to incremental development but important for consistency and clarity.". """ WRITE_CODE_GUIDELINE_NODE = ActionNode.from_children("WriteCodeGuideline", [GUIDELINES_AND_INCREMENTAL_CHANGE]) diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index 2eaed7114..7004bfffc 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -34,8 +34,7 @@ ORIGINAL_REQUIREMENTS = ActionNode( REFINED_REQUIREMENTS = ActionNode( key="Refined Requirements", expected_type=str, - instruction="Update and expand the original user's requirements to reflect the evolving needs of the project." - "Retain any content unrelated to incremental development", + instruction="Place the New user's requirements here.", example="Create a 2048 game with a new feature that ...", ) @@ -57,8 +56,7 @@ REFINED_PRODUCT_GOALS = ActionNode( key="Refined Product Goals", expected_type=List[str], instruction="Update and expand the original product goals to reflect the evolving needs due to incremental " - "development.Ensure that the refined goals align with the current project direction and contribute to its success." - "Retain any content unrelated to incremental development", + "development.Ensure that the refined goals align with the current project direction and contribute to its success.", example=[ "Enhance user engagement through new features", "Optimize performance for scalability", @@ -83,8 +81,7 @@ REFINED_USER_STORIES = ActionNode( key="Refined User Stories", expected_type=List[str], instruction="Update and expand the original scenario-based user stories to reflect the evolving needs due to " - "incremental development, no less than 5. Ensure that the refined user stories capture incremental features and " - "improvements. Retain any content unrelated to incremental development", + "incremental development. Ensure that the refined user stories capture incremental features and improvements. ", example=[ "As a player, I want to choose difficulty levels to challenge my skills", "As a player, I want a visually appealing score display after each game for a better gaming experience", @@ -160,8 +157,8 @@ REQUIREMENT_POOL = ActionNode( REFINED_REQUIREMENT_POOL = ActionNode( key="Refined Requirement Pool", expected_type=List[List[str]], - instruction="List no less than 5 requirements with their priority (P0, P1, P2) from high to low. " - "Cover both legacy content and incremental content. Retain any content unrelated to incremental development", + instruction="List down the top 5 to 7 requirements with their priority (P0, P1, P2). " + "Cover both legacy content and incremental content. Retain content unrelated to incremental development", example=[["P0", "The main code ..."], ["P0", "The game algorithm ..."]], ) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 2ff33dff3..223b0fb10 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -35,6 +35,7 @@ def test_refined_simple_calculator(): assert False else: tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + # After running, there will be new commit if tag == "base": assert False else: @@ -261,6 +262,87 @@ def test_refined_pygame_2048_3(): raise e +def test_refined_snake_game_1(): + project_path = f"{DATA_PATH}/snake_game" + check_or_create_base_tag(project_path) + + args = [ + "Incremental Idea Gradually increase the speed of the snake as the game progresses. In the current version of the game, the snake’s speed remains constant throughout the gameplay. Implement a feature where the snake’s speed gradually increases over time, making the game more challenging and intense as the player progresses.", + "--inc", + "--project-path", + project_path, + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine_1"], check=True) + except subprocess.CalledProcessError as e: + raise e + + +def test_refined_snake_game_2(): + project_path = f"{DATA_PATH}/snake_game" + check_or_create_base_tag(project_path) + + args = [ + "Introduce power-ups and obstacles to the game. The current version of the game only involves eating food and growing the snake. Add new elements such as power-ups that can enhance the snake’s speed or make it invincible for a short duration. At the same time, introduce obstacles like walls or enemies that the snake must avoid or overcome to continue growing.", + "--inc", + "--project-path", + project_path, + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine_2"], check=True) + except subprocess.CalledProcessError as e: + raise e + + +def test_refined_gomoku(): + project_path = f"{DATA_PATH}/Gomoku" + check_or_create_base_tag(project_path) + + args = [ + "Add an AI opponent with fixed difficulty levels. Currently, the game only allows players to compete against themselves. Implement an AI algorithm that can playing with player. This will provide a more engaging and challenging experience for players.", + "--inc", + "--project-path", + project_path, + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine"], check=True) + except subprocess.CalledProcessError as e: + raise e + + def check_or_create_base_tag(project_path): # Change the current working directory to the specified project path os.chdir(project_path) From d07760572073981fcbaefae190015dac2c4403ea Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Sat, 13 Jan 2024 08:51:35 +0800 Subject: [PATCH 308/637] Update product_requirement_pool to product_requirement_pools --- metagpt/actions/write_code_guideline_an.py | 2 +- metagpt/actions/write_code_review.py | 16 +++++++++++----- metagpt/roles/engineer.py | 22 +++++++++++++--------- 3 files changed, 25 insertions(+), 15 deletions(-) diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index cc532ed0f..528b4e8f3 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -110,7 +110,7 @@ CODE_GUIDELINE_CONTEXT = """ {user_requirement} ## Product Requirement Pool -{product_requirement_pool} +{product_requirement_pools} ## Design {design} diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index 7fca9748c..444af51d9 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -159,15 +159,21 @@ class WriteCodeReview(Action): docs_file_repo = CONFIG.git_repo.new_file_repository(relative_path=DOCS_FILE_REPO) requirement_doc = await docs_file_repo.get(filename=REQUIREMENT_FILENAME) user_requirement = requirement_doc.content if requirement_doc else "" - prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) - prd = await prd_file_repo.get_all() - prd_json = json.loads(prd[0].content) - product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) + prd = await CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO).get_all() + + contents = [] + for doc in prd: + prd_json = json.loads(doc.content) + product_requirement_pool = prd_json.get( + "Requirement Pool", prd_json.get("Refined Requirement Pool") + ) + contents.append(str(product_requirement_pool)) + product_requirement_pools = "\n".join(contents) context = "\n".join( [ "## User New Requirements\n" + str(user_requirement) + "\n", - "## Product Requirement Pool\n" + str(product_requirement_pool) + "\n", + "## Product Requirement Pool\n" + product_requirement_pools + "\n", "## Guidelines and Incremental Change\n" + guideline + "\n", "## System Design\n" + str(self.context.design_doc) + "\n", "## Tasks\n" + task_content + "\n", diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 1d1fed2b8..772cf0944 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -347,22 +347,26 @@ class Engineer(Role): logger.info("Writing code guideline..") user_requirement = str(self.rc.memory.get_by_role("Human")[0]) - prd_file_repo = CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO) - design_file_repo = CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO) - task_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) + contents = [] + prd = await CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO).get_all() + for doc in prd: + prd_json = json.loads(doc.content) + product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) + contents.append(str(product_requirement_pool)) - prd = await prd_file_repo.get_all() - prd_json = json.loads(prd[0].content) - product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) - design = await design_file_repo.get_all() + product_requirement_pools = "\n".join(contents) + + design = await CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO).get_all() design = "\n".join([doc.content for doc in design]) - tasks = await task_file_repo.get_all() + + tasks = await CONFIG.git_repo.new_file_repository(TASK_FILE_REPO).get_all() tasks = "\n".join([doc.content for doc in tasks]) + old_codes = await self.get_old_codes() context = CODE_GUIDELINE_CONTEXT.format( user_requirement=user_requirement, - product_requirement_pool=str(product_requirement_pool), + product_requirement_pools=product_requirement_pools, tasks=tasks, design=design, code=old_codes, From 28c700341620eb16179201d455f1802028bac1f2 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Sat, 13 Jan 2024 09:40:02 +0800 Subject: [PATCH 309/637] Add 2 code example to data --- data/Gomoku.zip | Bin 0 -> 94089 bytes data/snake_game.zip | Bin 0 -> 119511 bytes 2 files changed, 0 insertions(+), 0 deletions(-) create mode 100644 data/Gomoku.zip create mode 100644 data/snake_game.zip diff --git a/data/Gomoku.zip b/data/Gomoku.zip new file mode 100644 index 0000000000000000000000000000000000000000..d6c6b8d16148047dadbc180ad3974b612c48dac5 GIT binary patch literal 94089 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 F{|Do68TkMJ literal 0 HcmV?d00001 diff --git a/data/snake_game.zip b/data/snake_game.zip new file mode 100644 index 0000000000000000000000000000000000000000..d737ec0d3d862fb7211da4c0a523312fd8b91873 GIT binary patch literal 119511 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�v;=JNQFLwp)B2zTelPChT8Mx8E=s$EA5%7w&h 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¥3C)2XL(kJ+9LpzVY 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 z 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 literal 0 HcmV?d00001 From 1b29175116f7c21fde080f7a5f3a10af18ea7586 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Sat, 13 Jan 2024 10:02:54 +0800 Subject: [PATCH 310/637] Add 2 code example and modify 1 code example to data --- data/Gomoku.zip | Bin 0 -> 94089 bytes data/simple_add_calculator.zip | Bin 8105 -> 56538 bytes data/snake_game.zip | Bin 0 -> 119511 bytes 3 files changed, 0 insertions(+), 0 deletions(-) create mode 100644 data/Gomoku.zip create mode 100644 data/snake_game.zip diff --git a/data/Gomoku.zip b/data/Gomoku.zip new file mode 100644 index 0000000000000000000000000000000000000000..d6c6b8d16148047dadbc180ad3974b612c48dac5 GIT binary patch literal 94089 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 F{|Do68TkMJ literal 0 HcmV?d00001 diff --git a/data/simple_add_calculator.zip b/data/simple_add_calculator.zip index 20789d15ec59a45b9cafef3bc796985e565fad6d..05cc1cc6cf933737594b4e4bd7270d4cf3ca94e2 100644 GIT binary patch literal 56538 zcmb@u1yo$yvM!9f2MtbeZQLQaYk=U;bmQ(p0|fWrPH=Y!?iSoFxNC3^guiq2o_*gt zci(q@#wcuzHG6*LRdd!-Re*)Vg@S@Yg0fI%(KdN(oXUdyg$)Y@MGyJuY-#IYV`>Bh zgN;By8<49F(8b<~jTLO_U}^_8wF6n2I8}TI8SKOb@O}1x>ZaAWF|Z`HC)w_j!oX%53>m7T|I~h zMmmI|RHd3FMxi0=uf2UYg?2oNv>h;4p<0^QGhBu@D>Ys;D*j?{oRLaCs!Z5f&p8)@ z>-xxLct|Yxxh{eg3Yo*HseR5E=RU?z`V~^xi~p|+IMu&lgo46QWzmMf`o|YBDgDYbwYbcJxrNjELdlhfrOhrke z9SUico|~0sD&1!Wjhq?{>sj^&k=S2D{p;>{BM`|&A-j@;5beL;9oWo7C@(CEaw_w+;W zp$?&ji>NH8w_*OV1*pF;h{@8qhye1@zlr?cZ$VZ{Ow!5JOn?IN&6&-@6bN=^vjsZ4 zm^z{S9oB!B*8jkC3wwKO=Re3SNaY@x02&Hv4MIHseoKFS9O&R+uvI-9dP z1AiNUW~_lSEIV4@x-&|AB_mcoUX`q@10rjzU3+@*!@H9LmeAJidK*2%!EO5i!<7nN zk&2;`Al)u67@ognSZb)$7oLfg*zkMB-$WRG&$)7f%7*d-QO6@_hIs z?Vob?khDT+b;-@Hj&QFx_u!I?suMNDPK5+x#%M*dUUGvQcQ4z`gckF)jSUGhq3M;Z1#ir4vs&=YLLCHt)ZV_#2c`&^FxNB2zD_kX{kT7O?ZZe!gajr3cg$j^hf!u=jQeYDbU5Ss;P`&?}PNcd* zg?O&T9%U^q23FrP@fSrN5irxCJ&2x^NY>h`gb@jT%2nE38byYV8jP%hHLPPlA+YLU zT8Q?2KP@g*DUW=KyivXT(?l|wND^f~3onD?2A08CT~>`>jy?Z4+7)n=0IQWLXioZ1H zt6@m(gw?TdmZ3E(NvCQ=?%C_qpAo)UNxPw>Iw5=m@Ss95_NA9P)k_(^*WFgzi!xO^ zu+vXFp|Rkf6NEmgRZqN3aXO1v=3FRcl|~&sdoNy*JwFYkI&;_mRgKQPg*=E6p`hGw zpr8c)j2f9a+uGY%LgEsO`){+g1={`1YAfC^+Y$5Q_#}1YPGeIr#I|IfjEIM~bIO{T z)e%xlv!=suMZR@?hqH~pM7yK~7qhqASo&?g--!gV)x%*Hta&l}1NY%xO-nFcDRG?9 za>T9G4JV_K@ecH>9Y>2v*nRVO?jd68K#At2&aw5G(087qGu%{DUYpPvnYFR>as!?I z$L9lkm(+Ic6w#=A{X&wxR>=cM+jd{KbLJO-viQx zn?h-9NF}_Yp9r#8QK7YH)m?0XAwf+^ zfW)SD_#X-5YKRx7Wkn9GPd(kWZe(MMG}<~SRh5?61r_XMqmdu0BHO7Q*5rqfGRxtD z+WijGs<&xr_isYhK`v;rEQK_g5zx7B(a)h+k{p95X2gaxS$=pff@#7AkkymbHm~xuOD>k#0w zaKx~pOE(fY`z6fGw&yc-6n>ClbLN?t3VwU^p)km0O{-6!yu_4{(*~@fXn==2~e9vnWY1C+I19>%{j6 zi`MYHUKV94djeZDXIL0ct7>@v}V3IU)RW_@v!0^MnN z3PR*|$q}Kf+=&+bBCgXL_F{vT$&~to97@5u?^|CBHiIM#im{H~Vtw%0pjyPlqm4ij zk)u)o9)2-A+DpoY0a7w7|{r0#ruc}f6Xc|ZubNX$2gOii& z4JM5<%0xC@g^^(=I6b?EgwC^NE)TYeUjPeFR|!Y@oFV5eT{dD{8-e;72bq$$yaPv# z#!lzXm3hM22DzjITgl~*0!vzi2Y{jJRKlb1Yel{>bT2r#Fad^1s@qPU8jtA^N35A+ z9wd$Nq5Gdj^ijN+bNVqdf~c56_4D5h>_i_wiro(OE8U8!J{jYI;QsO01WmWFS6iDF-Jlx?qmSG`p7AZ0r~e3gOrEl@V*gXd$uO3W*h z^-!jW$Zm7AklCiY(G^t#_py~_33mq8yENtlALnzCjY>~|H@5xG?o0?BV&W zT2U8nV|2!J^5zNQ&<}P72X^jMz8_blbvL(F5}0b-YUFOSKG7;>rq}<3fG#>`-qZE-_u7XYkf|w4ad9ThISk!)Q{ALH!HHzAhr- zgzAmqk{@%;v^^ipW>Bc-pl-du>)ojxvzRta2m9|Y@c$Ze)6nj^!Xcry9YUOcCgeKU zJG-#BI)H&LrhkjIEMsao0gzH>@+He8$jG1b10ow1MIy$uOWelBkBWo+)A2T2w%(LW zV+9L|2t5Sy`b0Vqwhl19aLYkVjWr)lJjZgEh&zIYz~@c432$(lHWSC$GSd5DEy>l# zcO9sFGul9-xXj0f*QsAWzM64=#g%wt$HvLe>x1Nd9w_7VR64GGtd#7Oa1Q;i&?dSR zVUQt7Y4M-U{~er6S^ilA{3HL@`mt5$nR`kNig#Zq1r3^kYmp%#@=B4$Hc z*Bf!o0-=7#B0f?rnXv+CHkUlA+Z1G9+$;jn5Im0Eu|WY5*Q3@%{FnEmzvwigrnwKd z^+&1;;(oUFRYvLn3Hlu!6glwn&0hr*vC zEk98)Cn@+qxxV!79Y0L*RNXGyjAH&PbI|t(zBR(1%=mvZ_pe$1-xzDF0kmJ{ z!FslN4ljE}jczO;Cu-&j)jLuRC5BiIeVfCApev-4L8DH?Ajlg2eA}J}c87jc!i{l1 zYxi5#`XaHjD@J>!+|!XRVA8S$sVrvN%o@w;`t58-k} zWH+UK&_N*k)!_O!tFGPd-A8@1PLZCyV5Xf|wAb)(`idQS8swDYoP*{nC)clW-=Wh6 z7)wKUe!oyL-5sB)M=h6zxyr4^L(c0GoIAuyO?WN!@C3Q+6y|K%)lydQbck^?-vf3j)&0gGD@M2>)|G zKzHFo`lV)p!o%c4pb#KYK_~bNsXja+f-%u4gIh~rwiW3)TGH%Wqy)!m+uA69o9bb@ zw5(1Xu)IA!7oJYhbr36S9}>Cu);r6iO$MW>*Qt%nc%RNb8{`qZ!M#S>R(IYYYHv8? zkjL4?^fmI5wKao-FS&w7vjO!>XNf!_WJbAoITIbW#8Ad~qrB$~pHEzhPhDOxL==JP zLZfTS8_3Urn%3BmnBP_How6edoxb(OLx=!NFlipuO8co#Gwm8UcQvjsaA9q+ zB;|KZJxo2UM<7ql9G&gX%$U&RC+3W(3JFOXR91>!jQ{!~nW*tV^ylldPk+$Tmw zB@lqsfA);40`S6V^5RTOX@7lM*!7k;$5!Ce$m1cY+l5uk%$?WA;gG_s9H6=UyDb_& zp4R--4U;FByCph*N$+>J+js=#@L9mtrnOVjoS2ah@;-1(e_*bU>c*O%a zfPsRVh5wVaEyQ6u0RQ1M|4XYRNMlNU8Gz+&*Rkcm5g7Z8^@m=nOR%~;=R5h3e0!n* zlWY*4o)ZlbO7GwuHNWMTRGxT5ncLGJKkQak6_Uhh3lCfb4fsC$b$-xHMha4IG*)M^ zLe&zhsM&knjgM4P$_UKSU@D#@?C43UE^1?D|FjD$$<#tj7QSJ)*po+Hg{5&?<*X*a zXqU4wkcrh{YD**&TUM)a=RXDHK(h*rvrdDev_U;029!=Q3_#UUoxC-7&9gSe3Uo|2 zmVW1!)6OWa4Aaa@v7gHYM8r^+yA9R#uUX+SlbmD`49#Rq{F>6gHbN8?=Y2>};1z8L z_qrYZ7j?=P3afTg|4>acynfQb>H+^)%cWRoxY^L-NX3)J(I6iD1WA*sS<*`MsFw2z zN#%&V`r^ihN>Ic$5(@Ki1MrmD$)>Y3us?ue~oUFL!NL1a--k*gB0er@Pa~udjxtn(n^N$i2>c zeyNr=fO&Nhw+{10_*pkTeNt9_nM7mA^{G=ap09AQr5!vOQhX;awPTeiaKERe%6Y`k zf-NP3*>x~PMbys{jd04_W#^wDDY{amqWXHw6pnqx29J7|q2TiUJ3FKY6iJMH9{*>b zg9j?i5CB>4vBB2n{i2c}Ngyhhkn2xHHql5gM7s^r1qbC&sEkYl7FzO&j`5AKBSjeM zYb7UP(NN6w3B(H^n}?R4J3v(as6%hj5qY2OnUke1%@C$oj`LOAXBLKOQ!jZAOW*pD zqe{JTD(-CIj%z`bGmNua(b3h*gr?g73DkwzO$YNf15aWXU)X=OA52V?hjxhl7((m^ z@MoQpgR8T}-_ngBjdkl40GhAJ3!d)jV#B2V$F_?4AT>g!NREMFvH_k9$*^_-H#0#c z!P`NVTIcp$T*(tt7S+$-G7Xt+UM_-{P9h@@gePz^ngSpm7?=ZmeOSh3pL*=W2sc@zE&M+U6tK2dilE`;nZMpxdT! z8faf|>@4l`LaJ8)(5`}WKq*5&mdz_4zsEyfFH>BnhvlKG;ipgR>-cHc@qO_~JxUPQ zI8{-WpUZP~FL1Gd39i#*fsQb>_hmUVd6%~PRR zBj0F-nR+!eWZHep^4O1f=;CcH32gf$)*L3oRb%*R5b=nuOltUSF>g>~r3u3mP0TGU zE5`qPS~5a~_Y9RUC8N{4@RNIHQCDviR`V~@8UdER19T$uAgnftAh1 zGpIF`ei7RUY5!CnBitIdS? zo^9gGhi&QQ6TF=u9bwFiP*H|ucb$C2$6sBRW@DY#zEYQ2p?_$T2A*iEZ9&?kRImOQ zxwn(43DEiPC0nMpwf(X*{xkEha6PHkq*~9#M^%??fnX4#?d@l%k|`5x!Q0T+^5t?E z?q-BiQP1mlE2*2kwbM{ceqUwR?u75yJ#SZSKI3!NMQ2G(9Ox(NWme871z&{jg@mlf zJe*-($CS$AWzI}c5d=BaCtJ%stB-^wp#s6#^YW`|RXgRbj ztSw5WP2edRI$kRAK{)+!CTdu4zHvb`sE0Mf&c0a|iFLCuLiLMrPZ_$uTI5(p3&ru` zMf0_om$l5kYriMPjbM&fkU8CQ(kQ6W4PRL$EO}>X z@doX~P3ASNQYz4plewOLi`(0*d2N{nJ&`sw*-at6f%ysKd)8J~Z>A!=UCcv#`}%V- zQGMmyKvx*L&fI`hq9w@(<2VCU1sE}WqH+T}N~&Z}__+;!`WyO#N!U*n)615)LFRPj+$I->7jhpRW@DxU z%i(ArKbiXoRAzdqobuE3;XqPF$JdX>VFCylS>stTxoa1W zeJ{WCg8~R%hk~puB}pjV>^}o0oz#=m!Q85HLc^<|FMdjRPgxq`&45@@8c$Fxy62tB zcmZACGsAI{D(f9bsHAf3=v+{@7XhGa0N(m?eSYGn*CR~KRzq&`$IN#xEb@gW@7kpN zneV-Eg{x^7H(aUl$}GBfD}@W&G^LcM_2TM-^X2DbSL-{_+d1k zA!Sl&fsVXu7a3yMC)`81Z}i_w!}G>Wh=@^_wlE&wXo8oC&6@bXBP+j(h;j8rFcL== zo)ng}_F4vJe++7WJ^|X7Re}%~jPAPIqFsFMzD7?-g-YhLlJt>MDT$8sBXHMtG^46Jhn*J5=011-YtSbdqKTI8Kk*fnO;_;@_3@r5%|G(Vx=9rS(oMlJ+5KJAMenRM%|zKE2b8>@wGCVZ5yvmHFyPGNZI6#f|Yxv zccMo(uOBQn;O|5}41H?8vadRMxUtyTg?iO{rccblb+Gu-&ydk3nLq|plDDfpch|1| zCh>Y?xBvL=%mB9?y1i68_j3a#SuWkIPbnUWK?s%rhd!e1%dxxC(F)HbYE(?w%OH6b z>ssb|o39tP)N0O&8_FA;=F&jJdGxIdAwyKI>>x?xdeF4eddohj4S$3I72&={l{(T6 zsF-;5_GZHuTk1Mv4Xm9?Ua}m{>_N3q|82cu zLnO<0dWz5ZH>9_*Rfo2Zc9Dc{nXAkU(4*MPUmv~gl+-HMS2n2IQPz=-d6Y`bEniMe zzBGgFPOC8XWpGJyT7XaL%j8TdHeQH%5pKrlq*XSI`YgjsP>%yjsrON{tVU%=q5 zE2Cb(Rg50Y9EI(9tgCoVVLtXGm7CK`nDT+nbSUe<^MW#-1K_>rE3xj}n|oY#9Eaot z^=UtPAy=E-IKe%Pd!GLF&+6J$52Om2ayq0g_8Mh{(@+cz?aW5uT4A=FAjI) zBm8u@PB*HdED63@4C*JYk|( z^iwWZh>D0eU2=Y9oH2#Xj>gjkJdrwQ$W4N9g5u?o@U{*ysgpY0BEGA9z~%X&k*o(t zp6(CDY2#P%$kpYn?BVcP7Kb8}b6-cHgiuPdDi*DOZl817C03at6N;w*gI)lIaq$Q7 zl*2D`HCO1>)vUDd4gqWr`8_R*$VOevsR!#MRDPuC@YiO`7vrbfO?S9F7_tWB{^hou zh@TKeh%2d&zF4O=_w2z~;v0NWML8BfNh3muC>yG2Y{$~m`EJQ8SCILr>a>luRLz92 zyCNhI9!z*^t(JB>8TyOO;36_sFe9?ZAEinN!>7zrg6LV{JFEoJ1S-%+Ef59%)=T)j zcnEws#^kr=IQs45(Thq4C!Q$Q#*Vd^xmTE(;Kc2uOV%Zxgs$^cSy4jW_?&NL7M9x~ z{&?S&OoG4NSLcT8vbO^tJMr2E{mb{gaWkVfVcy?23l<66QFSC{H+y#5$;v$vDTyRd zzaU^S4%%1e(D7%zkEWADeP1JVbMn5F^?Y6;T?u(L<#|i*7iE89=-J#2wX5fNrp75$ zYy0P%VAE8+nw7&A-w+YZ%QSK*2icvgm(;TlwOu_G{qoXPS@az{(3#12kI=fjpFW)V z@8rc4{=_pDmAdCc`566nwwdiq?cgly7%5^+Bf#U>a3SU49jJ%M^T^y0}BrwRHP?%HXOxqTCC>5*jpOZ^HI@ zt2g|aQ8pCzdRmLBol?`T<+IW&!SGc?@h3sI;XEYAr8!UY0nTLs;>rXv-67rbAVGA2 z&{{^M=@fJut+L^6q68C(McsfQHQS27k_?XT7(z^m*GhH6jqg=y(RD&|?}LKld%6YG z1Wyv4!Rh%wf&vZm3JHc+Fz3onXe=kF&$r{Ut;Cl$tc5LL@)7d;hlw4h*~CK? z*h-`x9^%TXIX5(wp3^wg1-D=aG%#f5bE>+xL*!aA*s{oy>pTQfm6Gr3`nh&T^(L9s zhwvh9Yp0OW$e5yiH-GI*S@-6Ky5&OMnJNrjkrxwOq$u8_UNu2G(vw;kAs)Ft&qi)2 zjak3rG3O7P#tgH^>g2|$FuWXh^ztQ5*{-P?ge6PF7zX0{R#ROm(Kr_6IrytJ5;I$y zG@FqBYgsdcfONhG2L-i`@@IQ3kdlVQ#hwLZVG6RgcXjz&c@w3jMpFE^rQwY8Nu zG+(cpI8xa~ev#pi6b>8zj<}gP1H$SK=i}jsjTLvzCo3+o>2sE$F7KX+!Z3;K&|^`C za$dL0hCY9vULh4wd`F_mub$ti5v4AiVUQVNAR|q&{8^*>N-Y!yArILy&8b%AtwMBU zF6T!K)t&RYDNoY!WuyE>ID7&5B`9R&`0T`qDl59Eqw^{>{j6||;+%Ez6lAjjVR?u*vwY8zb9sg zPMr-rNy?Z}It8g=Lvg!0mD##feQtaMx7))^WqtGmNP?37+fWB`s{dQ zo!df0btxyqw#g(e;@I&sfV0Y!uj(j?=N6USYTUn zG@@dfQr6)1uUgYRC!+E|1!z+Ako@2_HwW06f+CNX$a+pD{8!3uyPx(0#`{J|e_=Q! z6r_x1E|1{~r(3bt&h4ms(D>kLj2%6f+k9aG8#q;a=G@p$5~{}64NH>f z$MhmULTUSG+X+#c*4lTwE5G&X1yPh?FYMJFQV+-CtOfH#L7Y zmcTZxom0-!8Wk0e3qM)3bf3`jh8*sAN)0nuu<}+CjJzLOldg%UUEO+vtHRnXDo4R~ z;y-Y;lF;A(7Jo})N!AZ&Ua=xu- zD652=ZROnB}{j6_*J# zi{;~4Yf6F0o!0%P8&3IPGw2$Gc~hyN~5(A3Iw0UM_#_Sv~(cb>E8(RtuJV ztC8S7T-KP3Zp)^0@>87))Vb-7dL*2z`=}{mZds#ryT&=TE+0e}$Ks^fds792)N+SW;~s z-ZSTfAPYR{vrACyCyDsxQ(YY=fB%3 z0^f>u`VxR-RxuNhIREt(4q#jzoD~FZ{cpG82w*9`Y> z=FNeA92dvgJ-X*?NlvgXL@H!3pQq`BOk-cl>i zA5_O{WwJySvg`_#Xb}-9gFj?RNVbx)rX5?Xz-g$xY1L)L&Kmy0X?D{XRPs?g+2J^i zwOv&EgL!4h%^2I4am(27FPqYMVX;)ZylvOFt?z5M_vGvNML+f&RZar#!4{E+t z*xH5(_Dj8EeMsCLaeqAqSj>5<)8K*+7E3M-(Qr9yRQ-~KIn-$XemdRmg86XlM^02R zh^C;FqC^!g%l6=PcdhEjtG-2+;yL<*(3y?v?CzwN_7k%EkJ6NqjI5E;ZytLixY_U3;@E!>!@0Yz7fPLNQEjBr5Iga?m#@DTG`5ry)tjcf z?{l} zPwdjedMli97?!CTKehNX{L5W5KqhaVqQ7ul19*BoF#po1OG;|%`V?ZFXuax$a){o> z)gR8Aal(D+6|y$hAW-N#a0`!PLDlnMowm|hTdDv{6$$5Nb?xnlK0 zWoxby*L^eC`E6)p3nRx3)?8MB29uqLu!tR=f_Gn9&f@G&_uG~E_!-PIf)7vB^6w|{ zO?a!664Jlh=j3mLE+=|qgpg4K>-KOBW|9ug7Y64avu1mu#8^1|m@FLXaO|~mqN4A> zC+>*b@a%jkrD*uABvoXX=}5DmEcK$k$S<@itb0)gjh_#=5-10} z{Ff^XzwhL4cS`NiK~C8zAl*KSfA8TeA(!$$IA5SSnHd`!!ZV+zx? zZ{=qz@}Czx56B(ldtM=BEGjK}V_gNwSvqm4S;XHb7)SfvImJ*(JOr8UB6HI*Gz`l6}DP_GnA7;X1y|on$y3*FkC&@R($QIMW&9|pl$DDwW7IF=x88^PPqFO5nOCQVG87SPg>DZC-=|KMaqSP9KuoO z0qSP3pPhHnJf%wd9}1*~O8Ote;8QR9;Y)@t!aNu*w0&aJ14d|(-ZrREQptU|%l@fb zhPObFva*zvanPv~*o_4=a`LqkvqA>ACKbR`D2|pekN4GMS0xtT4*A^sUUcmWB0i9y z@~n8&iFj@G87ZL&ApKOwQNY<2%jzS0zOR6Rt-nuxsddSkpQ=vz$uKHo6j4CfxokZ_ z5mSwsA}5RXR^uvjB6*HK+p2M4EpN5${4gh`E*V~jvzTg>O_?Q9Wn_qrZuL#^Ei(^0 zh9&p2YtWC`kE|06@2Y^wHYjuNNTa?d%sO?2f5O1QQX!x-w3*CQw z%Vrw6CWAo%mhOYC`DBkBCV&zbN~cTXGje$u<28r8le6)j(DOx1(0;1Zd7JP+Mns}s z1~Yl5Fv@=!UHIcr$p2_$X7+!WGH@3jqpX0K<~fAu{wEWKJZ<^`WaA1pWr56x{GOV* z%QE>AQM%~H)`X+)0qcmpoXr!rle`nM7-i0zcp5y|mWq;xy|`!rV#u}BrbV&-qGCVu zHVWf})z;E=!1|P@t;~V9gOGzk3>Ob>>04k+X%0+|`TjQr8-=R@XcXrS-W!-fwG)_a zM`66oFH5@P12Nw+PW%WGZ@&pq#QC}EEE=*iVWWtcWIx?jEAtUX1-+`(g?=OTc-7nd zZSc1Hs5$uT6!GtA*MA?&_#gOXV{iUnr!s!KG%lr-o>T~*m?7;5>VMC#zdZeWS|lo3 z45R1wFwJ@R%IDZsTIB$&0N!($LD!s+{CdfAA1BB$T&1*No-)y_dl;Ss48G~5A4P`A zYkx~Z{c}pfRTMMUpk}lxojU$JRO2wA=Hprt({X}NWF%DA2cytbU6mCcrF70O%9>g4 zJW>Q2)7|G>J9QMOGbK?|w510KrI|XqlbSFYMFQST!=?ZHw)}D3)>e~z_2u6v63w50 z_LupW|2uD){_GNe9ijRALVum+YM9}|%7iSy4q-F<|8Akb&f-9L4pC4nsfAly9mass z`XOp;Y_EhNa4B@WF7|{y0oj+2ccPfTU>EU!v;FSBuszL>!JNPwsnE%Kl%>i=7u>0swui6^O$jh%{ck_O}Y5^ zI801FKr0Lnsi*_k&CW+zzR8U#jWMaoYS53vIpX5HQ^&nTRgb_?;JCWNjr&J@gBnW5 z@&Aueasd8-(%+t-niyh@zv)k+gsj00>AtaZnwXe@z-Amkpa};%(2N(r$pr?P@bGZ* zuydK2aI$lKfOef!QGvu0D2EAJ-;iW7K7N>G!h>$GA{{MlN-^E`X2tOrIa~9Et@~Sb z!N}3_!8_kGBh(xe*}~1GqJH%}t|k12X1o6y!a*P_b!7;412&N?LYN?njI=_je2*1*G4nK5JxL`oz2Z)_r}u~Q^8$3Ym(!E zuzPFiL7WkJI(q1A#k`|k@md8117%E|Q@s7dwPgRfAbI{E$g@`sMQEm9)U9n5E3HIASXla8Gk-vHFKF zZb*V4mioXUPis8e9F)?5Ow6Yt#7StP*$OdeBDr2QCIG8Ral*jPn{T>0tGakT-*S4{ z;pWSE9YZe36PWyUU(JQH!=>uBW6$cc!x!2_DPQwvRRrk3Pk88i3}0`X-16u6sH<32 zezK&J#BBKPrb*IVpdv4G70B;>9N{Wb7?8N!i!e7sCe0V}z1e{5Yp4U zu#a6@_~ByPJ_f2IU-VUTuRh5EU~nAh;o;LxZ=8GMkm95frF%zeQ0=4B6-8;7BFuai z4ZMO8`F3&ReDw8;9O5s?Z3H&K34Yj76bcbC-?rpERT62Rlnal2A<(`hrND(~3`2~> zIn3oAf5(b%v)Sc%B=v-_y7e~EvgcwmVe@bf%HB5&xUM&olVnn2 zoS5ZK3NC;t)v^-~X+7G=KOq!@*Akw1OEUX{DyU66yODmvi1bIL8w z#ZQX5H@gI`$L#?|4Y!c#@*=(VoKQVE{*&Y9LFUJ1|yPjnP`-no01Ao(>wQGJh zJ~I<;9urdzE`AOW4?hH8ln+?M(QvWFM#gXV1IMqNAG z`$HjhoFBT3GQ+gHeI}yZu@VeRXr>Q`4u|;lb)?EhM&#HTL<)vCgU(ir;mTHO<59b! z!sa|wKmU9aEBI2W1AO~FX9fXz3*ZaYd7+S(<1%~;M!=YOQ&E0eZbBY088ZM#^Bf%J z3plKV)h)&;N9X-gijq(Uml2yIg{?JpV2*PwWVO4+K3Xx$0|f$%SF6oGVY?CK$MaLU zZo?O24NQLKx~prPjI_l{lO>Rb>%c>AELFx!^$y!_I} z`f*2-(vgRQZd0umdLGNZwszsoEo~KI(af>dQx0dzubA=Pt>{av=(2HL`ZM#l(5|KE zHy(NFex3ZCU$t4-6OGntJfPc^6u;-+s*y>!NK!A72!7w7CvB%@AVr(%%-u5~)DU=( zjyqIwpGjxim0+A8L=)Sum!cV;8${J1ru?CLA2u`_wDIN>CZTEf&>H-$AxY9V`IEqK zICdo-+la)^%wnsK1T}s-?SlCylmVeCixDmi+BRxE8qgkR1J%+?;+^Sitg$8FUX<@A z{wlm{U!qN7yb~^Sb@hKX4O^2dOLZQE0x3TRy9D#O2{_9?zQAP6IJx}i$`1a6vQztq z0)D^z;WzzRWe2nI0C;(L*iE@j`OUb&AWm*RejY9pAb^V(3u%Z0KS``IH{sP9)4&Hv=W@3L7{Iq~a0WcR=6e}RC*6vV>=<^!^W`9MHE z5D&oA%#<5aRq&bd@Nw{Pv9oi4IU(ly*QA5vD;U5I1ryX4FnYfgFuQZ#*OqV}*SB?I z63nk`_wk;!r43)Hlj>-b)8__q24{xVKlIQHUqX{glWJP>_9l!!u|KsR?M!QZWn-mw zrpw3ix_2URjZHF=B#vVE-Cx>T`1$$x=7sz08O^I7woOqwn_auxrI@o(xHIs>&-{v+ z>E(e)a~51n`rY@!zh;W6line(7nXZoK~d>t=n`fp#T$)plwrSnKA1_{f3ex_1FMv1 zD|+xT5SHfGHc@jEn5hwg@{T<_ynxtjKTJv8epuleD?r*aty5r$e=_usU7tT$^Ko(h zL9L*pKUwiW()r)?Cxrr817ya_!)waL2@wb{rx~P10dbh{aRMQV2I4Sd=Z91^+#H;c z!o~IT805wU+91EmcftFwlr?hOT{$6g?*{!PXzD95V|mO_^DVgDWvj^U~TNpSUllz|N1pP+K zN8;v^n2Ga)TO^sjh4nSuu;qcziOxyAA+j6|E_=ds>cpEi{}@lg*1 z`QsOo2gVzZoefe|LJa-BW~}}flGuRV0t*RSBB>>(6>-9Thq@u_t#KAcQJQ4wozKrn znoi$FfIQPb(l8ecn2%od%mFIhVZ$=qPsi@HY7SQx*?SDtcEo;m9oDP66#RlFvi+Ly zpUe1OEUO{l63}w+S0Jq)&7J_*|rjUlo2WV6E zb~{L)@029W{T>eX%U4-&NuMgR>rs#22WYwuT!m=T2Uc5;pWo$>{5``be)(?qJi^#7 zKUvniEwNeBr4g}78(~g2EzlgjCmd|m32x%jRMo9YvAKa+n1>JgfsuG4U-{E}vFhS+ zEgNk^5|BONRan1djs9=}i)kO{kXF=yX(ZHLFhvofN!Pfx=|ZbPTa6}zmAxXBa5rMK97OBcK6kDYg0}h3HC(qd*7(cVe~#~gLVTS z#50TO9@15IgP=YPyN_(##(}^L>^x@egvq{%LmNWPV|n(q92*G=x?7kZVViAVa#SBw zC7Mw+J}O*%`W9!;?Fti`k34$u9J{vhE-8}s-ab$6&85`PBCWq~5J`HE_M7CeidKCk zPY>;l4X>XJ@7^mY&Iumc_RdBX2q363j4}U&#bTcr`lk&4AEOO7=no1@S{K0K1<`@u z^k?+|#KvO^iS+;zkO_dBor@FFUgTjn;Q?@i0gz+{$PLK_09-(m-x<5QlFA!)wB5+Q zkLP>pTDdF^(D`x&1zmX5!$6fX#jR8DLOc;oRxZlL`h~X`4Wk&V+{ymkN%D@yu7jhe zYwUAVvr(n5=!$qZ4NEjeof{&@W|bD927%Z%7*R(j%%%&UmKI!*rR6>|f`9cHwNSEx zhb;S;u@*)Tet=@p{H$pbgLiGKzBR)SGpjwvvNl*XdrWz~d69E-Ybu$Wk{Nv|80|fK z`dygVXYYGVY`L;Ny&esUa4@!G#oP!g*XLj5Wgmp(Y!;V?kjsUE3ps(`15%YNFJBoN ztp~+Nb(qOc`6^9`&77>ig~nJQBv@+Ox2Q-bXm{5!Z2oZ&e??pkR-cyNBcmlIKFGMv zu_OGT7m=95;2eq4Mm%ATV#R?U5l$jqVb{M&pV!|9_<~qsADekbqdI^UpFH++w(=T-FZU@0yMvnRGASEX*ZR8{{3;gHg*sK9eXq{{N7`FJRlRNP z!yw&ANh2Vwz=ll=64IR_NNx6}yHmPBkZzEW+%$+FUD7EjAuTB>Ao2aU_r3S{`QCG0 z@Bc8y8bclK$zE&CHRGAjnrq90yKDY>xEWCtwDk0ED+=$g02Nnfh)6PE!EeWJDheRO zn9EoI%n5+1#wJD}0e%y(F(1r`7tCwK%?~r?6X4_l0$vDU#WRK!G22D4MUAF}yLM1y z$5NokcZKB`70vIBKam=?9bP(ZLmQsd*&spRPjaab8Aa!&C%DVB7a>=}EcF(9LS8&v zi_ZSt*Dvx|nJ?Oc!UfmwSd=~VHPqRIA+H(9Ja^bT9)Ej+u9+n1Ojh^F(#^0BKP70~ za?!SK(p1zh5d9(O1uc@3sW4vJjdA05qw&iVnnNP7p})O~e7~-u%8PgJ{!3p7$lwI= zgTOpSCY&%RkRI^x!vwfZKu|D%Q$vjSK_E^}US0qN`?r7ib6==!1)&_XxY)*=E4q1+ zss2`f?&?Su3&9(m!13?T_1EGB+m&qyCjbQ8j^ET6J{SiGVhrOo0dPwJ9%CpUkYYdp zG#iR#sf5cTt~r#karEa-per@VyOYv*|Fwjw;u^kVY); zY^P4;N&h`0WdjK_2(2F5g}JgqDe)#?^zgvIwdF+jW244zs5K19iNb2noLti)xQuFpw#xQneZMd<{buqD!T*)V!&Hv08vXm6=Aoa&k)Mr&~40AFn=>d~lSE#Z32P zO_rCldp$HczB_(}N*7*fOpX72c_%3%w;Jz=ZHHggL0qCD6Nl2mAhp@WkeqbD@GUk= zS1A#p>F0{+`c+W7?9;oi?_dR%K6C~^83~O-DyNSx#$|s@Kra{3WArMLV9f4&WJcyP z5NvK8?h_NQSv;gpp8C8uFJo!?JCSP1hL+emXvU8O*MEHd5UnS`j;5-!Ou-?X<6c2n zu=Sm$*Tjk85AD7UetKmnaQniC$_#^K5!V)zd}>C23b&()`Ft$<&|>rDS});6;g)8J*2O_?Pqfq3Gm0}BFQ1k9 zzLH|DldaZulc`d@j7SmBYOdOI zd#A-Sw7)`Zy)aU%uJt1FYZWeh3cB@{Cu*i$Qcs1Vh9+rYv!O!hTJeY4TB_Uzn$q{z zOO^v&R*Hyd?hAB&n?)LLzGm4e>4!`@TpN3o2R#3j{+K{hKCeYNIkk2%^WLsArNi2!Z>l>hOM^UH|`S{z)+o-S5((JpD4kZZ%?SWC6 zLJs+E`*iGFOB`X@TDZe+=nbVuKTW3i*L)wsif5w7Ee?p&=tY3Q{l!p1oSjcJHe-$C zobgoB45?-xPtnR0i7&q|;cW<8P4O#eG$dF*%Jc!xS()+XyP~J$YB~>Tvsk9;Sriv3 zqP)7-j)8ixdC2kMh?zOPvAG}8#{kv0K35kOoyi~aq-NZm~io0Qqz>cC&BC_>p>b-M~Ipk);|AQj>^Vg=6MA7w>c% zjMDfr1~(?h`PSSQ#ih0Laz4$m(?91yLdpA5e9q`u^-MaGwbY4>RIC4Hst3cn!|iZS zvm3QfIJA|XoWYp$nohnCvfxH$Ft zGQrL;Ja$fY_*@>sV z2ya4Xgs`xhZR0i3e-utkXiYXgkeIr9U>#sYXp>%jd6L|0$~Roa-FGT2`cd?TpCzaY&6Q4ws+x^>loU~$LYo`;&SYh4 zbi_4LiPlD^`B|CbW6L}D{d_~?_F84%DP(;SFs+xdR*4VWG6`HTiB+8pXuTpJ>59Wd z-~NzrR1)#fp}&jY&MIK;ZbiIZ{hs5x*-X?F#U2?G+^d;4YUp`*r!7Qj@s6F!s{$nu zn)F2Z8@(#d%7(wY`)Ohk2L~r_cWAEv^F0poN z)#(5Ko}jb$2_zGUXm7{wdjcRchw&P5@o)m!I+)j(#~92h0EY1L++r~T{QNxp+z=4| zzw>o4fW+J)ptsb;U|GK;XQxBly{kTo${a5r?@wi#Sf8+Qum= zD#dp*9vIJGo5kTg)suzTyMO}7w`Zewx<+rBCrLh-CF=okxat`R0wqNJFIYyu30(63 zno5$$Sr(`QvEJ?Y&9ww%@R{&Kj6pD7ZUFVZjbg!^TmZtrXClB0<>LcNy$}$W5kJfg zF(JO~)hs7|u&2~Al~e?4WeHB;e7?N8=2l%uv6^y|mau{qTB)E5R0t*fVoyjZJzk@t zFc2%V?a0T+gz*_;NEeshhtx&iqfLf1p_XUugUOC|+%(bM^qAfxI@%gB6wL)6nqocX zpQZ(LxFH|2M4rm}2ySc=6`>?@}5 zY@w-r4RuRbjqM0@6hS|w#Jlo#uaLO5(SH$`Pk-_3`-RA)hHO?>|J=7X-nKoW{lQE! zK4As$6hB^_7h`=E+u}4@QWhLm3em5$-!Gi`V88c0?G$)=RYssd5{tvnG%bTVC`9$47MWHyY@(o)YUBl$-?XI8E*NuE<)paZAo% z%*k{$zNNe%d;0@DSBtvdDWUy~OG?Z&euPMu(1)T6XMEHn4T-FZ8f{`*tjI(coI@Y$ z-L4^~!qrp~BEm}DSNyEG8J7&cSkeMD_~r}=g&Nw51DJ2-5-#2ly`WQwJ{?V~h%OTv zF_xzHM2K?4HxYycj#pQc_&<2nJe+Uaxci40_rwSje7CD-Av zV;IG9-ABpV=&l>@VmiMWi*n|R5c)o_0tBOZ`W&Y36i_fC#n1|=o{^J;o@`Nib*+kO@Fbk{%>N z8M!wZ;~7M}&re>NoA$9M&hj*vD|6q76UFDPlOO;(XQI960x^d#OXxQGKD$#6UC*lc zy7n9wf#boGzu;j0rdJa9)hltZ#03chOL#kecNvX2z<^&jF$VA!5Eu;PcRaj&P%wxG z43r-P0G$y45D6$3_a930uT}h&Sfyw*va^_a^--N^+&Cpu%1w-=+VA6g2Lww-+~dEk zF-E^u4&%3^7&!n&)$RDrsRA;%fz%b484ov)i4lM)@d6}K6HX(rfB;y42g(bO1aDc7 zw`4#~)d)f{?9=xnH~8P$gpOG;!;imY9}_8;eyr{4q0bnsGd4AyOxgKFckLQCGry47 zAre3T;Al;r>)d6f9d+Tu&9}XoqYXRVPp?=HBdr3^O1DWzcDi#9%!m;p=+xv|h5Th& z$SEAHJW$a51DtVFpU(W~HJX(mtIUxIJtU$_mhsdw?}T=Dlp(W&#ekA38xdyv%=5VOe)j`b6S9ESgD!#b;gYVY z$Ws&=JM$bk8NX=IkGgPan{y1DIkeQ)=rU~@{(F(v=tAuw^w@SvwepR3c7Tpl(AKhj z|3r1ja+u3=lW)Cp&xaT%zq1(`U*(Bxtz>h<{bVmf6~5M6{)H{$TJH1?YUe3)0 zgII8)-VcG#S-R8rQJ;JCRDwQ#baRwYO4!Ig`1tYhcvKf$!Rhgxa~Rfua<%^xxSlu1 zTF#5Jx3el<+(Ba~tItN7TXq*XbdQ!I^B9}!18v*XEw67QEW)`mF@d)NI^mopg z=Av#O*&3Tr)^l_gcKK%BDu)yt_vtWrNghWgz|W%fV(Y?#CnyDURkoG+seX61f#Aji zlaIG|UZDM=PO-EnQT?p&$23k!6PEMB`Mv5Kw^1o}1_w~kgC7sNg{+m9UoI4g8Pv65 zVPq<63SQp>C8aRyKX~t>pZLDf2&)#0e$Y|6$li;#8P}yqsi>-4oL{c>Im4pvf}=EP z7T3L}%b7m}o(UB2RibRq3=S-W1mr?WW|uq~#_hA3!Mb`ece&j&laXpm_Vk+Ewg$TbruABy)(kd9NX*JzRJ4sxCawtLmZ^AS zjPLjQdhX%ZDoNJRy5hnLIw}?3hEu}&N6KQG!RVxtH~|Q~Z}ztD�k`l$bNP77p9` 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 literal 8105 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�v;=JNQFLwp)B2zTelPChT8Mx8E=s$EA5%7w&h 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¥3C)2XL(kJ+9LpzVY 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 z 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 literal 0 HcmV?d00001 From e079b8b1522faf64836fb2b5899a22512f00d5e9 Mon Sep 17 00:00:00 2001 From: yzlin Date: Sat, 13 Jan 2024 10:34:45 +0800 Subject: [PATCH 311/637] remove sensitive and recover rsp_cache.json --- examples/sd_tool_usage.py | 2 +- tests/data/rsp_cache.json | 145 ++++++++++++++++++++++++++++++++++++++ 2 files changed, 146 insertions(+), 1 deletion(-) create mode 100644 tests/data/rsp_cache.json diff --git a/examples/sd_tool_usage.py b/examples/sd_tool_usage.py index e470ff0b6..92f4cd5b0 100644 --- a/examples/sd_tool_usage.py +++ b/examples/sd_tool_usage.py @@ -13,7 +13,7 @@ async def main(requirement: str = ""): if __name__ == "__main__": - sd_url = "http://106.75.10.65:19094" + sd_url = "http://your.sd.service.ip:port" requirement = ( f"I want to generate an image of a beautiful girl using the stable diffusion text2image tool, sd_url={sd_url}" ) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json new file mode 100644 index 000000000..db452f676 --- /dev/null +++ b/tests/data/rsp_cache.json @@ -0,0 +1,145 @@ +{ + "\n## context\n\n### Project Name\n\n\n### Original Requirements\n['需要一个基于LLM做总结的搜索引擎']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"zh_cn\",\n \"Programming Language\": \"\",\n \"Original Requirements\": \"需要一个基于LLM做总结的搜索引擎\",\n \"Project Name\": \"search_engine_llm\",\n \"Product Goals\": [\n \"提供基于LLM的搜索功能\",\n \"提高搜索结果的准确性和相关性\",\n \"提供用户友好的搜索界面\"\n ],\n \"User Stories\": [\n \"作为用户,我希望能够通过关键词搜索到相关的结果\",\n \"作为用户,我希望搜索结果能够按照相关性排序\",\n \"作为用户,我希望搜索界面简洁明了,易于使用\"\n ],\n \"Competitive Analysis\": [\n \"百度搜索引擎:提供全面的搜索功能,但结果可能不够准确\",\n \"谷歌搜索引擎:提供准确的搜索结果,但在中国访问速度较慢\",\n \"搜狗搜索引擎:提供快速的搜索结果,但广告较多\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"搜索引擎的准确性和速度\\\"\\n x-axis \\\"准确性低\\\" --> \\\"准确性高\\\"\\n y-axis \\\"速度慢\\\" --> \\\"速度快\\\"\\n quadrant-1 \\\"需要改进\\\"\\n quadrant-2 \\\"需要提高速度\\\"\\n quadrant-3 \\\"需要提高准确性\\\"\\n quadrant-4 \\\"目标产品\\\"\\n \\\"百度搜索引擎\\\": [0.3, 0.6]\\n \\\"谷歌搜索引擎\\\": [0.45, 0.23]\\n \\\"搜狗搜索引擎\\\": [0.57, 0.69]\\n \\\"目标产品\\\": [0.8, 0.8]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"基于LLM算法实现搜索功能\"\n ],\n [\n \"P0\",\n \"提高搜索结果的准确性和相关性\"\n ]\n ],\n \"UI Design draft\": \"搜索界面设计简洁明了,提供关键词搜索框和搜索结果展示区域。\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "hello chatgpt": "Hello! How can I assist you today?", + "hello world": "Hello! How can I assist you today?", + "\n## context\n```\nclass UIDesign(Action):\n #Class representing the UI Design action.\n def __init__(self, name, context=None, llm=None):\n super().__init__(name, context, llm) # 需要调用LLM进一步丰富UI设计的prompt\n @parse\n def parse_requirement(self, context: str):\n #Parse UI Design draft from the context using regex.\n pattern = r\"## UI Design draft.*?\n(.*?)## Anything UNCLEAR\"\n return context, pattern\n @parse\n def parse_ui_elements(self, context: str):\n #Parse Selected Elements from the context using regex.\n pattern = r\"## Selected Elements.*?\n(.*?)## HTML Layout\"\n return context, pattern\n @parse\n def parse_css_code(self, context: str):\n pattern = r\"```css.*?\n(.*?)## Anything UNCLEAR\"\n return context, pattern\n @parse\n def parse_html_code(self, context: str):\n pattern = r\"```html.*?\n(.*?)```\"\n return context, pattern\n async def draw_icons(self, context, *args, **kwargs):\n #Draw icons using SDEngine.\n engine = SDEngine()\n icon_prompts = self.parse_ui_elements(context)\n icons = icon_prompts.split(\"\n\")\n icons = [s for s in icons if len(s.strip()) > 0]\n prompts_batch = []\n for icon_prompt in icons:\n # fixme: 添加icon lora\n prompt = engine.construct_payload(icon_prompt + \".\")\n prompts_batch.append(prompt)\n await engine.run_t2i(prompts_batch)\n logger.info(\"Finish icon design using StableDiffusion API\")\n async def _save(self, css_content, html_content):\n save_dir = CONFIG.workspace_path / \"resources\" / \"codes\"\n if not os.path.exists(save_dir):\n os.makedirs(save_dir, exist_ok=True)\n # Save CSS and HTML content to files\n css_file_path = save_dir / \"ui_design.css\"\n html_file_path = save_dir / \"ui_design.html\"\n with open(css_file_path, \"w\") as css_file:\n css_file.write(css_content)\n with open(html_file_path, \"w\") as html_file:\n html_file.write(html_content)\n async def run(self, requirements: list[Message], *args, **kwargs) -> ActionOutput:\n #Run the UI Design action.\n # fixme: update prompt (根据需求细化prompt)\n context = requirements[-1].content\n ui_design_draft = self.parse_requirement(context=context)\n # todo: parse requirements str\n prompt = PROMPT_TEMPLATE.format(context=ui_design_draft, format_example=FORMAT_EXAMPLE)\n logger.info(prompt)\n ui_describe = await self._aask_v1(prompt, \"ui_design\", OUTPUT_MAPPING)\n logger.info(ui_describe.content)\n logger.info(ui_describe.instruct_content)\n css = self.parse_css_code(context=ui_describe.content)\n html = self.parse_html_code(context=ui_describe.content)\n await self._save(css_content=css, html_content=html)\n await self.draw_icons(ui_describe.content)\n return ui_describe\n```\n-----\n## format example\n[CONTENT]\n{\n \"ClassView\": \"classDiagram\n class A {\n -int x\n +int y\n -int speed\n -int direction\n +__init__(x: int, y: int, speed: int, direction: int)\n +change_direction(new_direction: int) None\n +move() None\n }\n \"\n}\n[/CONTENT]\n## nodes: \": # \"\n- ClassView: # Generate the mermaid class diagram corresponding to source code in \"context.\"\n## constraint\n- Language: Please use the same language as the user input.\n- Format: output wrapped inside [CONTENT][/CONTENT] as format example, nothing else.\n## action\nFill in the above nodes(ClassView) based on the format example.\n": "ClassView: str # Generate the mermaid class diagram corresponding to source code in \"context.\"", + "\n## context\n\n### Project Name\n\n\n### Original Requirements\n['Make a cli snake game']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Make a cli snake game\",\n \"Project Name\": \"cli_snake_game\",\n \"Product Goals\": [\n \"Create an engaging and enjoyable snake game experience\",\n \"Implement smooth and responsive controls\",\n \"Include different difficulty levels\"\n ],\n \"User Stories\": [\n \"As a player, I want to control the snake using arrow keys\",\n \"As a player, I want to see my score increase as I eat food\",\n \"As a player, I want the game to end if the snake collides with itself or the boundaries\",\n \"As a player, I want to be able to choose between different difficulty levels\",\n \"As a player, I want to see a game over message when the game ends\"\n ],\n \"Competitive Analysis\": [\n \"Snake Game A: Simple interface, lacks difficulty levels\",\n \"Snake Game B: Responsive controls, but limited features\",\n \"Snake Game C: Multiple difficulty levels, but outdated UI\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Engagement and Features of Snake Games\\\"\\n x-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n y-axis \\\"Low Features\\\" --> \\\"High Features\\\"\\n quadrant-1 \\\"Improve Engagement & Features\\\"\\n quadrant-2 \\\"Improve Engagement\\\"\\n quadrant-3 \\\"Improve Features\\\"\\n quadrant-4 \\\"Satisfactory\\\"\\n \\\"Snake Game A\\\": [0.4, 0.2]\\n \\\"Snake Game B\\\": [0.6, 0.4]\\n \\\"Snake Game C\\\": [0.7, 0.6]\\n \\\"Our Snake Game\\\": [0.8, 0.8]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"Implement snake movement and collision detection\"\n ],\n [\n \"P0\",\n \"Generate food at random positions\"\n ],\n [\n \"P0\",\n \"Increase score when snake eats food\"\n ],\n [\n \"P1\",\n \"Implement game over condition\"\n ],\n [\n \"P1\",\n \"Allow player to choose difficulty level\"\n ]\n ],\n \"UI Design draft\": \"The game will be displayed in the command line interface (CLI). The snake and food will be represented by characters. The score and game over message will be displayed at the bottom of the screen.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\n## context\n{\"Language\":\"en_us\",\"Programming Language\":\"Python\",\"Original Requirements\":\"Make a cli snake game\",\"Project Name\":\"cli_snake_game\",\"Product Goals\":[\"Create an engaging and enjoyable snake game experience\",\"Implement smooth and responsive controls\",\"Include different difficulty levels\"],\"User Stories\":[\"As a player, I want to control the snake using arrow keys\",\"As a player, I want to see my score increase as I eat food\",\"As a player, I want the game to end if the snake collides with itself or the boundaries\",\"As a player, I want to be able to choose between different difficulty levels\",\"As a player, I want to see a game over message when the game ends\"],\"Competitive Analysis\":[\"Snake Game A: Simple interface, lacks difficulty levels\",\"Snake Game B: Responsive controls, but limited features\",\"Snake Game C: Multiple difficulty levels, but outdated UI\"],\"Competitive Quadrant Chart\":\"quadrantChart\\n title \\\"Engagement and Features of Snake Games\\\"\\n x-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n y-axis \\\"Low Features\\\" --> \\\"High Features\\\"\\n quadrant-1 \\\"Improve Engagement & Features\\\"\\n quadrant-2 \\\"Improve Engagement\\\"\\n quadrant-3 \\\"Improve Features\\\"\\n quadrant-4 \\\"Satisfactory\\\"\\n \\\"Snake Game A\\\": [0.4, 0.2]\\n \\\"Snake Game B\\\": [0.6, 0.4]\\n \\\"Snake Game C\\\": [0.7, 0.6]\\n \\\"Our Snake Game\\\": [0.8, 0.8]\",\"Requirement Analysis\":\"\",\"Requirement Pool\":[[\"P0\",\"Implement snake movement and collision detection\"],[\"P0\",\"Generate food at random positions\"],[\"P0\",\"Increase score when snake eats food\"],[\"P1\",\"Implement game over condition\"],[\"P1\",\"Allow player to choose difficulty level\"]],\"UI Design draft\":\"The game will be displayed in the command line interface (CLI). The snake and food will be represented by characters. The score and game over message will be displayed at the bottom of the screen.\",\"Anything UNCLEAR\":\"\"}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\n## context\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Required Python packages\": [\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"openapi: 3.0.0 ...\",\n \"Shared Knowledge\": \"'game.py' contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on how to start and initialize third-party libraries.\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Required Python packages: typing.List[str] # Provide required Python packages in requirements.txt format.\n- Required Other language third-party packages: typing.List[str] # List down the required packages for languages other than Python.\n- Logic Analysis: typing.List[typing.List[str]] # Provide a list of files with the classes/methods/functions to be implemented, including dependency analysis and imports.\n- Task list: typing.List[str] # Break down the tasks into a list of filenames, prioritized by dependency order.\n- Full API spec: # Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end and back-end communication is not required, leave it blank.\n- Shared Knowledge: # Detail any shared knowledge, like common utility functions or configuration variables.\n- Anything UNCLEAR: # Mention any unclear aspects in the project management context and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Required Python packages\": [\n \"python-dotenv==0.17.1\",\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"main.py\",\n \"Contains the main function to start the game\"\n ],\n [\n \"game.py\",\n \"Contains the Game class and related functions\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"\",\n \"Shared Knowledge\": \"\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n```\n\n\n\n# Format example 1\n## Code Review: game.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: game.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - No, the code is not implemented as per the requirements. The logic for moving the snake's body, changing the direction of the snake, checking collision, generating food, starting the game, updating the game state, ending the game, and changing the difficulty of the game is missing. To achieve the requirements, the logic for each of these functions needs to be implemented step by step.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - No, the code logic is not correct as the functions are not implemented. To correct the logic, each function needs to be implemented with the appropriate logic for the game.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, all functions are not implemented. The following steps can be followed to implement each function:\n - Snake.move(): Implement the logic to move the snake's body based on the current direction.\n - Snake.change_direction(): Implement the logic to change the direction of the snake.\n - Snake.check_collision(): Implement the logic to check if the snake has collided with itself or the boundaries of the game.\n - Food.generate_food(): Implement the logic to generate a new position for the food.\n - SnakeGame.start_game(): Implement the logic to start the game.\n - SnakeGame.update_game(): Implement the logic to update the game state.\n - SnakeGame.end_game(): Implement the logic to end the game.\n - SnakeGame.change_difficulty(): Implement the logic to change the difficulty of the game.\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, there are no pre-dependencies imported in the code.\n\n6. Are methods from other files being reused correctly?\n - No, there are no methods from other files being reused in the code.\n\n## Actions: Implement the logic for each function step by step.\n\n## Code Review Result: LBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n```\n\n## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - No, the code is not implemented as per the requirements. The logic for moving the snake's body, changing the direction of the snake, checking collision, generating food, starting the game, updating the game state, ending the game, and changing the difficulty of the game is missing. To achieve the requirements, you need to implement these logic steps in their respective methods.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - The code logic is not complete, as mentioned in the previous point. You need to implement the missing logic steps in their respective methods to ensure the code's correctness.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, not all functions are implemented. The following methods need to be implemented:\n - Snake.move(): Implement the logic to move the snake's body based on the current direction.\n - Snake.change_direction(direction: Position): Implement the logic to change the direction of the snake.\n - Snake.check_collision(): Implement the logic to check if the snake has collided with itself or the boundaries of the game.\n - Food.generate_food(): Implement the logic to generate a new position for the food.\n - SnakeGame.start_game(): Implement the logic to start the game.\n - SnakeGame.update_game(): Implement the logic to update the game state.\n - SnakeGame.end_game(): Implement the logic to end the game.\n - SnakeGame.change_difficulty(difficulty: int): Implement the logic to change the difficulty of the game.\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, there are no pre-dependencies required for this code.\n\n6. Are methods from other files being reused correctly?\n - No, there are no methods from other files being reused in this code.\n\n## Actions: Implement the missing logic steps in their respective methods.\n\n## Code Review Result: LBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite game.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## game.py\n...\n```\n": "```python\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # Implement the logic to move the snake's body based on the current direction\n head = self.body[0]\n new_head = Position(head.x + self.direction.x, head.y + self.direction.y)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction: Position):\n # Implement the logic to change the direction of the snake\n self.direction = direction\n\n def check_collision(self):\n # Implement the logic to check if the snake has collided with itself or the boundaries of the game\n head = self.body[0]\n if head.x < 0 or head.x >= GAME_WIDTH or head.y < 0 or head.y >= GAME_HEIGHT:\n return True\n for body_part in self.body[1:]:\n if head.x == body_part.x and head.y == body_part.y:\n return True\n return False\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # Implement the logic to generate a new position for the food\n self.position = Position(random.randint(0, GAME_WIDTH - 1), random.randint(0, GAME_HEIGHT - 1))\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # Implement the logic to start the game\n self.snake.body = [Position(0, 0)]\n self.snake.direction = Position(1, 0)\n self.food.generate_food()\n\n def update_game(self):\n # Implement the logic to update the game state\n self.snake.move()\n if self.snake.check_collision():\n self.end_game()\n elif self.snake.body[0].x == self.food.position.x and self.snake.body[0].y == self.food.position.y:\n self.score += 1\n self.snake.body.append(Position(0, 0))\n self.food.generate_food()\n\n def end_game(self):\n # Implement the logic to end the game\n print(\"Game Over\")\n print(\"Score:\", self.score)\n\n def change_difficulty(self, difficulty: int):\n # Implement the logic to change the difficulty of the game\n self.difficulty = difficulty\n```\n", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n```\n\n## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - No, the code is not implemented as per the requirements. The logic for moving the snake's body, changing the direction of the snake, checking collision, generating food, starting the game, updating the game state, ending the game, and changing the difficulty of the game is missing. To achieve the requirements, the logic for each of these functions needs to be implemented step by step.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - No, the code logic is not correct as the functions are not implemented. To correct the logic, each function needs to be implemented with the appropriate logic for the game.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, all functions are not implemented. The following steps can be followed to implement each function:\n - Snake.move(): Implement the logic to move the snake's body based on the current direction.\n - Snake.change_direction(): Implement the logic to change the direction of the snake.\n - Snake.check_collision(): Implement the logic to check if the snake has collided with itself or the boundaries of the game.\n - Food.generate_food(): Implement the logic to generate a new position for the food.\n - SnakeGame.start_game(): Implement the logic to start the game.\n - SnakeGame.update_game(): Implement the logic to update the game state.\n - SnakeGame.end_game(): Implement the logic to end the game.\n - SnakeGame.change_difficulty(): Implement the logic to change the difficulty of the game.\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, there are no pre-dependencies imported in the code.\n\n6. Are methods from other files being reused correctly?\n - No, there are no methods from other files being reused in the code.\n\n## Actions: Implement the logic for each function step by step.\n\n## Code Review Result: LBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite game.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## game.py\n...\n```\n": "```python\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n pass\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n pass\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n pass\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n pass\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n pass\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n pass\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n pass\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n pass\n```\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Legacy Code\n```Code\n----- game.py\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: main.py\n```python\n## main.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: main.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: main.py\n\n```python\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n```\n\nIn the `main.py` file, we import the `SnakeGame` class from the `game` module. We then create an instance of the `SnakeGame` class and call the `start_game` method to start the game. We then enter a loop where we continuously call the `update_game` method until the `end_game` method returns `True`, indicating that the game is over.", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n----- game.py\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n\n\n# Format example 1\n## Code Review: main.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: main.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - The code in `game.py` is not fully implemented as per the requirements. There are several TODO comments indicating where the logic needs to be implemented. To achieve the requirements, the following steps can be taken:\n - Implement the logic to move the snake's body based on the current direction in the `move` method of the `Snake` class.\n - Implement the logic to change the direction of the snake in the `change_direction` method of the `Snake` class.\n - Implement the logic to check if the snake has collided with itself or the boundaries of the game in the `check_collision` method of the `Snake` class.\n - Implement the logic to generate a new position for the food in the `generate_food` method of the `Food` class.\n - Implement the logic to start the game in the `start_game` method of the `SnakeGame` class.\n - Implement the logic to update the game state in the `update_game` method of the `SnakeGame` class.\n - Implement the logic to end the game in the `end_game` method of the `SnakeGame` class.\n - Implement the logic to change the difficulty of the game in the `change_difficulty` method of the `SnakeGame` class.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - Since the code is not fully implemented, it is not possible to determine if the logic is completely correct. The missing logic needs to be implemented to ensure the correctness of the code.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces. The classes `SnakeGame`, `Snake`, `Food`, and `Position` are defined as per the class diagram.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, not all functions are implemented. The following functions need to be implemented:\n - `move` method in the `Snake` class\n - `change_direction` method in the `Snake` class\n - `check_collision` method in the `Snake` class\n - `generate_food` method in the `Food` class\n - `start_game` method in the `SnakeGame` class\n - `update_game` method in the `SnakeGame` class\n - `end_game` method in the `SnakeGame` class\n - `change_difficulty` method in the `SnakeGame` class\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, the necessary pre-dependencies have not been imported in the code. The following packages need to be imported:\n - `from typing import List` in `game.py`\n\n6. Are methods from other files being reused correctly?\n - Yes, the methods from other files are being reused correctly. The `SnakeGame` class in `game.py` is imported and used correctly in `main.py`.\n\n## Actions: Implement the missing logic in the code.\n\n## Code Review Result: LBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n----- game.py\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n## Code Review: game.py\n\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n - No, the code is not implemented as per the requirements. The logic for moving the snake's body, changing the direction of the snake, and checking collision is missing. To achieve the requirements, you need to implement the missing logic in the respective methods.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - The code logic is not complete as the implementation is missing for the methods `move`, `change_direction`, and `check_collision`. You need to add the correct logic to these methods to ensure the snake moves, changes direction, and checks for collisions correctly.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, not all functions are implemented. The methods `move`, `change_direction`, `check_collision`, and `generate_food` are missing their respective logic. To implement them, you need to add the necessary code to perform the desired functionality.\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, there are no pre-dependencies required for the code in `game.py`.\n\n6. Are methods from other files being reused correctly?\n - No, there are no methods from other files being reused in `game.py`.\n\n## Actions: Implement the missing logic in the respective methods.\n\n## Code Review Result: LBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite main.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## main.py\n...\n```\n": "```python\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n```\n\n```python\n## game.py\n\nfrom typing import List\nimport random\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n new_head = Position(self.body[0].x + self.direction.x, self.body[0].y + self.direction.y)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction: Position):\n self.direction = direction\n\n def check_collision(self):\n head = self.body[0]\n if head.x < 0 or head.x >= 10 or head.y < 0 or head.y >= 10:\n return True\n for segment in self.body[1:]:\n if segment.x == head.x and segment.y == head.y:\n return True\n return False\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n self.position = Position(random.randint(0, 9), random.randint(0, 9))\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n self.snake.body = [Position(0, 0)]\n self.food.generate_food()\n\n def update_game(self):\n self.snake.move()\n if self.snake.check_collision():\n self.end_game()\n\n def end_game(self):\n print(\"Game Over\")\n print(\"Score:\", self.score)\n exit()\n\n def change_difficulty(self, difficulty: int):\n self.difficulty = difficulty\n```\n\nThe missing logic has been implemented in the respective methods. The snake can now move, change direction, and check for collisions correctly. The food is also generated at random positions.", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will implement the snake game using Python and the command line interface (CLI). We will analyze the difficult points of the requirements and select the appropriate open-source framework to assist with the game development.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class SnakeGame {\\n -int score\\n -int difficulty\\n -Snake snake\\n -Food food\\n +start_game()\\n +update_game()\\n +end_game()\\n +change_difficulty(difficulty: int)\\n }\\n class Snake {\\n -List[Position] body\\n -Position direction\\n +move()\\n +change_direction(direction: Position)\\n +check_collision()\\n }\\n class Food {\\n -Position position\\n +generate_food()\\n }\\n class Position {\\n -int x\\n -int y\\n }\\n SnakeGame --> Snake\\n SnakeGame --> Food\\n Snake --> Position\\n Food --> Position\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant G as SnakeGame\\n participant S as Snake\\n participant F as Food\\n participant P as Position\\n G->>S: start_game()\\n S->>F: generate_food()\\n F-->>S: return food\\n S->>G: update_game()\\n G->>S: move()\\n S->>S: check_collision()\\n S->>G: end_game()\\n G->>G: change_difficulty(difficulty)\\n G-->>S: return score\\n\",\"Anything UNCLEAR\":\"\"}\n\n## Tasks\n{\"Required Python packages\":[\"python-dotenv==0.17.1\",\"flask==1.1.2\",\"bcrypt==3.2.0\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"main.py\",\"Contains the main function to start the game\"],[\"game.py\",\"Contains the Game class and related functions\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"\",\"Anything UNCLEAR\":\"\"}\n\n## Code Files\n----- game.py\n## game.py\n\nfrom typing import List\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # TODO: Implement the logic to move the snake's body based on the current direction\n\n def change_direction(self, direction: Position):\n # TODO: Implement the logic to change the direction of the snake\n\n def check_collision(self):\n # TODO: Implement the logic to check if the snake has collided with itself or the boundaries of the game\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # TODO: Implement the logic to generate a new position for the food\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # TODO: Implement the logic to start the game\n\n def update_game(self):\n # TODO: Implement the logic to update the game state\n\n def end_game(self):\n # TODO: Implement the logic to end the game\n\n def change_difficulty(self, difficulty: int):\n # TODO: Implement the logic to change the difficulty of the game\n\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n## Code Review: game.py\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyze it step by step.\n - The code in `game.py` is not fully implemented as per the requirements. There are several TODO comments indicating where the logic needs to be implemented. To achieve the requirements, the following steps can be taken:\n - Implement the logic to move the snake's body based on the current direction in the `move` method of the `Snake` class.\n - Implement the logic to change the direction of the snake in the `change_direction` method of the `Snake` class.\n - Implement the logic to check if the snake has collided with itself or the boundaries of the game in the `check_collision` method of the `Snake` class.\n - Implement the logic to generate a new position for the food in the `generate_food` method of the `Food` class.\n - Implement the logic to start the game in the `start_game` method of the `SnakeGame` class.\n - Implement the logic to update the game state in the `update_game` method of the `SnakeGame` class.\n - Implement the logic to end the game in the `end_game` method of the `SnakeGame` class.\n - Implement the logic to change the difficulty of the game in the `change_difficulty` method of the `SnakeGame` class.\n\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n - Since the code is not fully implemented, it is not possible to determine if the logic is completely correct. The missing logic needs to be implemented to ensure the correctness of the code.\n\n3. Does the existing code follow the \"Data structures and interfaces\"?\n - Yes, the existing code follows the defined data structures and interfaces. The classes `SnakeGame`, `Snake`, `Food`, and `Position` are defined as per the class diagram.\n\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n - No, not all functions are implemented. The following functions need to be implemented:\n - `move` method in the `Snake` class\n - `change_direction` method in the `Snake` class\n - `check_collision` method in the `Snake` class\n - `generate_food` method in the `Food` class\n - `start_game` method in the `SnakeGame` class\n - `update_game` method in the `SnakeGame` class\n - `end_game` method in the `SnakeGame` class\n - `change_difficulty` method in the `SnakeGame` class\n\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported.\n - No, the necessary pre-dependencies have not been imported in the code. The following packages need to be imported:\n - `from typing import List` in `game.py`\n\n6. Are methods from other files being reused correctly?\n - Yes, the methods from other files are being reused correctly. The `SnakeGame` class in `game.py` is imported and used correctly in `main.py`.\n\n## Actions: Implement the missing logic in the code.\n\n## Code Review Result: LBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite main.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## main.py\n...\n```\n": "```python\n## main.py\n\nfrom game import SnakeGame\n\ndef main():\n game = SnakeGame()\n game.start_game()\n while True:\n game.update_game()\n if game.end_game():\n break\n\nif __name__ == \"__main__\":\n main()\n```\n\n```python\n## game.py\n\nfrom typing import List\nimport random\n\nclass Position:\n def __init__(self, x: int, y: int):\n self.x = x\n self.y = y\n\nclass Snake:\n def __init__(self):\n self.body = []\n self.direction = Position(0, 0)\n\n def move(self):\n # Update the position of the snake's body based on the current direction\n head = self.body[0]\n new_head = Position(head.x + self.direction.x, head.y + self.direction.y)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction: Position):\n # Change the direction of the snake\n self.direction = direction\n\n def check_collision(self):\n # Check if the snake has collided with itself or the boundaries of the game\n head = self.body[0]\n if head.x < 0 or head.x >= 10 or head.y < 0 or head.y >= 10:\n return True\n for i in range(1, len(self.body)):\n if self.body[i].x == head.x and self.body[i].y == head.y:\n return True\n return False\n\nclass Food:\n def __init__(self):\n self.position = Position(0, 0)\n\n def generate_food(self):\n # Generate a new position for the food\n self.position = Position(random.randint(0, 9), random.randint(0, 9))\n\nclass SnakeGame:\n def __init__(self):\n self.score = 0\n self.difficulty = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n # Start the game\n self.snake.body.append(Position(0, 0))\n self.food.generate_food()\n\n def update_game(self):\n # Update the game state\n self.snake.move()\n if self.snake.check_collision():\n self.end_game()\n elif self.snake.body[0].x == self.food.position.x and self.snake.body[0].y == self.food.position.y:\n self.score += 1\n self.snake.body.append(Position(0, 0))\n self.food.generate_food()\n\n def end_game(self):\n # End the game\n print(\"Game Over\")\n print(\"Score:\", self.score)\n\n def change_difficulty(self, difficulty: int):\n # Change the difficulty of the game\n self.difficulty = difficulty\n```\n```", + "## History Messages\n0: Human: Topic: climate change. Under 80 words per message.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.", + "## History Messages\n0: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.", + "## History Messages\n0: Bob(Republican candidate): I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.\n1: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n2: Human: Topic: climate change. Under 80 words per message.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "Bob: Climate change is a matter of utmost importance! We cannot ignore the urgency it demands. The potential consequences are truly alarming, and we must act now to protect our planet. Let's unite as a global community and take bold steps towards a sustainable future. Our children and future generations deserve nothing less!", + "## History Messages\n0: Alex(Democratic candidate): Bob: Climate change is a matter of utmost importance! We cannot ignore the urgency it demands. The potential consequences are truly alarming, and we must act now to protect our planet. Let's unite as a global community and take bold steps towards a sustainable future. Our children and future generations deserve nothing less!\n1: Bob(Republican candidate): I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.\n2: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "Alex(Democratic candidate): Bob, I am truly passionate about the urgency of addressing climate change. The potential consequences are alarming, and we cannot ignore them any longer. Our planet's well-being is at stake, and it is our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!", + "## History Messages\n0: Bob(Republican candidate): Alex(Democratic candidate): Bob, I am truly passionate about the urgency of addressing climate change. The potential consequences are alarming, and we cannot ignore them any longer. Our planet's well-being is at stake, and it is our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!\n1: Alex(Democratic candidate): Bob: Climate change is a matter of utmost importance! We cannot ignore the urgency it demands. The potential consequences are truly alarming, and we must act now to protect our planet. Let's unite as a global community and take bold steps towards a sustainable future. Our children and future generations deserve nothing less!\n2: Bob(Republican candidate): I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.\n3: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n4: Human: Topic: climate change. Under 80 words per message.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "Bob: Alex, I am genuinely alarmed by the potential consequences of climate change. We cannot ignore this urgent issue any longer! Our planet's well-being is at stake, and it's our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!", + "## History Messages\n0: Alex(Democratic candidate): Bob: Alex, I am genuinely alarmed by the potential consequences of climate change. We cannot ignore this urgent issue any longer! Our planet's well-being is at stake, and it's our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!\n1: Bob(Republican candidate): Alex(Democratic candidate): Bob, I am truly passionate about the urgency of addressing climate change. The potential consequences are alarming, and we cannot ignore them any longer. Our planet's well-being is at stake, and it is our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!\n2: Alex(Democratic candidate): Bob: Climate change is a matter of utmost importance! We cannot ignore the urgency it demands. The potential consequences are truly alarming, and we must act now to protect our planet. Let's unite as a global community and take bold steps towards a sustainable future. Our children and future generations deserve nothing less!\n3: Bob(Republican candidate): I completely agree with Alex's concern about climate change. It is indeed a pressing issue that requires urgent attention. The potential consequences of inaction are truly alarming, and we cannot afford to turn a blind eye to the warning signs any longer. Our planet's well-being is at stake, and it is crucial that we take bold and decisive action to address the challenges posed by climate change. Let us unite as a global community and strive towards a sustainable future for the sake of future generations.\n4: Alex(Democratic candidate): I am deeply concerned about climate change. It is a pressing issue that demands immediate attention. The consequences of inaction are dire, and we cannot afford to ignore the warning signs any longer. Our planet is at stake, and we must take bold and decisive action to mitigate the effects of climate change. Let us come together as a global community and work towards a sustainable future for generations to come.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "I share your deep concern about climate change. The potential consequences are truly alarming, and we cannot afford to ignore this urgent issue any longer. Our planet's well-being is at stake, and it is our responsibility to take immediate action. Let's unite as a global community and work towards a sustainable future for the sake of our children and future generations. This is a matter of utmost importance, and we must act now!", + "## History Messages\n0: user: Topic: climate change. Under 80 words per message.\n\n## Actions\nLanguage: Please use the same language as Human INPUT.\nExpress your opinion with emotion and don't repeat it": "1: Climate change is a pressing issue that demands immediate action. The consequences of inaction are dire, and we cannot afford to ignore the warnings any longer. Our planet is at stake, and it's time to prioritize sustainability and reduce our carbon footprint. Let's come together and fight for a better future for ourselves and future generations. #ActNow #SaveOurPlanet 💚🌍\n\n2: It breaks my heart to see the devastating effects of climate change. The rising sea levels, extreme weather events, and loss of biodiversity are all clear signs that we need to take action now. We owe it to our planet and future generations to make a change. Let's be responsible stewards of the Earth and work towards a sustainable and greener future. #ClimateAction #ProtectOurHome 🌱🌎\n\n3: Climate change is not just an environmental issue; it's a matter of social justice. The most vulnerable communities are disproportionately affected by its impacts. We cannot turn a blind eye to the injustice and inequality that climate change exacerbates. It's time to stand up for climate justice and ensure that everyone has equal access to a clean and safe environment. Together, we can create a more equitable and sustainable world. #ClimateJustice #EqualityForAll 🌍✊\n\n4: The science is clear: climate change is real and caused by human activities. It's frustrating to see the denial and skepticism surrounding this issue. We need to listen to the experts, trust the evidence, and take decisive action. Let's not allow ignorance and short-term interests to hinder our progress. We have the power to make a difference, so let's use it and combat climate change head-on. #ListenToScience #TakeAction 🌏🔬\n\n5: Climate change is not just a problem for future generations; it's affecting us right now. The heatwaves, droughts, and wildfires we're experiencing are all linked to climate change. We need to adapt to these changes and mitigate their impacts. It's time to prioritize renewable energy, sustainable agriculture, and conservation efforts. Our actions today will determine the world we leave behind for our children. Let's make it a better one. #ActNow #ForOurChildren 🌱🌍", + "\n## context\nwhat's the answer to 123+456?\n\n-----\n\n## format example\n[CONTENT]\n{\n \"reasoning\": \"\",\n \"answer\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- reasoning: # reasoning step by step\n- answer: # the final answer\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"reasoning\": \"To find the sum of 123 and 456, we add the corresponding digits together starting from the rightmost digit. \\n\\n3 + 6 = 9\\n2 + 5 = 7\\n1 + 4 = 5\\n\\nTherefore, the sum of 123 and 456 is 579.\",\n \"answer\": \"579\"\n}\n[/CONTENT]", + "\n## context\nwhat's the answer to 123+456?\n\n-----\n\n## format example\n[CONTENT]\n{\n \"reasoning\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- reasoning: # reasoning step by step\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"reasoning\": \"To find the sum of 123 and 456, we add the corresponding digits together. Starting from the rightmost digit, we have 3 + 6 = 9. Moving to the next digit, we have 2 + 5 = 7. Finally, adding the leftmost digits, we have 1 + 4 = 5. Therefore, the sum of 123 and 456 is 579.\"\n}\n[/CONTENT]", + "\n## context\nwhat's the answer to 123+456?\n\n-----\n\n## format example\n[CONTENT]\n{\n \"answer\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- answer: # the final answer\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"answer\": \"579\"\n}\n[/CONTENT]", + "\n## context\n我们需要一个音乐播放器,它应该有播放、暂停、上一曲、下一曲等功能。\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"We will use a Python open-source framework, such as Pygame or tkinter, to develop the music player. These frameworks provide built-in functions and classes for handling audio playback and user interface. We will analyze the difficult points of the requirements and select the framework that best meets our needs.\",\n \"File list\": [\n \"main.py\",\n \"music_player.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class MusicPlayer {\\n -current_song: Song\\n -playlist: List[Song]\\n +play()\\n +pause()\\n +next_song()\\n +previous_song()\\n }\\n class Song {\\n -title: str\\n -artist: str\\n -duration: int\\n +get_title() str\\n +get_artist() str\\n +get_duration() int\\n }\\n MusicPlayer --> Song\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant MP as MusicPlayer\\n participant S as Song\\n MP->>S: play()\\n S-->>MP: return\\n MP->>S: pause()\\n S-->>MP: return\\n MP->>S: next_song()\\n S-->>MP: return\\n MP->>S: previous_song()\\n S-->>MP: return\\n\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\n## context\n\n### Legacy Content\n{\"Implementation approach\":\"We will use a Python open-source framework, such as Pygame or tkinter, to develop the music player. These frameworks provide built-in functions and classes for handling audio playback and user interface. We will analyze the difficult points of the requirements and select the framework that best meets our needs.\",\"File list\":[\"main.py\",\"music_player.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class MusicPlayer {\\n -current_song: Song\\n -playlist: List[Song]\\n +play()\\n +pause()\\n +next_song()\\n +previous_song()\\n }\\n class Song {\\n -title: str\\n -artist: str\\n -duration: int\\n +get_title() str\\n +get_artist() str\\n +get_duration() int\\n }\\n MusicPlayer --> Song\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant MP as MusicPlayer\\n participant S as Song\\n MP->>S: play()\\n S-->>MP: return\\n MP->>S: pause()\\n S-->>MP: return\\n MP->>S: next_song()\\n S-->>MP: return\\n MP->>S: previous_song()\\n S-->>MP: return\\n\",\"Anything UNCLEAR\":\"\"}\n\n### New Requirements\n## Original Requirements\nThe original requirement is to create a game similar to the classic text-based adventure game, Zork.\n\n## Product Goals\n```python\nproduct_goals = [\n \"Create an engaging text-based adventure game\",\n \"Ensure the game is easy to navigate and user-friendly\",\n \"Incorporate compelling storytelling and puzzles\"\n]\n```\n\n## User Stories\n```python\nuser_stories = [\n \"As a player, I want to be able to easily input commands so that I can interact with the game world\",\n \"As a player, I want to explore various rooms and locations to uncover the game's story\",\n \"As a player, I want to solve puzzles to progress in the game\",\n \"As a player, I want to interact with various in-game objects to enhance my gameplay experience\",\n \"As a player, I want a game that challenges my problem-solving skills and keeps me engaged\"\n]\n```\n\n## Competitive Analysis\n```python\ncompetitive_analysis = [\n \"Zork: The original text-based adventure game with complex puzzles and engaging storytelling\",\n \"The Hitchhiker's Guide to the Galaxy: A text-based game with a unique sense of humor and challenging gameplay\",\n \"Colossal Cave Adventure: The first text adventure game which set the standard for the genre\",\n \"Quest: A platform that lets users create their own text adventure games\",\n \"ChatGPT: An AI that can generate text-based adventure games\",\n \"The Forest of Doom: A text-based game with a fantasy setting and multiple endings\",\n \"Wizards Choice: A text-based game with RPG elements and a focus on player choice\"\n]\n```\n\n## Competitive Quadrant Chart\n```mermaid\nquadrantChart\n title Reach and engagement of text-based adventure games\n x-axis Low Reach --> High Reach\n y-axis Low Engagement --> High Engagement\n quadrant-1 High potential games\n quadrant-2 Popular but less engaging games\n quadrant-3 Less popular and less engaging games\n quadrant-4 Popular and engaging games\n \"Zork\": [0.9, 0.8]\n \"Hitchhiker's Guide\": [0.7, 0.7]\n \"Colossal Cave Adventure\": [0.8, 0.6]\n \"Quest\": [0.4, 0.5]\n \"ChatGPT\": [0.3, 0.6]\n \"Forest of Doom\": [0.5, 0.4]\n \"Wizards Choice\": [0.6, 0.5]\n \"Our Target Product\": [0.5, 0.6]\n```\n\n## Requirement Analysis\nThe goal is to create a text-based adventure game similar to Zork. The game should be engaging, user-friendly, and feature compelling storytelling and puzzles. It should allow players to explore various rooms and locations, interact with in-game objects, and solve puzzles to progress. The game should also challenge players' problem-solving skills and keep them engaged.\n\n## Requirement Pool\n```python\nrequirement_pool = [\n (\"Design an intuitive command input system for player interactions\", \"P0\"),\n (\"Create a variety of rooms and locations for players to explore\", \"P0\"),\n (\"Develop engaging puzzles that players need to solve to progress\", \"P0\"),\n (\"Incorporate a compelling story that unfolds as players explore the game world\", \"P1\"),\n (\"Ensure the game is user-friendly and easy to navigate\", \"P1\")\n]\n```\n\n## Anything UNCLEAR\nThe original requirement did not specify the platform for the game (web, mobile, desktop) or any specific features or themes for the game's story and puzzles. More information on these aspects could help in further refining the product requirements and design.\n\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"We will use a Python open-source framework, such as Pygame or tkinter, to develop the text-based adventure game. These frameworks provide built-in functions and classes for handling user input, managing game state, and displaying text-based interfaces. We will analyze the difficult points of the requirements and select the framework that best meets our needs.\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Game {\\n -current_room: Room\\n -player: Player\\n +start()\\n +process_input(input: str)\\n +move(direction: str)\\n +interact(object: str)\\n }\\n class Room {\\n -name: str\\n -description: str\\n -exits: Dict[str, Room]\\n -objects: List[str]\\n +get_name() str\\n +get_description() str\\n +get_exits() Dict[str, Room]\\n +get_objects() List[str]\\n }\\n class Player {\\n -inventory: List[str]\\n +get_inventory() List[str]\\n +add_to_inventory(item: str)\\n +remove_from_inventory(item: str)\\n }\\n Game --> Room\\n Game --> Player\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant G as Game\\n participant R as Room\\n participant P as Player\\n G->>G: start()\\n G->>G: process_input(input)\\n G->>G: move(direction)\\n G->>R: get_exits()\\n R-->>G: return exits\\n G->>R: get_objects()\\n R-->>G: return objects\\n G->>P: get_inventory()\\n P-->>G: return inventory\\n G->>P: add_to_inventory(item)\\n P->>G: return\\n G->>P: remove_from_inventory(item)\\n P->>G: return\\n G-->>G: continue\\n\",\n \"Anything UNCLEAR\": \"The original requirement did not specify the platform for the game (web, mobile, desktop) or any specific features or themes for the game's story and puzzles. More information on these aspects could help in further refining the product requirements and design.\"\n}\n[/CONTENT]", + "Here is the Product Requirement Document (PRD):\n\n我们需要一个音乐播放器,它应该有播放、暂停、上一曲、下一曲等功能。\n\nHere is the list of APIs designed based on this PRD:\n\n\n数据结构:\n1. Song: 包含歌曲信息,如标题、艺术家等。\n2. Playlist: 包含一系列歌曲。\n\nAPI列表:\n1. play(song: Song): 开始播放指定的歌曲。\n2. pause(): 暂停当前播放的歌曲。\n3. next(): 跳到播放列表的下一首歌曲。\n4. previous(): 跳到播放列表的上一首歌曲。\n\n\nPlease review whether this API design meets the requirements of the PRD, and whether it complies with good design practices.": "Based on the provided PRD, the API design seems to meet the requirements. It includes the necessary functions such as play, pause, next, and previous, which are all mentioned in the PRD.\n\nIn terms of compliance with good design practices, the API design could be improved by considering the following suggestions:\n\n1. Use consistent naming conventions: It is recommended to use consistent naming conventions for API functions. For example, instead of using \"next\" and \"previous\", it would be better to use \"playNext\" and \"playPrevious\" to make the naming more intuitive and consistent.\n\n2. Error handling: It is important to consider error handling in the API design. For example, if the play function is called with an invalid song, the API should return an appropriate error response.\n\n3. Return values: It would be helpful to define the expected return values for each API function. For example, the play function could return a boolean value indicating whether the song started playing successfully.\n\n4. Additional functionalities: Depending on the requirements, it might be beneficial to include additional functionalities in the API design. For example, adding a function to create or modify playlists could enhance the overall user experience.\n\nOverall, the provided API design meets the requirements of the PRD, but there are some areas where it could be further improved to align with good design practices.", + "\n## context\n\n## topic\n如何做一个生日蛋糕\n\n## record\n我认为应该先准备好材料,然后再开始做蛋糕。\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Questions\": [\n \"1. What ...\",\n \"2. How ...\",\n \"3. ...\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Questions: list[str] # Task: Refer to the context to further inquire about the details that interest you, within a word limit of 150 words. Please provide the specific details you would like to inquire about here\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Questions\": [\n \"1. 做生日蛋糕需要准备哪些材料?\",\n \"2. 做生日蛋糕的步骤是什么?\",\n \"3. 你有什么建议或技巧可以分享吗?\"\n ]\n}\n[/CONTENT]", + "Now I will provide you with the OCR text recognition results for the invoice.\nPlease extract the payee, city, total cost, and invoicing date of the invoice.\n\nThe OCR data of the invoice are as follows:\n[[[[[391.0, 43.0], [801.0, 43.0], [801.0, 81.0], [391.0, 81.0]], ('某地增值税电子普通发票', 1.0)], [[[844.0, 45.0], [1028.0, 45.0], [1028.0, 62.0], [844.0, 62.0]], ('发票代码:00100210001', 1.0)], [[[842.0, 73.0], [917.0, 73.0], [917.0, 94.0], [842.0, 94.0]], ('发票号码:', 1.0)], [[[924.0, 76.0], [1004.0, 76.0], [1004.0, 93.0], [924.0, 93.0]], ('07099363', 1.0)], [[[842.0, 107.0], [919.0, 107.0], [919.0, 124.0], [842.0, 124.0]], ('开票日期:', 1.0)], [[[930.0, 107.0], [1056.0, 107.0], [1056.0, 124.0], [930.0, 124.0]], ('2023年02月03日', 1.0)], [[[30.0, 141.0], [104.0, 141.0], [104.0, 163.0], [30.0, 163.0]], ('机器编号:', 1.0)], [[[124.0, 143.0], [236.0, 143.0], [236.0, 160.0], [124.0, 160.0]], ('499090000000', 1.0)], [[[842.0, 138.0], [1139.0, 138.0], [1139.0, 155.0], [842.0, 155.0]], ('校验码:10014320023319800000', 1.0)], [[[38.0, 187.0], [61.0, 187.0], [61.0, 208.0], [38.0, 208.0]], ('购', 1.0)], [[[77.0, 187.0], [96.0, 187.0], [96.0, 206.0], [77.0, 206.0]], ('名', 1.0)], [[[164.0, 186.0], [192.0, 186.0], [192.0, 206.0], [164.0, 206.0]], ('称:', 1.0)], [[[210.0, 185.0], [373.0, 185.0], [373.0, 206.0], [210.0, 206.0]], ('北京A科技有限公司', 1.0)], [[[686.0, 191.0], [698.0, 191.0], [698.0, 205.0], [686.0, 205.0]], ('密', 0.55)], [[[717.0, 190.0], [1162.0, 190.0], [1162.0, 207.0], [717.0, 207.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.99)], [[[76.0, 213.0], [192.0, 213.0], [192.0, 236.0], [76.0, 236.0]], ('纳税人识别号:', 1.0)], [[[212.0, 216.0], [414.0, 216.0], [414.0, 233.0], [212.0, 233.0]], ('91011111AA2AAAAA00', 1.0)], [[[715.0, 212.0], [1146.0, 213.0], [1146.0, 235.0], [715.0, 233.0]], ('07-*123<><>8000087*<64>4<8*,', 0.96)], [[[38.0, 223.0], [60.0, 223.0], [60.0, 246.0], [38.0, 246.0]], ('买', 1.0)], [[[682.0, 222.0], [701.0, 222.0], [701.0, 241.0], [682.0, 241.0]], ('码', 1.0)], [[[74.0, 239.0], [195.0, 242.0], [194.0, 267.0], [73.0, 264.0]], ('地址电话:', 0.98)], [[[715.0, 239.0], [1150.0, 239.0], [1150.0, 261.0], [715.0, 261.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[38.0, 258.0], [60.0, 258.0], [60.0, 282.0], [38.0, 282.0]], ('方', 1.0)], [[[74.0, 272.0], [194.0, 272.0], [194.0, 294.0], [74.0, 294.0]], ('开户行及账号:', 1.0)], [[[713.0, 263.0], [1153.0, 266.0], [1152.0, 287.0], [713.0, 284.0]], ('24-004*96-012>9819<<>97>>000', 1.0)], [[[65.0, 303.0], [283.0, 303.0], [283.0, 328.0], [65.0, 328.0]], ('货物或应税劳务、服务名称', 1.0)], [[[360.0, 299.0], [435.0, 299.0], [435.0, 321.0], [360.0, 321.0]], ('规格型号', 1.0)], [[[483.0, 299.0], [525.0, 299.0], [525.0, 323.0], [483.0, 323.0]], ('单位', 1.0)], [[[561.0, 299.0], [620.0, 299.0], [620.0, 323.0], [561.0, 323.0]], ('数量', 1.0)], [[[682.0, 299.0], [734.0, 299.0], [734.0, 323.0], [682.0, 323.0]], ('单价', 1.0)], [[[855.0, 301.0], [880.0, 301.0], [880.0, 321.0], [855.0, 321.0]], ('额', 1.0)], [[[942.0, 299.0], [986.0, 299.0], [986.0, 323.0], [942.0, 323.0]], ('税率', 1.0)], [[[1058.0, 301.0], [1084.0, 301.0], [1084.0, 321.0], [1058.0, 321.0]], ('税', 1.0)], [[[1093.0, 301.0], [1119.0, 301.0], [1119.0, 321.0], [1093.0, 321.0]], ('额', 1.0)], [[[30.0, 330.0], [200.0, 330.0], [200.0, 351.0], [30.0, 351.0]], ('餐饮服务*餐饮服务', 1.0)], [[[627.0, 328.0], [643.0, 328.0], [643.0, 346.0], [627.0, 346.0]], ('1', 1.0)], [[[692.0, 330.0], [752.0, 330.0], [752.0, 349.0], [692.0, 349.0]], ('379.25', 1.0)], [[[861.0, 329.0], [922.0, 329.0], [922.0, 351.0], [861.0, 351.0]], ('379.25', 1.0)], [[[968.0, 325.0], [999.0, 325.0], [999.0, 346.0], [968.0, 346.0]], ('6%', 1.0)], [[[1104.0, 329.0], [1158.0, 329.0], [1158.0, 351.0], [1104.0, 351.0]], ('22.75', 1.0)], [[[27.0, 357.0], [221.0, 357.0], [221.0, 378.0], [27.0, 378.0]], ('*日用杂品*灵感保温袋', 1.0)], [[[627.0, 351.0], [643.0, 351.0], [643.0, 372.0], [627.0, 372.0]], ('1', 1.0)], [[[710.0, 355.0], [751.0, 355.0], [751.0, 373.0], [710.0, 373.0]], ('8.85', 1.0)], [[[880.0, 354.0], [923.0, 354.0], [923.0, 376.0], [880.0, 376.0]], ('8.85', 1.0)], [[[957.0, 354.0], [1000.0, 354.0], [1000.0, 376.0], [957.0, 376.0]], ('13%', 0.96)], [[[1117.0, 351.0], [1159.0, 351.0], [1159.0, 375.0], [1117.0, 375.0]], ('1.15', 1.0)], [[[853.0, 526.0], [926.0, 529.0], [925.0, 551.0], [852.0, 548.0]], ('¥388.10', 0.94)], [[[128.0, 536.0], [153.0, 536.0], [153.0, 557.0], [128.0, 557.0]], ('合', 1.0)], [[[184.0, 536.0], [213.0, 536.0], [213.0, 557.0], [184.0, 557.0]], ('计', 1.0)], [[[1097.0, 529.0], [1160.0, 529.0], [1160.0, 551.0], [1097.0, 551.0]], ('¥23.90', 0.93)], [[[97.0, 564.0], [223.0, 564.0], [223.0, 589.0], [97.0, 589.0]], ('价税合计 (大写)', 1.0)], [[[329.0, 562.0], [498.0, 566.0], [497.0, 591.0], [329.0, 587.0]], ('肆佰壹拾贰圆整', 1.0)], [[[869.0, 563.0], [1005.0, 566.0], [1005.0, 588.0], [868.0, 585.0]], ('(小写)¥412.00', 0.96)], [[[38.0, 610.0], [61.0, 610.0], [61.0, 634.0], [38.0, 634.0]], ('销', 1.0)], [[[77.0, 604.0], [94.0, 604.0], [94.0, 623.0], [77.0, 623.0]], ('名', 1.0)], [[[155.0, 603.0], [406.0, 604.0], [406.0, 625.0], [155.0, 624.0]], ('称:深圳蛋糕餐饮有限公司', 1.0)], [[[681.0, 617.0], [703.0, 617.0], [703.0, 641.0], [681.0, 641.0]], ('备', 1.0)], [[[78.0, 629.0], [365.0, 629.0], [365.0, 646.0], [78.0, 646.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[40.0, 649.0], [58.0, 649.0], [58.0, 667.0], [40.0, 667.0]], ('售', 1.0)], [[[74.0, 650.0], [438.0, 651.0], [438.0, 676.0], [74.0, 675.0]], ('地址、电话:深圳市南山区成功大厦B座', 1.0)], [[[76.0, 674.0], [360.0, 675.0], [360.0, 697.0], [76.0, 696.0]], ('开户行及账号:中国银行深圳支行', 1.0)], [[[681.0, 672.0], [703.0, 672.0], [703.0, 695.0], [681.0, 695.0]], ('注', 1.0)], [[[41.0, 685.0], [57.0, 685.0], [57.0, 702.0], [41.0, 702.0]], ('方', 1.0)], [[[38.0, 717.0], [174.0, 717.0], [174.0, 738.0], [38.0, 738.0]], ('收款人:小明', 1.0)], [[[361.0, 718.0], [484.0, 718.0], [484.0, 739.0], [361.0, 739.0]], ('复核:小蔡', 1.0)], [[[597.0, 718.0], [682.0, 718.0], [682.0, 739.0], [597.0, 739.0]], ('开票人:', 0.99)], [[[707.0, 717.0], [752.0, 717.0], [752.0, 741.0], [707.0, 741.0]], ('小红', 1.0)], [[[870.0, 712.0], [1000.0, 712.0], [1000.0, 733.0], [870.0, 733.0]], ('销售方: (章)', 0.99)]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. The total cost refers to the total price and tax. Do not include `¥`.\n2. The city must be the recipient's city.\n2. The returned JSON dictionary must be returned in ch\n3. Mandatory requirement to output in JSON format: {\"收款人\":\"x\",\"城市\":\"x\",\"总费用/元\":\"\",\"开票日期\":\"\"}.\n": "Based on the OCR data provided, the extracted information from the invoice is as follows:\n\n- Payee: 小明 (收款人)\n- City: 深圳市 (城市)\n- Total cost: 412.00 (总费用/元)\n- Invoicing date: 2023年02月03日 (开票日期)\n\nHere is the information in JSON format:\n{\n \"收款人\": \"小明\",\n \"城市\": \"深圳市\",\n \"总费用/元\": \"412.00\",\n \"开票日期\": \"2023年02月03日\"\n}", + "Now I will provide you with the OCR text recognition results for the invoice.\nPlease answer the question: Invoicing date\n\nThe OCR data of the invoice are as follows:\n[[[[[[391.0, 43.0], [801.0, 43.0], [801.0, 81.0], [391.0, 81.0]], ('某地增值税电子普通发票', 1.0)], [[[844.0, 45.0], [1028.0, 45.0], [1028.0, 62.0], [844.0, 62.0]], ('发票代码:00100210001', 1.0)], [[[842.0, 73.0], [917.0, 73.0], [917.0, 94.0], [842.0, 94.0]], ('发票号码:', 1.0)], [[[924.0, 76.0], [1004.0, 76.0], [1004.0, 93.0], [924.0, 93.0]], ('07099363', 1.0)], [[[842.0, 107.0], [919.0, 107.0], [919.0, 124.0], [842.0, 124.0]], ('开票日期:', 1.0)], [[[930.0, 107.0], [1056.0, 107.0], [1056.0, 124.0], [930.0, 124.0]], ('2023年02月03日', 1.0)], [[[30.0, 141.0], [104.0, 141.0], [104.0, 163.0], [30.0, 163.0]], ('机器编号:', 1.0)], [[[124.0, 143.0], [236.0, 143.0], [236.0, 160.0], [124.0, 160.0]], ('499090000000', 1.0)], [[[842.0, 138.0], [1139.0, 138.0], [1139.0, 155.0], [842.0, 155.0]], ('校验码:10014320023319800000', 1.0)], [[[38.0, 187.0], [61.0, 187.0], [61.0, 208.0], [38.0, 208.0]], ('购', 1.0)], [[[77.0, 187.0], [96.0, 187.0], [96.0, 206.0], [77.0, 206.0]], ('名', 1.0)], [[[164.0, 186.0], [192.0, 186.0], [192.0, 206.0], [164.0, 206.0]], ('称:', 1.0)], [[[210.0, 185.0], [373.0, 185.0], [373.0, 206.0], [210.0, 206.0]], ('北京A科技有限公司', 1.0)], [[[686.0, 191.0], [698.0, 191.0], [698.0, 205.0], [686.0, 205.0]], ('密', 0.55)], [[[717.0, 190.0], [1162.0, 190.0], [1162.0, 207.0], [717.0, 207.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.99)], [[[76.0, 213.0], [192.0, 213.0], [192.0, 236.0], [76.0, 236.0]], ('纳税人识别号:', 1.0)], [[[212.0, 216.0], [414.0, 216.0], [414.0, 233.0], [212.0, 233.0]], ('91011111AA2AAAAA00', 1.0)], [[[715.0, 212.0], [1146.0, 213.0], [1146.0, 235.0], [715.0, 233.0]], ('07-*123<><>8000087*<64>4<8*,', 0.96)], [[[38.0, 223.0], [60.0, 223.0], [60.0, 246.0], [38.0, 246.0]], ('买', 1.0)], [[[682.0, 222.0], [701.0, 222.0], [701.0, 241.0], [682.0, 241.0]], ('码', 1.0)], [[[74.0, 239.0], [195.0, 242.0], [194.0, 267.0], [73.0, 264.0]], ('地址电话:', 0.98)], [[[715.0, 239.0], [1150.0, 239.0], [1150.0, 261.0], [715.0, 261.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[38.0, 258.0], [60.0, 258.0], [60.0, 282.0], [38.0, 282.0]], ('方', 1.0)], [[[74.0, 272.0], [194.0, 272.0], [194.0, 294.0], [74.0, 294.0]], ('开户行及账号:', 1.0)], [[[713.0, 263.0], [1153.0, 266.0], [1152.0, 287.0], [713.0, 284.0]], ('24-004*96-012>9819<<>97>>000', 1.0)], [[[65.0, 303.0], [283.0, 303.0], [283.0, 328.0], [65.0, 328.0]], ('货物或应税劳务、服务名称', 1.0)], [[[360.0, 299.0], [435.0, 299.0], [435.0, 321.0], [360.0, 321.0]], ('规格型号', 1.0)], [[[483.0, 299.0], [525.0, 299.0], [525.0, 323.0], [483.0, 323.0]], ('单位', 1.0)], [[[561.0, 299.0], [620.0, 299.0], [620.0, 323.0], [561.0, 323.0]], ('数量', 1.0)], [[[682.0, 299.0], [734.0, 299.0], [734.0, 323.0], [682.0, 323.0]], ('单价', 1.0)], [[[855.0, 301.0], [880.0, 301.0], [880.0, 321.0], [855.0, 321.0]], ('额', 1.0)], [[[942.0, 299.0], [986.0, 299.0], [986.0, 323.0], [942.0, 323.0]], ('税率', 1.0)], [[[1058.0, 301.0], [1084.0, 301.0], [1084.0, 321.0], [1058.0, 321.0]], ('税', 1.0)], [[[1093.0, 301.0], [1119.0, 301.0], [1119.0, 321.0], [1093.0, 321.0]], ('额', 1.0)], [[[30.0, 330.0], [200.0, 330.0], [200.0, 351.0], [30.0, 351.0]], ('餐饮服务*餐饮服务', 1.0)], [[[627.0, 328.0], [643.0, 328.0], [643.0, 346.0], [627.0, 346.0]], ('1', 1.0)], [[[692.0, 330.0], [752.0, 330.0], [752.0, 349.0], [692.0, 349.0]], ('379.25', 1.0)], [[[861.0, 329.0], [922.0, 329.0], [922.0, 351.0], [861.0, 351.0]], ('379.25', 1.0)], [[[968.0, 325.0], [999.0, 325.0], [999.0, 346.0], [968.0, 346.0]], ('6%', 1.0)], [[[1104.0, 329.0], [1158.0, 329.0], [1158.0, 351.0], [1104.0, 351.0]], ('22.75', 1.0)], [[[27.0, 357.0], [221.0, 357.0], [221.0, 378.0], [27.0, 378.0]], ('*日用杂品*灵感保温袋', 1.0)], [[[627.0, 351.0], [643.0, 351.0], [643.0, 372.0], [627.0, 372.0]], ('1', 1.0)], [[[710.0, 355.0], [751.0, 355.0], [751.0, 373.0], [710.0, 373.0]], ('8.85', 1.0)], [[[880.0, 354.0], [923.0, 354.0], [923.0, 376.0], [880.0, 376.0]], ('8.85', 1.0)], [[[957.0, 354.0], [1000.0, 354.0], [1000.0, 376.0], [957.0, 376.0]], ('13%', 0.96)], [[[1117.0, 351.0], [1159.0, 351.0], [1159.0, 375.0], [1117.0, 375.0]], ('1.15', 1.0)], [[[853.0, 526.0], [926.0, 529.0], [925.0, 551.0], [852.0, 548.0]], ('¥388.10', 0.94)], [[[128.0, 536.0], [153.0, 536.0], [153.0, 557.0], [128.0, 557.0]], ('合', 1.0)], [[[184.0, 536.0], [213.0, 536.0], [213.0, 557.0], [184.0, 557.0]], ('计', 1.0)], [[[1097.0, 529.0], [1160.0, 529.0], [1160.0, 551.0], [1097.0, 551.0]], ('¥23.90', 0.93)], [[[97.0, 564.0], [223.0, 564.0], [223.0, 589.0], [97.0, 589.0]], ('价税合计 (大写)', 1.0)], [[[329.0, 562.0], [498.0, 566.0], [497.0, 591.0], [329.0, 587.0]], ('肆佰壹拾贰圆整', 1.0)], [[[869.0, 563.0], [1005.0, 566.0], [1005.0, 588.0], [868.0, 585.0]], ('(小写)¥412.00', 0.96)], [[[38.0, 610.0], [61.0, 610.0], [61.0, 634.0], [38.0, 634.0]], ('销', 1.0)], [[[77.0, 604.0], [94.0, 604.0], [94.0, 623.0], [77.0, 623.0]], ('名', 1.0)], [[[155.0, 603.0], [406.0, 604.0], [406.0, 625.0], [155.0, 624.0]], ('称:深圳蛋糕餐饮有限公司', 1.0)], [[[681.0, 617.0], [703.0, 617.0], [703.0, 641.0], [681.0, 641.0]], ('备', 1.0)], [[[78.0, 629.0], [365.0, 629.0], [365.0, 646.0], [78.0, 646.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[40.0, 649.0], [58.0, 649.0], [58.0, 667.0], [40.0, 667.0]], ('售', 1.0)], [[[74.0, 650.0], [438.0, 651.0], [438.0, 676.0], [74.0, 675.0]], ('地址、电话:深圳市南山区成功大厦B座', 1.0)], [[[76.0, 674.0], [360.0, 675.0], [360.0, 697.0], [76.0, 696.0]], ('开户行及账号:中国银行深圳支行', 1.0)], [[[681.0, 672.0], [703.0, 672.0], [703.0, 695.0], [681.0, 695.0]], ('注', 1.0)], [[[41.0, 685.0], [57.0, 685.0], [57.0, 702.0], [41.0, 702.0]], ('方', 1.0)], [[[38.0, 717.0], [174.0, 717.0], [174.0, 738.0], [38.0, 738.0]], ('收款人:小明', 1.0)], [[[361.0, 718.0], [484.0, 718.0], [484.0, 739.0], [361.0, 739.0]], ('复核:小蔡', 1.0)], [[[597.0, 718.0], [682.0, 718.0], [682.0, 739.0], [597.0, 739.0]], ('开票人:', 0.99)], [[[707.0, 717.0], [752.0, 717.0], [752.0, 741.0], [707.0, 741.0]], ('小红', 1.0)], [[[870.0, 712.0], [1000.0, 712.0], [1000.0, 733.0], [870.0, 733.0]], ('销售方: (章)', 0.99)]]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. Answer in ch language.\n2. Enforce restrictions on not returning OCR data sent to you.\n3. Return with markdown syntax layout.\n": "The invoicing date on the invoice is **2023年02月03日**.", + "\n## context\nI just graduated and hope to find a job as a Python engineer\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Questions\": [\n \"1. What ...\",\n \"2. How ...\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Questions: list[str] # Role: You are an interviewer of our company who is well-knonwn in frontend or backend develop;\nRequirement: Provide a list of questions for the interviewer to ask the interviewee, by reading the resume of the interviewee in the context.\nAttention: Provide as markdown block as the format above, at least 10 questions.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Questions\": [\n \"1. Can you tell me about your experience with Python programming?\",\n \"2. Have you worked on any projects using Python? If so, can you describe one of them?\",\n \"3. What frameworks or libraries are you familiar with in Python?\",\n \"4. How do you handle errors and exceptions in Python?\",\n \"5. Can you explain the concept of generators in Python?\",\n \"6. How do you manage dependencies in Python projects?\",\n \"7. Have you used any testing frameworks in Python? If yes, which ones?\",\n \"8. Can you explain the difference between list comprehension and generator expression in Python?\",\n \"9. How do you optimize the performance of Python code?\",\n \"10. Can you describe a situation where you had to debug a complex Python program? How did you approach it?\"\n ]\n}\n[/CONTENT]", + "\n## context\n{'Implementation approach': '我们将使用Python编程语言,并选择合适的开源框架来实现贪吃蛇游戏。我们将分析需求中的难点,并选择合适的开源框架来简化开发流程。', 'File list': ['main.py', 'game.py'], 'Data structures and interfaces': '\\nclassDiagram\\n class Game {\\n -int width\\n -int height\\n -int score\\n -int speed\\n -List snake\\n -Point food\\n +__init__(width: int, height: int, speed: int)\\n +start_game()\\n +change_direction(direction: str)\\n +game_over()\\n +update_snake()\\n +update_food()\\n +check_collision()\\n }\\n class Point {\\n -int x\\n -int y\\n +__init__(x: int, y: int)\\n }\\n Game --> Point\\n', 'Program call flow': '\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: start_game()\\n M->>G: change_direction(direction)\\n G->>G: update_snake()\\n G->>G: update_food()\\n G->>G: check_collision()\\n G-->>G: game_over()\\n', 'Anything UNCLEAR': ''}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Required Python packages\": [\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"openapi: 3.0.0 ...\",\n \"Shared Knowledge\": \"'game.py' contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on how to start and initialize third-party libraries.\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Required Python packages: typing.List[str] # Provide required Python packages in requirements.txt format.\n- Required Other language third-party packages: typing.List[str] # List down the required packages for languages other than Python.\n- Logic Analysis: typing.List[typing.List[str]] # Provide a list of files with the classes/methods/functions to be implemented, including dependency analysis and imports.\n- Task list: typing.List[str] # Break down the tasks into a list of filenames, prioritized by dependency order.\n- Full API spec: # Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end and back-end communication is not required, leave it blank.\n- Shared Knowledge: # Detail any shared knowledge, like common utility functions or configuration variables.\n- Anything UNCLEAR: # Mention any unclear aspects in the project management context and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Required Python packages\": [\n \"pygame==2.0.1\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and related functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, imports Game class from game.py\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"\",\n \"Shared Knowledge\": \"'game.py' contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\nimport asyncio\nfrom pathlib import Path\n\nimport typer\n\nfrom metagpt.config import CONFIG\n\napp = typer.Typer(add_completion=False)\n\n\n@app.command()\ndef startup(\n idea: str = typer.Argument(..., help=\"Your innovative idea, such as 'Create a 2048 game.'\"),\n investment: float = typer.Option(default=3.0, help=\"Dollar amount to invest in the AI company.\"),\n n_round: int = typer.Option(default=5, help=\"Number of rounds for the simulation.\"),\n code_review: bool = typer.Option(default=True, help=\"Whether to use code review.\"),\n run_tests: bool = typer.Option(default=False, help=\"Whether to enable QA for adding & running tests.\"),\n implement: bool = typer.Option(default=True, help=\"Enable or disable code implementation.\"),\n project_name: str = typer.Option(default=\"\", help=\"Unique project name, such as 'game_2048'.\"),\n inc: bool = typer.Option(default=False, help=\"Incremental mode. Use it to coop with existing repo.\"),\n project_path: str = typer.Option(\n default=\"\",\n help=\"Specify the directory path of the old version project to fulfill the incremental requirements.\",\n ),\n reqa_file: str = typer.Option(\n default=\"\", help=\"Specify the source file name for rewriting the quality assurance code.\"\n ),\n max_auto_summarize_code: int = typer.Option(\n default=0,\n help=\"The maximum number of times the 'SummarizeCode' action is automatically invoked, with -1 indicating \"\n \"unlimited. This parameter is used for debugging the workflow.\",\n ),\n recover_path: str = typer.Option(default=None, help=\"recover the project from existing serialized storage\"),\n):\n \"\"\"Run a startup. Be a boss.\"\"\"\n from metagpt.roles import (\n Architect,\n Engineer,\n ProductManager,\n ProjectManager,\n QaEngineer,\n )\n from metagpt.team import Team\n\n CONFIG.update_via_cli(project_path, project_name, inc, reqa_file, max_auto_summarize_code)\n\n if not recover_path:\n company = Team()\n company.hire(\n [\n ProductManager(),\n Architect(),\n ProjectManager(),\n ]\n )\n\n if implement or code_review:\n company.hire([Engineer(n_borg=5, use_code_review=code_review)])\n\n if run_tests:\n company.hire([QaEngineer()])\n else:\n # # stg_path = SERDESER_PATH.joinpath(\"team\")\n stg_path = Path(recover_path)\n if not stg_path.exists() or not str(stg_path).endswith(\"team\"):\n raise FileNotFoundError(f\"{recover_path} not exists or not endswith `team`\")\n\n company = Team.deserialize(stg_path=stg_path)\n idea = company.idea # use original idea\n\n company.invest(investment)\n company.run_project(idea)\n asyncio.run(company.run(n_round=n_round))\n\n\nif __name__ == \"__main__\":\n app()\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "The Mermaid Sequence Diagram for the given code is as follows:\n\n```mermaid\nsequenceDiagram\n participant User\n participant Typer\n participant Team\n participant ProductManager\n participant Architect\n participant ProjectManager\n participant Engineer\n participant QaEngineer\n\n User ->> Typer: Run startup command\n Typer ->> Team: Create Team instance\n Team ->> Team: Hire ProductManager, Architect, ProjectManager\n Team ->> Team: Hire Engineer (if implement or code_review is True)\n Team ->> Team: Hire QaEngineer (if run_tests is True)\n User ->> Team: Set project_path, project_name, inc, reqa_file, max_auto_summarize_code\n Team ->> Team: Update CONFIG with CLI arguments\n Team ->> Team: Invest in the company\n Team ->> Team: Run project with the given idea\n Team ->> Team: Run simulation for n_rounds\n\n```\n\nNote: The diagram represents the sequence of interactions between different participants (User, Typer, Team, ProductManager, Architect, ProjectManager, Engineer, QaEngineer) in the code.", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n\nfrom __future__ import annotations\n\nimport asyncio\nimport json\nfrom concurrent import futures\nfrom typing import Literal, overload\n\ntry:\n from duckduckgo_search import DDGS\nexcept ImportError:\n raise ImportError(\n \"To use this module, you should have the `duckduckgo_search` Python package installed. \"\n \"You can install it by running the command: `pip install -e.[search-ddg]`\"\n )\n\nfrom metagpt.config import CONFIG\n\n\nclass DDGAPIWrapper:\n \"\"\"Wrapper around duckduckgo_search API.\n\n To use this module, you should have the `duckduckgo_search` Python package installed.\n \"\"\"\n\n def __init__(\n self,\n *,\n loop: asyncio.AbstractEventLoop | None = None,\n executor: futures.Executor | None = None,\n ):\n kwargs = {}\n if CONFIG.global_proxy:\n kwargs[\"proxies\"] = CONFIG.global_proxy\n self.loop = loop\n self.executor = executor\n self.ddgs = DDGS(**kwargs)\n\n @overload\n def run(\n self,\n query: str,\n max_results: int = 8,\n as_string: Literal[True] = True,\n focus: list[str] | None = None,\n ) -> str:\n ...\n\n @overload\n def run(\n self,\n query: str,\n max_results: int = 8,\n as_string: Literal[False] = False,\n focus: list[str] | None = None,\n ) -> list[dict[str, str]]:\n ...\n\n async def run(\n self,\n query: str,\n max_results: int = 8,\n as_string: bool = True,\n ) -> str | list[dict]:\n \"\"\"Return the results of a Google search using the official Google API\n\n Args:\n query: The search query.\n max_results: The number of results to return.\n as_string: A boolean flag to determine the return type of the results. If True, the function will\n return a formatted string with the search results. If False, it will return a list of dictionaries\n containing detailed information about each search result.\n\n Returns:\n The results of the search.\n \"\"\"\n loop = self.loop or asyncio.get_event_loop()\n future = loop.run_in_executor(\n self.executor,\n self._search_from_ddgs,\n query,\n max_results,\n )\n search_results = await future\n\n # Return the list of search result URLs\n if as_string:\n return json.dumps(search_results, ensure_ascii=False)\n return search_results\n\n def _search_from_ddgs(self, query: str, max_results: int):\n return [\n {\"link\": i[\"href\"], \"snippet\": i[\"body\"], \"title\": i[\"title\"]}\n for (_, i) in zip(range(max_results), self.ddgs.text(query))\n ]\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(DDGAPIWrapper().run)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant User\n participant DDGAPIWrapper\n participant DDGS\n participant asyncio\n participant futures\n participant CONFIG\n participant fire\n\n User->>DDGAPIWrapper: Instantiate DDGAPIWrapper\n Note over DDGAPIWrapper: Wrapper around duckduckgo_search API\n \n alt Check if duckduckgo_search package is installed\n DDGAPIWrapper->>DDGAPIWrapper: Raise ImportError\n else\n DDGAPIWrapper->>DDGAPIWrapper: Set kwargs with global_proxy if available\n DDGAPIWrapper->>DDGAPIWrapper: Set loop and executor attributes\n DDGAPIWrapper->>DDGS: Instantiate DDGS with kwargs\n end\n\n User->>DDGAPIWrapper: Call run() method\n Note over DDGAPIWrapper: Overloaded method with different return types\n\n alt Return type is True\n DDGAPIWrapper->>asyncio: Get event loop\n DDGAPIWrapper->>loop: Run search_from_ddgs() in executor\n loop->>futures: Run search_from_ddgs() in executor\n futures->>DDGAPIWrapper: Return search results\n DDGAPIWrapper->>DDGAPIWrapper: Format search results as string\n DDGAPIWrapper->>User: Return search results as string\n else\n DDGAPIWrapper->>asyncio: Get event loop\n DDGAPIWrapper->>loop: Run search_from_ddgs() in executor\n loop->>futures: Run search_from_ddgs() in executor\n futures->>DDGAPIWrapper: Return search results\n DDGAPIWrapper->>User: Return search results as list of dictionaries\n end\n\n Note over DDGAPIWrapper: Private method _search_from_ddgs()\n\n DDGAPIWrapper->>DDGS: Call text() method with query\n DDGS->>DDGAPIWrapper: Return search results\n DDGAPIWrapper->>DDGAPIWrapper: Format search results as list of dictionaries\n DDGAPIWrapper->>User: Return search results as list of dictionaries\n\n User->>fire: Import fire module\n fire->>DDGAPIWrapper: Call run() method\n Note over DDGAPIWrapper: Run search() method with default parameters\n```", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/8/17\n@Author : mashenquan\n@File : metagpt_oas3_api_svc.py\n@Desc : MetaGPT OpenAPI Specification 3.0 REST API service\n\n curl -X 'POST' \\\n 'http://localhost:8080/openapi/greeting/dave' \\\n -H 'accept: text/plain' \\\n -H 'Content-Type: application/json' \\\n -d '{}'\n\"\"\"\n\nfrom pathlib import Path\n\nimport connexion\n\n\ndef oas_http_svc():\n \"\"\"Start the OAS 3.0 OpenAPI HTTP service\"\"\"\n print(\"http://localhost:8080/oas3/ui/\")\n specification_dir = Path(__file__).parent.parent.parent / \"docs/.well-known\"\n app = connexion.AsyncApp(__name__, specification_dir=str(specification_dir))\n app.add_api(\"metagpt_oas3_api.yaml\")\n app.add_api(\"openapi.yaml\")\n app.run(port=8080)\n\n\nif __name__ == \"__main__\":\n oas_http_svc()\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "To translate the given Python code into a Mermaid Sequence Diagram, we need to understand the flow of the code and identify the interactions between different components. Here's the translated code into a Mermaid Sequence Diagram:\n\n```mermaid\nsequenceDiagram\n participant User\n participant metagpt_oas3_api_svc.py\n participant connexion\n participant metagpt_oas3_api.yaml\n participant openapi.yaml\n\n User->>metagpt_oas3_api_svc.py: Start the OAS 3.0 OpenAPI HTTP service\n metagpt_oas3_api_svc.py->>connexion: Create an AsyncApp instance\n metagpt_oas3_api_svc.py->>connexion: Add the metagpt_oas3_api.yaml specification\n metagpt_oas3_api_svc.py->>connexion: Add the openapi.yaml specification\n metagpt_oas3_api_svc.py->>connexion: Run the HTTP service on port 8080\n connexion->>User: Display the URL for accessing the OAS 3.0 UI\n\n Note over metagpt_oas3_api_svc.py, connexion: The HTTP service is running on http://localhost:8080/oas3/ui/\n```\n\nIn the diagram, the User starts the OAS 3.0 OpenAPI HTTP service by executing the `oas_http_svc()` function in the `metagpt_oas3_api_svc.py` file. This function creates an instance of the `connexion.AsyncApp` class from the `connexion` library. The `metagpt_oas3_api.yaml` and `openapi.yaml` specifications are added to the app. Finally, the HTTP service is run on port 8080, and the URL for accessing the OAS 3.0 UI is displayed to the User.", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/5/23 18:27\n@Author : alexanderwu\n@File : search_engine_serpapi.py\n\"\"\"\nfrom typing import Any, Dict, Optional, Tuple\n\nimport aiohttp\nfrom pydantic import BaseModel, ConfigDict, Field, field_validator\n\nfrom metagpt.config import CONFIG\n\n\nclass SerpAPIWrapper(BaseModel):\n model_config = ConfigDict(arbitrary_types_allowed=True)\n\n search_engine: Any = None #: :meta private:\n params: dict = Field(\n default_factory=lambda: {\n \"engine\": \"google\",\n \"google_domain\": \"google.com\",\n \"gl\": \"us\",\n \"hl\": \"en\",\n }\n )\n # should add `validate_default=True` to check with default value\n serpapi_api_key: Optional[str] = Field(default=None, validate_default=True)\n aiosession: Optional[aiohttp.ClientSession] = None\n\n @field_validator(\"serpapi_api_key\", mode=\"before\")\n @classmethod\n def check_serpapi_api_key(cls, val: str):\n val = val or CONFIG.serpapi_api_key\n if not val:\n raise ValueError(\n \"To use, make sure you provide the serpapi_api_key when constructing an object. Alternatively, \"\n \"ensure that the environment variable SERPAPI_API_KEY is set with your API key. You can obtain \"\n \"an API key from https://serpapi.com/.\"\n )\n return val\n\n async def run(self, query, max_results: int = 8, as_string: bool = True, **kwargs: Any) -> str:\n \"\"\"Run query through SerpAPI and parse result async.\"\"\"\n result = await self.results(query, max_results)\n return self._process_response(result, as_string=as_string)\n\n async def results(self, query: str, max_results: int) -> dict:\n \"\"\"Use aiohttp to run query through SerpAPI and return the results async.\"\"\"\n\n def construct_url_and_params() -> Tuple[str, Dict[str, str]]:\n params = self.get_params(query)\n params[\"source\"] = \"python\"\n params[\"num\"] = max_results\n params[\"output\"] = \"json\"\n url = \"https://serpapi.com/search\"\n return url, params\n\n url, params = construct_url_and_params()\n if not self.aiosession:\n async with aiohttp.ClientSession() as session:\n async with session.get(url, params=params) as response:\n res = await response.json()\n else:\n async with self.aiosession.get(url, params=params) as response:\n res = await response.json()\n\n return res\n\n def get_params(self, query: str) -> Dict[str, str]:\n \"\"\"Get parameters for SerpAPI.\"\"\"\n _params = {\n \"api_key\": self.serpapi_api_key,\n \"q\": query,\n }\n params = {**self.params, **_params}\n return params\n\n @staticmethod\n def _process_response(res: dict, as_string: bool) -> str:\n \"\"\"Process response from SerpAPI.\"\"\"\n # logger.debug(res)\n focus = [\"title\", \"snippet\", \"link\"]\n get_focused = lambda x: {i: j for i, j in x.items() if i in focus}\n\n if \"error\" in res.keys():\n raise ValueError(f\"Got error from SerpAPI: {res['error']}\")\n if \"answer_box\" in res.keys() and \"answer\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"answer\"]\n elif \"answer_box\" in res.keys() and \"snippet\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"snippet\"]\n elif \"answer_box\" in res.keys() and \"snippet_highlighted_words\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"snippet_highlighted_words\"][0]\n elif \"sports_results\" in res.keys() and \"game_spotlight\" in res[\"sports_results\"].keys():\n toret = res[\"sports_results\"][\"game_spotlight\"]\n elif \"knowledge_graph\" in res.keys() and \"description\" in res[\"knowledge_graph\"].keys():\n toret = res[\"knowledge_graph\"][\"description\"]\n elif \"snippet\" in res[\"organic_results\"][0].keys():\n toret = res[\"organic_results\"][0][\"snippet\"]\n else:\n toret = \"No good search result found\"\n\n toret_l = []\n if \"answer_box\" in res.keys() and \"snippet\" in res[\"answer_box\"].keys():\n toret_l += [get_focused(res[\"answer_box\"])]\n if res.get(\"organic_results\"):\n toret_l += [get_focused(i) for i in res.get(\"organic_results\")]\n\n return str(toret) + \"\\n\" + str(toret_l) if as_string else toret_l\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(SerpAPIWrapper().run)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "The Mermaid Sequence Diagram for the given code is as follows:\n\n```mermaid\nsequenceDiagram\n participant User\n participant SerpAPIWrapper\n participant aiohttp.ClientSession\n participant SerpAPI\n\n User->>SerpAPIWrapper: Run query\n SerpAPIWrapper->>SerpAPIWrapper: Check serpapi_api_key\n alt serpapi_api_key is not provided\n SerpAPIWrapper-->>User: Raise ValueError\n else serpapi_api_key is provided\n SerpAPIWrapper->>SerpAPIWrapper: Get params\n SerpAPIWrapper->>SerpAPI: Send request\n SerpAPI-->>SerpAPIWrapper: Return response\n SerpAPIWrapper->>SerpAPIWrapper: Process response\n SerpAPIWrapper-->>User: Return result\n end\n```\n\nPlease note that the diagram is a simplified representation of the code logic and may not include all the details.", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/5/23 18:27\n@Author : alexanderwu\n@File : search_engine_serpapi.py\n\"\"\"\nimport json\nfrom typing import Any, Dict, Optional, Tuple\n\nimport aiohttp\nfrom pydantic import BaseModel, ConfigDict, Field, field_validator\n\nfrom metagpt.config import CONFIG\n\n\nclass SerperWrapper(BaseModel):\n model_config = ConfigDict(arbitrary_types_allowed=True)\n\n search_engine: Any = None #: :meta private:\n payload: dict = Field(default_factory=lambda: {\"page\": 1, \"num\": 10})\n serper_api_key: Optional[str] = Field(default=None, validate_default=True)\n aiosession: Optional[aiohttp.ClientSession] = None\n\n @field_validator(\"serper_api_key\", mode=\"before\")\n @classmethod\n def check_serper_api_key(cls, val: str):\n val = val or CONFIG.serper_api_key\n if not val:\n raise ValueError(\n \"To use, make sure you provide the serper_api_key when constructing an object. Alternatively, \"\n \"ensure that the environment variable SERPER_API_KEY is set with your API key. You can obtain \"\n \"an API key from https://serper.dev/.\"\n )\n return val\n\n async def run(self, query: str, max_results: int = 8, as_string: bool = True, **kwargs: Any) -> str:\n \"\"\"Run query through Serper and parse result async.\"\"\"\n if isinstance(query, str):\n return self._process_response((await self.results([query], max_results))[0], as_string=as_string)\n else:\n results = [self._process_response(res, as_string) for res in await self.results(query, max_results)]\n return \"\\n\".join(results) if as_string else results\n\n async def results(self, queries: list[str], max_results: int = 8) -> dict:\n \"\"\"Use aiohttp to run query through Serper and return the results async.\"\"\"\n\n def construct_url_and_payload_and_headers() -> Tuple[str, Dict[str, str]]:\n payloads = self.get_payloads(queries, max_results)\n url = \"https://google.serper.dev/search\"\n headers = self.get_headers()\n return url, payloads, headers\n\n url, payloads, headers = construct_url_and_payload_and_headers()\n if not self.aiosession:\n async with aiohttp.ClientSession() as session:\n async with session.post(url, data=payloads, headers=headers) as response:\n res = await response.json()\n else:\n async with self.aiosession.get.post(url, data=payloads, headers=headers) as response:\n res = await response.json()\n\n return res\n\n def get_payloads(self, queries: list[str], max_results: int) -> Dict[str, str]:\n \"\"\"Get payloads for Serper.\"\"\"\n payloads = []\n for query in queries:\n _payload = {\n \"q\": query,\n \"num\": max_results,\n }\n payloads.append({**self.payload, **_payload})\n return json.dumps(payloads, sort_keys=True)\n\n def get_headers(self) -> Dict[str, str]:\n headers = {\"X-API-KEY\": self.serper_api_key, \"Content-Type\": \"application/json\"}\n return headers\n\n @staticmethod\n def _process_response(res: dict, as_string: bool = False) -> str:\n \"\"\"Process response from SerpAPI.\"\"\"\n # logger.debug(res)\n focus = [\"title\", \"snippet\", \"link\"]\n\n def get_focused(x):\n return {i: j for i, j in x.items() if i in focus}\n\n if \"error\" in res.keys():\n raise ValueError(f\"Got error from SerpAPI: {res['error']}\")\n if \"answer_box\" in res.keys() and \"answer\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"answer\"]\n elif \"answer_box\" in res.keys() and \"snippet\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"snippet\"]\n elif \"answer_box\" in res.keys() and \"snippet_highlighted_words\" in res[\"answer_box\"].keys():\n toret = res[\"answer_box\"][\"snippet_highlighted_words\"][0]\n elif \"sports_results\" in res.keys() and \"game_spotlight\" in res[\"sports_results\"].keys():\n toret = res[\"sports_results\"][\"game_spotlight\"]\n elif \"knowledge_graph\" in res.keys() and \"description\" in res[\"knowledge_graph\"].keys():\n toret = res[\"knowledge_graph\"][\"description\"]\n elif \"snippet\" in res[\"organic\"][0].keys():\n toret = res[\"organic\"][0][\"snippet\"]\n else:\n toret = \"No good search result found\"\n\n toret_l = []\n if \"answer_box\" in res.keys() and \"snippet\" in res[\"answer_box\"].keys():\n toret_l += [get_focused(res[\"answer_box\"])]\n if res.get(\"organic\"):\n toret_l += [get_focused(i) for i in res.get(\"organic\")]\n\n return str(toret) + \"\\n\" + str(toret_l) if as_string else toret_l\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(SerperWrapper().run)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "The Mermaid Sequence Diagram for the given code can be represented as follows:\n\n```mermaid\nsequenceDiagram\n participant User\n participant SerperWrapper\n participant aiohttp.ClientSession\n participant SerpAPI\n\n User->>SerperWrapper: run(query, max_results, as_string, **kwargs)\n SerperWrapper->>SerperWrapper: _process_response()\n SerperWrapper->>SerperWrapper: get_payloads()\n SerperWrapper->>SerperWrapper: get_headers()\n SerperWrapper->>aiohttp.ClientSession: post(url, data, headers)\n aiohttp.ClientSession->>SerpAPI: POST /search\n SerpAPI-->>aiohttp.ClientSession: Response\n aiohttp.ClientSession-->>SerperWrapper: Response\n SerperWrapper->>SerperWrapper: _process_response()\n SerperWrapper->>User: Response\n```\n\nNote: This diagram represents the flow of execution for the `run()` method in the `SerperWrapper` class. It shows the interaction between the user, the `SerperWrapper` object, the `aiohttp.ClientSession`, and the SerpAPI.", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\nfrom __future__ import annotations\n\nimport asyncio\nimport json\nfrom concurrent import futures\nfrom typing import Optional\nfrom urllib.parse import urlparse\n\nimport httplib2\nfrom pydantic import BaseModel, ConfigDict, Field, field_validator\n\nfrom metagpt.config import CONFIG\nfrom metagpt.logs import logger\n\ntry:\n from googleapiclient.discovery import build\n from googleapiclient.errors import HttpError\nexcept ImportError:\n raise ImportError(\n \"To use this module, you should have the `google-api-python-client` Python package installed. \"\n \"You can install it by running the command: `pip install -e.[search-google]`\"\n )\n\n\nclass GoogleAPIWrapper(BaseModel):\n model_config = ConfigDict(arbitrary_types_allowed=True)\n\n google_api_key: Optional[str] = Field(default=None, validate_default=True)\n google_cse_id: Optional[str] = Field(default=None, validate_default=True)\n loop: Optional[asyncio.AbstractEventLoop] = None\n executor: Optional[futures.Executor] = None\n\n @field_validator(\"google_api_key\", mode=\"before\")\n @classmethod\n def check_google_api_key(cls, val: str):\n val = val or CONFIG.google_api_key\n if not val:\n raise ValueError(\n \"To use, make sure you provide the google_api_key when constructing an object. Alternatively, \"\n \"ensure that the environment variable GOOGLE_API_KEY is set with your API key. You can obtain \"\n \"an API key from https://console.cloud.google.com/apis/credentials.\"\n )\n return val\n\n @field_validator(\"google_cse_id\", mode=\"before\")\n @classmethod\n def check_google_cse_id(cls, val: str):\n val = val or CONFIG.google_cse_id\n if not val:\n raise ValueError(\n \"To use, make sure you provide the google_cse_id when constructing an object. Alternatively, \"\n \"ensure that the environment variable GOOGLE_CSE_ID is set with your API key. You can obtain \"\n \"an API key from https://programmablesearchengine.google.com/controlpanel/create.\"\n )\n return val\n\n @property\n def google_api_client(self):\n build_kwargs = {\"developerKey\": self.google_api_key}\n if CONFIG.global_proxy:\n parse_result = urlparse(CONFIG.global_proxy)\n proxy_type = parse_result.scheme\n if proxy_type == \"https\":\n proxy_type = \"http\"\n build_kwargs[\"http\"] = httplib2.Http(\n proxy_info=httplib2.ProxyInfo(\n getattr(httplib2.socks, f\"PROXY_TYPE_{proxy_type.upper()}\"),\n parse_result.hostname,\n parse_result.port,\n ),\n )\n service = build(\"customsearch\", \"v1\", **build_kwargs)\n return service.cse()\n\n async def run(\n self,\n query: str,\n max_results: int = 8,\n as_string: bool = True,\n focus: list[str] | None = None,\n ) -> str | list[dict]:\n \"\"\"Return the results of a Google search using the official Google API.\n\n Args:\n query: The search query.\n max_results: The number of results to return.\n as_string: A boolean flag to determine the return type of the results. If True, the function will\n return a formatted string with the search results. If False, it will return a list of dictionaries\n containing detailed information about each search result.\n focus: Specific information to be focused on from each search result.\n\n Returns:\n The results of the search.\n \"\"\"\n loop = self.loop or asyncio.get_event_loop()\n future = loop.run_in_executor(\n self.executor, self.google_api_client.list(q=query, num=max_results, cx=self.google_cse_id).execute\n )\n try:\n result = await future\n # Extract the search result items from the response\n search_results = result.get(\"items\", [])\n\n except HttpError as e:\n # Handle errors in the API call\n logger.exception(f\"fail to search {query} for {e}\")\n search_results = []\n\n focus = focus or [\"snippet\", \"link\", \"title\"]\n details = [{i: j for i, j in item_dict.items() if i in focus} for item_dict in search_results]\n # Return the list of search result URLs\n if as_string:\n return safe_google_results(details)\n\n return details\n\n\ndef safe_google_results(results: str | list) -> str:\n \"\"\"Return the results of a google search in a safe format.\n\n Args:\n results: The search results.\n\n Returns:\n The results of the search.\n \"\"\"\n if isinstance(results, list):\n safe_message = json.dumps([result for result in results])\n else:\n safe_message = results.encode(\"utf-8\", \"ignore\").decode(\"utf-8\")\n return safe_message\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(GoogleAPIWrapper().run)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "Here is the Mermaid Sequence Diagram representation of the given code:\n\n```mermaid\nsequenceDiagram\n participant User\n participant GoogleAPIWrapper\n participant asyncio\n participant httplib2\n participant GoogleAPI\n\n User->>GoogleAPIWrapper: Construct object\n GoogleAPIWrapper->>GoogleAPIWrapper: check_google_api_key()\n GoogleAPIWrapper->>GoogleAPIWrapper: check_google_cse_id()\n GoogleAPIWrapper->>GoogleAPIWrapper: google_api_client()\n GoogleAPIWrapper->>User: Object with google_api_client\n\n User->>GoogleAPIWrapper: run(query, max_results, as_string, focus)\n GoogleAPIWrapper->>asyncio: get_event_loop()\n asyncio->>GoogleAPIWrapper: loop\n GoogleAPIWrapper->>GoogleAPIWrapper: executor\n GoogleAPIWrapper->>loop: run_in_executor()\n loop->>GoogleAPIWrapper: future\n GoogleAPIWrapper->>GoogleAPI: google_api_client.list()\n GoogleAPI-->GoogleAPIWrapper: Response\n GoogleAPIWrapper->>GoogleAPIWrapper: Extract search results\n GoogleAPIWrapper->>User: Search results\n```\n\nPlease note that this diagram represents the flow of the code and the interactions between different components. It does not include all the details of the code implementation.", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/5/2 16:03\n@Author : mashenquan\n@File : openapi_v3_hello.py\n@Desc : Implement the OpenAPI Specification 3.0 demo and use the following command to test the HTTP service:\n\n curl -X 'POST' \\\n 'http://localhost:8082/openapi/greeting/dave' \\\n -H 'accept: text/plain' \\\n -H 'Content-Type: application/json' \\\n -d '{}'\n\"\"\"\nfrom pathlib import Path\n\nimport connexion\n\n\n# openapi implement\nasync def post_greeting(name: str) -> str:\n return f\"Hello {name}\\n\"\n\n\nif __name__ == \"__main__\":\n specification_dir = Path(__file__).parent.parent.parent / \"docs/.well-known\"\n app = connexion.AsyncApp(__name__, specification_dir=str(specification_dir))\n app.add_api(\"openapi.yaml\", arguments={\"title\": \"Hello World Example\"})\n app.run(port=8082)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "To translate the given Python code into a Mermaid Sequence Diagram, we need to understand the flow of the code and identify the interactions between different components. Here's the translation:\n\n```mermaid\nsequenceDiagram\n participant User\n participant HTTPService\n participant OpenAPIImplementation\n participant Connexion\n\n User->>HTTPService: HTTP POST Request\n HTTPService->>OpenAPIImplementation: Call post_greeting(name)\n OpenAPIImplementation->>Connexion: Return \"Hello {name}\"\n Connexion->>HTTPService: Return \"Hello {name}\"\n HTTPService->>User: HTTP Response\n```\n\nIn the diagram above, the User sends an HTTP POST request to the HTTPService. The HTTPService then calls the `post_greeting` function in the OpenAPIImplementation. The OpenAPIImplementation returns the greeting message to Connexion, which in turn returns the message to the HTTPService. Finally, the HTTPService sends the HTTP response back to the User.\n\nNote that this diagram represents the high-level interactions between the components and does not include all the internal details of the code.", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n# -*- coding: utf-8 -*-\n# @Date : 2023/7/19 16:28\n# @Author : stellahong (stellahong@deepwisdom.ai)\n# @Desc :\nimport asyncio\nimport base64\nimport io\nimport json\nfrom os.path import join\nfrom typing import List\n\nfrom aiohttp import ClientSession\nfrom PIL import Image, PngImagePlugin\n\nfrom metagpt.config import CONFIG\nfrom metagpt.const import SD_OUTPUT_FILE_REPO\nfrom metagpt.logs import logger\n\npayload = {\n \"prompt\": \"\",\n \"negative_prompt\": \"(easynegative:0.8),black, dark,Low resolution\",\n \"override_settings\": {\"sd_model_checkpoint\": \"galaxytimemachinesGTM_photoV20\"},\n \"seed\": -1,\n \"batch_size\": 1,\n \"n_iter\": 1,\n \"steps\": 20,\n \"cfg_scale\": 7,\n \"width\": 512,\n \"height\": 768,\n \"restore_faces\": False,\n \"tiling\": False,\n \"do_not_save_samples\": False,\n \"do_not_save_grid\": False,\n \"enable_hr\": False,\n \"hr_scale\": 2,\n \"hr_upscaler\": \"Latent\",\n \"hr_second_pass_steps\": 0,\n \"hr_resize_x\": 0,\n \"hr_resize_y\": 0,\n \"hr_upscale_to_x\": 0,\n \"hr_upscale_to_y\": 0,\n \"truncate_x\": 0,\n \"truncate_y\": 0,\n \"applied_old_hires_behavior_to\": None,\n \"eta\": None,\n \"sampler_index\": \"DPM++ SDE Karras\",\n \"alwayson_scripts\": {},\n}\n\ndefault_negative_prompt = \"(easynegative:0.8),black, dark,Low resolution\"\n\n\nclass SDEngine:\n def __init__(self):\n # Initialize the SDEngine with configuration\n self.sd_url = CONFIG.get(\"SD_URL\")\n self.sd_t2i_url = f\"{self.sd_url}{CONFIG.get('SD_T2I_API')}\"\n # Define default payload settings for SD API\n self.payload = payload\n logger.info(self.sd_t2i_url)\n\n def construct_payload(\n self,\n prompt,\n negtive_prompt=default_negative_prompt,\n width=512,\n height=512,\n sd_model=\"galaxytimemachinesGTM_photoV20\",\n ):\n # Configure the payload with provided inputs\n self.payload[\"prompt\"] = prompt\n self.payload[\"negtive_prompt\"] = negtive_prompt\n self.payload[\"width\"] = width\n self.payload[\"height\"] = height\n self.payload[\"override_settings\"][\"sd_model_checkpoint\"] = sd_model\n logger.info(f\"call sd payload is {self.payload}\")\n return self.payload\n\n def _save(self, imgs, save_name=\"\"):\n save_dir = CONFIG.workspace_path / SD_OUTPUT_FILE_REPO\n if not save_dir.exists():\n save_dir.mkdir(parents=True, exist_ok=True)\n batch_decode_base64_to_image(imgs, str(save_dir), save_name=save_name)\n\n async def run_t2i(self, prompts: List):\n # Asynchronously run the SD API for multiple prompts\n session = ClientSession()\n for payload_idx, payload in enumerate(prompts):\n results = await self.run(url=self.sd_t2i_url, payload=payload, session=session)\n self._save(results, save_name=f\"output_{payload_idx}\")\n await session.close()\n\n async def run(self, url, payload, session):\n # Perform the HTTP POST request to the SD API\n async with session.post(url, json=payload, timeout=600) as rsp:\n data = await rsp.read()\n\n rsp_json = json.loads(data)\n imgs = rsp_json[\"images\"]\n logger.info(f\"callback rsp json is {rsp_json.keys()}\")\n return imgs\n\n async def run_i2i(self):\n # todo: 添加图生图接口调用\n raise NotImplementedError\n\n async def run_sam(self):\n # todo:添加SAM接口调用\n raise NotImplementedError\n\n\ndef decode_base64_to_image(img, save_name):\n image = Image.open(io.BytesIO(base64.b64decode(img.split(\",\", 1)[0])))\n pnginfo = PngImagePlugin.PngInfo()\n logger.info(save_name)\n image.save(f\"{save_name}.png\", pnginfo=pnginfo)\n return pnginfo, image\n\n\ndef batch_decode_base64_to_image(imgs, save_dir=\"\", save_name=\"\"):\n for idx, _img in enumerate(imgs):\n save_name = join(save_dir, save_name)\n decode_base64_to_image(_img, save_name=save_name)\n\n\nif __name__ == \"__main__\":\n engine = SDEngine()\n prompt = \"pixel style, game design, a game interface should be minimalistic and intuitive with the score and high score displayed at the top. The snake and its food should be easily distinguishable. The game should have a simple color scheme, with a contrasting color for the snake and its food. Complete interface boundary\"\n\n engine.construct_payload(prompt)\n\n event_loop = asyncio.get_event_loop()\n event_loop.run_until_complete(engine.run_t2i(prompt))\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant SDEngine\n participant ClientSession\n participant SD_API\n participant Image\n participant PngImagePlugin\n participant io\n participant base64\n participant json\n\n SDEngine->>+SDEngine: Initialize SDEngine with configuration\n SDEngine-->>-ClientSession: Create a ClientSession\n SDEngine->>+ClientSession: Send POST request to SD API\n ClientSession->>+SD_API: POST /sd_t2i_api\n ClientSession-->>-SD_API: Payload\n SD_API->>-SD_API: Process the request\n SD_API-->>-ClientSession: Response\n ClientSession->>-ClientSession: Close the session\n SDEngine->>+SDEngine: Save the images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-ClientSession: Response\n ClientSession-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine: Return the saved images\n SDEngine-->>-SDEngine", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/6/5 01:44\n@Author : alexanderwu\n@File : skill_manager.py\n@Modified By: mashenquan, 2023/8/20. Remove useless `llm`\n\"\"\"\nfrom metagpt.actions import Action\nfrom metagpt.const import PROMPT_PATH\nfrom metagpt.document_store.chromadb_store import ChromaStore\nfrom metagpt.logs import logger\n\nSkill = Action\n\n\nclass SkillManager:\n \"\"\"Used to manage all skills\"\"\"\n\n def __init__(self):\n self._store = ChromaStore(\"skill_manager\")\n self._skills: dict[str:Skill] = {}\n\n def add_skill(self, skill: Skill):\n \"\"\"\n Add a skill, add the skill to the skill pool and searchable storage\n :param skill: Skill\n :return:\n \"\"\"\n self._skills[skill.name] = skill\n self._store.add(skill.desc, {\"name\": skill.name, \"desc\": skill.desc}, skill.name)\n\n def del_skill(self, skill_name: str):\n \"\"\"\n Delete a skill, remove the skill from the skill pool and searchable storage\n :param skill_name: Skill name\n :return:\n \"\"\"\n self._skills.pop(skill_name)\n self._store.delete(skill_name)\n\n def get_skill(self, skill_name: str) -> Skill:\n \"\"\"\n Obtain a specific skill by skill name\n :param skill_name: Skill name\n :return: Skill\n \"\"\"\n return self._skills.get(skill_name)\n\n def retrieve_skill(self, desc: str, n_results: int = 2) -> list[Skill]:\n \"\"\"\n Obtain skills through the search engine\n :param desc: Skill description\n :return: Multiple skills\n \"\"\"\n return self._store.search(desc, n_results=n_results)[\"ids\"][0]\n\n def retrieve_skill_scored(self, desc: str, n_results: int = 2) -> dict:\n \"\"\"\n Obtain skills through the search engine\n :param desc: Skill description\n :return: Dictionary consisting of skills and scores\n \"\"\"\n return self._store.search(desc, n_results=n_results)\n\n def generate_skill_desc(self, skill: Skill) -> str:\n \"\"\"\n Generate descriptive text for each skill\n :param skill:\n :return:\n \"\"\"\n path = PROMPT_PATH / \"generate_skill.md\"\n text = path.read_text()\n logger.info(text)\n\n\nif __name__ == \"__main__\":\n manager = SkillManager()\n manager.generate_skill_desc(Action())\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "The Mermaid Sequence Diagram for the given code can be represented as follows:\n\n```mermaid\nsequenceDiagram\n participant SkillManager\n participant ChromaStore\n participant Skill\n participant Action\n\n SkillManager -> ChromaStore: add_skill(skill)\n ChromaStore -> SkillManager: add(skill.desc, {\"name\": skill.name, \"desc\": skill.desc}, skill.name)\n SkillManager -> Skill: skill.name, skill.desc\n Skill -> SkillManager: skill\n SkillManager -> ChromaStore: delete(skill_name)\n ChromaStore -> SkillManager: delete(skill_name)\n SkillManager -> Skill: skill_name\n Skill -> SkillManager: None\n SkillManager -> Skill: skill_name\n Skill -> SkillManager: skill\n SkillManager -> ChromaStore: search(desc, n_results)\n ChromaStore -> SkillManager: Multiple skills\n SkillManager -> ChromaStore: search(desc, n_results)\n ChromaStore -> SkillManager: Dictionary consisting of skills and scores\n SkillManager -> PROMPT_PATH: read_text()\n PROMPT_PATH -> SkillManager: text\n```\n\nNote: The `PROMPT_PATH` is not defined in the given code, so it is assumed to be a constant or variable that represents a file path.", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n\"\"\"\n@Modified By: mashenquan, 2023/8/22. A definition has been provided for the return value of _think: returning false indicates that further reasoning cannot continue.\n@Modified By: mashenquan, 2023-11-1. According to Chapter 2.2.1 and 2.2.2 of RFC 116, change the data type of\n the `cause_by` value in the `Message` to a string to support the new message distribution feature.\n\"\"\"\n\nimport asyncio\nimport re\n\nfrom pydantic import BaseModel\n\nfrom metagpt.actions import Action, CollectLinks, ConductResearch, WebBrowseAndSummarize\nfrom metagpt.actions.research import get_research_system_text\nfrom metagpt.const import RESEARCH_PATH\nfrom metagpt.logs import logger\nfrom metagpt.roles.role import Role, RoleReactMode\nfrom metagpt.schema import Message\n\n\nclass Report(BaseModel):\n topic: str\n links: dict[str, list[str]] = None\n summaries: list[tuple[str, str]] = None\n content: str = \"\"\n\n\nclass Researcher(Role):\n name: str = \"David\"\n profile: str = \"Researcher\"\n goal: str = \"Gather information and conduct research\"\n constraints: str = \"Ensure accuracy and relevance of information\"\n language: str = \"en-us\"\n\n def __init__(self, **kwargs):\n super().__init__(**kwargs)\n self._init_actions(\n [CollectLinks(name=self.name), WebBrowseAndSummarize(name=self.name), ConductResearch(name=self.name)]\n )\n self._set_react_mode(react_mode=RoleReactMode.BY_ORDER.value)\n if self.language not in (\"en-us\", \"zh-cn\"):\n logger.warning(f\"The language `{self.language}` has not been tested, it may not work.\")\n\n async def _think(self) -> bool:\n if self.rc.todo is None:\n self._set_state(0)\n return True\n\n if self.rc.state + 1 < len(self.states):\n self._set_state(self.rc.state + 1)\n else:\n self.rc.todo = None\n return False\n\n async def _act(self) -> Message:\n logger.info(f\"{self._setting}: to do {self.rc.todo}({self.rc.todo.name})\")\n todo = self.rc.todo\n msg = self.rc.memory.get(k=1)[0]\n if isinstance(msg.instruct_content, Report):\n instruct_content = msg.instruct_content\n topic = instruct_content.topic\n else:\n topic = msg.content\n\n research_system_text = self.research_system_text(topic, todo)\n if isinstance(todo, CollectLinks):\n links = await todo.run(topic, 4, 4)\n ret = Message(\n content=\"\", instruct_content=Report(topic=topic, links=links), role=self.profile, cause_by=todo\n )\n elif isinstance(todo, WebBrowseAndSummarize):\n links = instruct_content.links\n todos = (todo.run(*url, query=query, system_text=research_system_text) for (query, url) in links.items())\n summaries = await asyncio.gather(*todos)\n summaries = list((url, summary) for i in summaries for (url, summary) in i.items() if summary)\n ret = Message(\n content=\"\", instruct_content=Report(topic=topic, summaries=summaries), role=self.profile, cause_by=todo\n )\n else:\n summaries = instruct_content.summaries\n summary_text = \"\\n---\\n\".join(f\"url: {url}\\nsummary: {summary}\" for (url, summary) in summaries)\n content = await self.rc.todo.run(topic, summary_text, system_text=research_system_text)\n ret = Message(\n content=\"\",\n instruct_content=Report(topic=topic, content=content),\n role=self.profile,\n cause_by=self.rc.todo,\n )\n self.rc.memory.add(ret)\n return ret\n\n def research_system_text(self, topic, current_task: Action) -> str:\n \"\"\"BACKWARD compatible\n This allows sub-class able to define its own system prompt based on topic.\n return the previous implementation to have backward compatible\n Args:\n topic:\n language:\n\n Returns: str\n \"\"\"\n return get_research_system_text(topic, self.language)\n\n async def react(self) -> Message:\n msg = await super().react()\n report = msg.instruct_content\n self.write_report(report.topic, report.content)\n return msg\n\n def write_report(self, topic: str, content: str):\n filename = re.sub(r'[\\\\/:\"*?<>|]+', \" \", topic)\n filename = filename.replace(\"\\n\", \"\")\n if not RESEARCH_PATH.exists():\n RESEARCH_PATH.mkdir(parents=True)\n filepath = RESEARCH_PATH / f\"{filename}.md\"\n filepath.write_text(content)\n\n\nif __name__ == \"__main__\":\n import fire\n\n async def main(topic: str, language=\"en-us\"):\n role = Researcher(language=language)\n await role.run(topic)\n\n fire.Fire(main)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "Here is the Mermaid Sequence Diagram representation of the provided code:\n\n```mermaid\nsequenceDiagram\n participant Researcher\n participant Action\n participant CollectLinks\n participant WebBrowseAndSummarize\n participant ConductResearch\n participant Message\n participant Report\n\n Researcher->>Action: Initialize actions\n Researcher->>Action: Set react mode\n Researcher->>Researcher: Check language compatibility\n Researcher->>Researcher: Think\n Researcher->>Action: Act\n Action->>Message: Get message from memory\n Message-->>Action: Return message\n Action->>Researcher: Act on message\n Researcher->>CollectLinks: Run CollectLinks action\n CollectLinks->>CollectLinks: Collect links\n CollectLinks-->>Researcher: Return links\n Researcher->>Message: Create Report message\n Message-->>Researcher: Return Report message\n Researcher->>WebBrowseAndSummarize: Run WebBrowseAndSummarize action\n WebBrowseAndSummarize->>WebBrowseAndSummarize: Browse and summarize links\n WebBrowseAndSummarize-->>Researcher: Return summaries\n Researcher->>Message: Create Report message\n Message-->>Researcher: Return Report message\n Researcher->>ConductResearch: Run ConductResearch action\n ConductResearch->>ConductResearch: Conduct research\n ConductResearch-->>Researcher: Return research content\n Researcher->>Message: Create Report message\n Message-->>Researcher: Return Report message\n Researcher->>Researcher: Add message to memory\n Researcher->>Researcher: Think\n Researcher->>Action: Act\n Action->>Message: Get message from memory\n Message-->>Action: Return message\n Action->>Researcher: Act on message\n Researcher->>Researcher: Write report\n Researcher->>Message: Return message\n```\n\nPlease note that this is a simplified representation of the code logic and may not include all the details.", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/12/14 11:40\n@Author : alexanderwu\n@File : write_prd_an.py\n\"\"\"\nfrom typing import List\n\nfrom metagpt.actions.action_node import ActionNode\nfrom metagpt.logs import logger\n\nLANGUAGE = ActionNode(\n key=\"Language\",\n expected_type=str,\n instruction=\"Provide the language used in the project, typically matching the user's requirement language.\",\n example=\"en_us\",\n)\n\nPROGRAMMING_LANGUAGE = ActionNode(\n key=\"Programming Language\",\n expected_type=str,\n instruction=\"Python/JavaScript or other mainstream programming language.\",\n example=\"Python\",\n)\n\nORIGINAL_REQUIREMENTS = ActionNode(\n key=\"Original Requirements\",\n expected_type=str,\n instruction=\"Place the original user's requirements here.\",\n example=\"Create a 2048 game\",\n)\n\nPROJECT_NAME = ActionNode(\n key=\"Project Name\",\n expected_type=str,\n instruction=\"According to the content of \\\"Original Requirements,\\\" name the project using snake case style , like 'game_2048' or 'simple_crm.\",\n example=\"game_2048\",\n)\n\nPRODUCT_GOALS = ActionNode(\n key=\"Product Goals\",\n expected_type=List[str],\n instruction=\"Provide up to three clear, orthogonal product goals.\",\n example=[\"Create an engaging user experience\", \"Improve accessibility, be responsive\", \"More beautiful UI\"],\n)\n\nUSER_STORIES = ActionNode(\n key=\"User Stories\",\n expected_type=List[str],\n instruction=\"Provide up to 3 to 5 scenario-based user stories.\",\n example=[\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\",\n ],\n)\n\nCOMPETITIVE_ANALYSIS = ActionNode(\n key=\"Competitive Analysis\",\n expected_type=List[str],\n instruction=\"Provide 5 to 7 competitive products.\",\n example=[\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\",\n ],\n)\n\nCOMPETITIVE_QUADRANT_CHART = ActionNode(\n key=\"Competitive Quadrant Chart\",\n expected_type=str,\n instruction=\"Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\",\n example=\"\"\"quadrantChart\n title \"Reach and engagement of campaigns\"\n x-axis \"Low Reach\" --> \"High Reach\"\n y-axis \"Low Engagement\" --> \"High Engagement\"\n quadrant-1 \"We should expand\"\n quadrant-2 \"Need to promote\"\n quadrant-3 \"Re-evaluate\"\n quadrant-4 \"May be improved\"\n \"Campaign A\": [0.3, 0.6]\n \"Campaign B\": [0.45, 0.23]\n \"Campaign C\": [0.57, 0.69]\n \"Campaign D\": [0.78, 0.34]\n \"Campaign E\": [0.40, 0.34]\n \"Campaign F\": [0.35, 0.78]\n \"Our Target Product\": [0.5, 0.6]\"\"\",\n)\n\nREQUIREMENT_ANALYSIS = ActionNode(\n key=\"Requirement Analysis\",\n expected_type=str,\n instruction=\"Provide a detailed analysis of the requirements.\",\n example=\"\",\n)\n\nREQUIREMENT_POOL = ActionNode(\n key=\"Requirement Pool\",\n expected_type=List[List[str]],\n instruction=\"List down the top-5 requirements with their priority (P0, P1, P2).\",\n example=[[\"P0\", \"The main code ...\"], [\"P0\", \"The game algorithm ...\"]],\n)\n\nUI_DESIGN_DRAFT = ActionNode(\n key=\"UI Design draft\",\n expected_type=str,\n instruction=\"Provide a simple description of UI elements, functions, style, and layout.\",\n example=\"Basic function description with a simple style and layout.\",\n)\n\nANYTHING_UNCLEAR = ActionNode(\n key=\"Anything UNCLEAR\",\n expected_type=str,\n instruction=\"Mention any aspects of the project that are unclear and try to clarify them.\",\n example=\"\",\n)\n\nISSUE_TYPE = ActionNode(\n key=\"issue_type\",\n expected_type=str,\n instruction=\"Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement\",\n example=\"BUG\",\n)\n\nIS_RELATIVE = ActionNode(\n key=\"is_relative\",\n expected_type=str,\n instruction=\"Answer YES/NO. If the requirement is related to the old PRD, answer YES, otherwise NO\",\n example=\"YES\",\n)\n\nREASON = ActionNode(\n key=\"reason\", expected_type=str, instruction=\"Explain the reasoning process from question to answer\", example=\"...\"\n)\n\n\nNODES = [\n LANGUAGE,\n PROGRAMMING_LANGUAGE,\n ORIGINAL_REQUIREMENTS,\n PROJECT_NAME,\n PRODUCT_GOALS,\n USER_STORIES,\n COMPETITIVE_ANALYSIS,\n COMPETITIVE_QUADRANT_CHART,\n REQUIREMENT_ANALYSIS,\n REQUIREMENT_POOL,\n UI_DESIGN_DRAFT,\n ANYTHING_UNCLEAR,\n]\n\nWRITE_PRD_NODE = ActionNode.from_children(\"WritePRD\", NODES)\nWP_ISSUE_TYPE_NODE = ActionNode.from_children(\"WP_ISSUE_TYPE\", [ISSUE_TYPE, REASON])\nWP_IS_RELATIVE_NODE = ActionNode.from_children(\"WP_IS_RELATIVE\", [IS_RELATIVE, REASON])\n\n\ndef main():\n prompt = WRITE_PRD_NODE.compile(context=\"\")\n logger.info(prompt)\n\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant Language\n participant Programming Language\n participant Original Requirements\n participant Project Name\n participant Product Goals\n participant User Stories\n participant Competitive Analysis\n participant Competitive Quadrant Chart\n participant Requirement Analysis\n participant Requirement Pool\n participant UI Design draft\n participant Anything UNCLEAR\n participant issue_type\n participant is_relative\n participant reason\n participant WritePRD\n participant WP_ISSUE_TYPE\n participant WP_IS_RELATIVE\n \n WritePRD->>Language: Provide the language used in the project, typically matching the user's requirement language.\n WritePRD->>Programming Language: Python/JavaScript or other mainstream programming language.\n WritePRD->>Original Requirements: Place the original user's requirements here.\n WritePRD->>Project Name: According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n WritePRD->>Product Goals: Provide up to three clear, orthogonal product goals.\n WritePRD->>User Stories: Provide up to 3 to 5 scenario-based user stories.\n WritePRD->>Competitive Analysis: Provide 5 to 7 competitive products.\n WritePRD->>Competitive Quadrant Chart: Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n WritePRD->>Requirement Analysis: Provide a detailed analysis of the requirements.\n WritePRD->>Requirement Pool: List down the top-5 requirements with their priority (P0, P1, P2).\n WritePRD->>UI Design draft: Provide a simple description of UI elements, functions, style, and layout.\n WritePRD->>Anything UNCLEAR: Mention any aspects of the project that are unclear and try to clarify them.\n \n WritePRD->>WP_ISSUE_TYPE: Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement\n WP_ISSUE_TYPE->>issue_type: BUG\n WP_ISSUE_TYPE->>reason: Explain the reasoning process from question to answer\n \n WritePRD->>WP_IS_RELATIVE: Answer YES/NO. If the requirement is related to the old PRD, answer YES, otherwise NO\n WP_IS_RELATIVE->>is_relative: YES\n WP_IS_RELATIVE->>reason: Explain the reasoning process from question to answer\n \n WritePRD-->>Language: expected_type: str\n WritePRD-->>Programming Language: expected_type: str\n WritePRD-->>Original Requirements: expected_type: str\n WritePRD-->>Project Name: expected_type: str\n WritePRD-->>Product Goals: expected_type: List[str]\n WritePRD-->>User Stories: expected_type: List[str]\n WritePRD-->>Competitive Analysis: expected_type: List[str]\n WritePRD-->>Competitive Quadrant Chart: expected_type: str\n WritePRD-->>Requirement Analysis: expected_type: str\n WritePRD-->>Requirement Pool: expected_type: List[List[str]]\n WritePRD-->>UI Design draft: expected_type: str\n WritePRD-->>Anything UNCLEAR: expected_type: str\n \n WP_ISSUE_TYPE-->>issue_type: expected_type: str\n WP_ISSUE_TYPE-->>reason: expected_type: str\n \n WP_IS_RELATIVE-->>is_relative: expected_type: str\n WP_IS_RELATIVE-->>reason: expected_type: str\n```", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n\"\"\"Code Docstring Generator.\n\nThis script provides a tool to automatically generate docstrings for Python code. It uses the specified style to create\ndocstrings for the given code and system text.\n\nUsage:\n python3 -m metagpt.actions.write_docstring [--overwrite] [--style=]\n\nArguments:\n filename The path to the Python file for which you want to generate docstrings.\n\nOptions:\n --overwrite If specified, overwrite the original file with the code containing docstrings.\n --style= Specify the style of the generated docstrings.\n Valid values: 'google', 'numpy', or 'sphinx'.\n Default: 'google'\n\nExample:\n python3 -m metagpt.actions.write_docstring ./metagpt/startup.py --overwrite False --style=numpy\n\nThis script uses the 'fire' library to create a command-line interface. It generates docstrings for the given Python code using\nthe specified docstring style and adds them to the code.\n\"\"\"\nfrom __future__ import annotations\n\nimport ast\nfrom pathlib import Path\nfrom typing import Literal, Optional\n\nfrom metagpt.actions.action import Action\nfrom metagpt.utils.common import OutputParser, aread, awrite\nfrom metagpt.utils.pycst import merge_docstring\n\nPYTHON_DOCSTRING_SYSTEM = \"\"\"### Requirements\n1. Add docstrings to the given code following the {style} style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\n{example}\n```\n\"\"\"\n\n# https://www.sphinx-doc.org/en/master/usage/extensions/napoleon.html\n\nPYTHON_DOCSTRING_EXAMPLE_GOOGLE = '''\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Args:\n param1: The first parameter.\n\n Returns:\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Args:\n msg: Human readable string describing the exception.\n\n Attributes:\n msg: Human readable string describing the exception.\n \"\"\"\n ...\n'''\n\nPYTHON_DOCSTRING_EXAMPLE_NUMPY = '''\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"\n Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Parameters\n ----------\n param1\n The first parameter.\n\n Returns\n -------\n bool\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"\n Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Parameters\n ----------\n msg\n Human readable string describing the exception.\n\n Attributes\n ----------\n msg\n Human readable string describing the exception.\n \"\"\"\n ...\n'''\n\nPYTHON_DOCSTRING_EXAMPLE_SPHINX = '''\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n :param param1: The first parameter.\n :type param1: int\n\n :return: The return value. True for success, False otherwise.\n :rtype: bool\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n :param msg: Human-readable string describing the exception.\n :type msg: str\n \"\"\"\n ...\n'''\n\n_python_docstring_style = {\n \"google\": PYTHON_DOCSTRING_EXAMPLE_GOOGLE.strip(),\n \"numpy\": PYTHON_DOCSTRING_EXAMPLE_NUMPY.strip(),\n \"sphinx\": PYTHON_DOCSTRING_EXAMPLE_SPHINX.strip(),\n}\n\n\nclass WriteDocstring(Action):\n \"\"\"This class is used to write docstrings for code.\n\n Attributes:\n desc: A string describing the action.\n \"\"\"\n\n desc: str = \"Write docstring for code.\"\n context: Optional[str] = None\n\n async def run(\n self,\n code: str,\n system_text: str = PYTHON_DOCSTRING_SYSTEM,\n style: Literal[\"google\", \"numpy\", \"sphinx\"] = \"google\",\n ) -> str:\n \"\"\"Writes docstrings for the given code and system text in the specified style.\n\n Args:\n code: A string of Python code.\n system_text: A string of system text.\n style: A string specifying the style of the docstring. Can be 'google', 'numpy', or 'sphinx'.\n\n Returns:\n The Python code with docstrings added.\n \"\"\"\n system_text = system_text.format(style=style, example=_python_docstring_style[style])\n simplified_code = _simplify_python_code(code)\n documented_code = await self._aask(f\"```python\\n{simplified_code}\\n```\", [system_text])\n documented_code = OutputParser.parse_python_code(documented_code)\n return merge_docstring(code, documented_code)\n\n @staticmethod\n async def write_docstring(\n filename: str | Path, overwrite: bool = False, style: Literal[\"google\", \"numpy\", \"sphinx\"] = \"google\"\n ) -> str:\n data = await aread(str(filename))\n code = await WriteDocstring().run(data, style=style)\n if overwrite:\n await awrite(filename, code)\n return code\n\n\ndef _simplify_python_code(code: str) -> None:\n \"\"\"Simplifies the given Python code by removing expressions and the last if statement.\n\n Args:\n code: A string of Python code.\n\n Returns:\n The simplified Python code.\n \"\"\"\n code_tree = ast.parse(code)\n code_tree.body = [i for i in code_tree.body if not isinstance(i, ast.Expr)]\n if isinstance(code_tree.body[-1], ast.If):\n code_tree.body.pop()\n return ast.unparse(code_tree)\n\n\nif __name__ == \"__main__\":\n import fire\n\n fire.Fire(WriteDocstring.write_docstring)\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant User\n participant Script\n participant Action\n participant OutputParser\n participant ast\n participant fire\n\n User->>Script: Run script with arguments\n Script->>Action: Call run() method\n Action->>OutputParser: Parse system text\n Action->>ast: Parse code into AST\n Action->>Action: Simplify code\n Action->>Action: Generate system text\n Action->>OutputParser: Parse documented code\n Action->>Action: Merge docstrings\n Action->>Script: Return code with docstrings\n Script->>fire: Call write_docstring() method\n fire->>Action: Call write_docstring() method\n Action->>OutputParser: Parse code from file\n Action->>Action: Run run() method\n Action->>Action: Write docstrings\n Action->>OutputParser: Parse code with docstrings\n Action->>Script: Return code with docstrings\n Script->>User: Return code with docstrings\n```\n```", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Author : alexanderwu\n@File : write_review.py\n\"\"\"\nimport asyncio\nfrom typing import List\n\nfrom metagpt.actions import Action\nfrom metagpt.actions.action_node import ActionNode\n\nREVIEW = ActionNode(\n key=\"Review\",\n expected_type=List[str],\n instruction=\"Act as an experienced reviewer and critically assess the given output. Provide specific and\"\n \" constructive feedback, highlighting areas for improvement and suggesting changes.\",\n example=[\n \"The logic in the function `calculate_total` seems flawed. Shouldn't it consider the discount rate as well?\",\n \"The TODO function is not implemented yet? Should we implement it before commit?\",\n ],\n)\n\nLGTM = ActionNode(\n key=\"LGTM\",\n expected_type=str,\n instruction=\"LGTM/LBTM. If the code is fully implemented, \"\n \"give a LGTM (Looks Good To Me), otherwise provide a LBTM (Looks Bad To Me).\",\n example=\"LBTM\",\n)\n\nACTIONS = ActionNode(\n key=\"Actions\",\n expected_type=str,\n instruction=\"Based on the code review outcome, suggest actionable steps. This can include code changes, \"\n \"refactoring suggestions, or any follow-up tasks.\",\n example=\"\"\"1. Refactor the `process_data` method to improve readability and efficiency.\n2. Cover edge cases in the `validate_user` function.\n3. Implement a the TODO in the `calculate_total` function.\n4. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n\"\"\",\n)\n\nWRITE_DRAFT = ActionNode(\n key=\"WriteDraft\",\n expected_type=str,\n instruction=\"Could you write draft code for move function in order to implement it?\",\n example=\"Draft: ...\",\n)\n\n\nWRITE_MOVE_FUNCTION = ActionNode(\n key=\"WriteFunction\",\n expected_type=str,\n instruction=\"write code for the function not implemented.\",\n example=\"\"\"\n```Code\n...\n```\n\"\"\",\n)\n\n\nREWRITE_CODE = ActionNode(\n key=\"RewriteCode\",\n expected_type=str,\n instruction=\"\"\"rewrite code based on the Review and Actions\"\"\",\n example=\"\"\"\n```python\n## example.py\ndef calculate_total(price, quantity):\n total = price * quantity\n```\n\"\"\",\n)\n\n\nCODE_REVIEW_CONTEXT = \"\"\"\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\n\n# Context\n## System Design\n{\"Implementation approach\": \"我们将使用HTML、CSS和JavaScript来实现这个单机的响应式2048游戏。为了确保游戏性能流畅和响应式设计,我们会选择使用Vue.js框架,因为它易于上手且适合构建交互式界面。我们还将使用localStorage来记录玩家的最高分。\", \"File list\": [\"index.html\", \"styles.css\", \"main.js\", \"game.js\", \"storage.js\"], \"Data structures and interfaces\": \"classDiagram\\\n class Game {\\\n -board Array\\\n -score Number\\\n -bestScore Number\\\n +constructor()\\\n +startGame()\\\n +move(direction: String)\\\n +getBoard() Array\\\n +getScore() Number\\\n +getBestScore() Number\\\n +setBestScore(score: Number)\\\n }\\\n class Storage {\\\n +getBestScore() Number\\\n +setBestScore(score: Number)\\\n }\\\n class Main {\\\n +init()\\\n +bindEvents()\\\n }\\\n Game --> Storage : uses\\\n Main --> Game : uses\", \"Program call flow\": \"sequenceDiagram\\\n participant M as Main\\\n participant G as Game\\\n participant S as Storage\\\n M->>G: init()\\\n G->>S: getBestScore()\\\n S-->>G: return bestScore\\\n M->>G: bindEvents()\\\n M->>G: startGame()\\\n loop Game Loop\\\n M->>G: move(direction)\\\n G->>S: setBestScore(score)\\\n S-->>G: return\\\n end\", \"Anything UNCLEAR\": \"目前项目要求明确,没有不清楚的地方。\"}\n\n## Tasks\n{\"Required Python packages\": [\"无需Python包\"], \"Required Other language third-party packages\": [\"vue.js\"], \"Logic Analysis\": [[\"index.html\", \"作为游戏的入口文件和主要的HTML结构\"], [\"styles.css\", \"包含所有的CSS样式,确保游戏界面美观\"], [\"main.js\", \"包含Main类,负责初始化游戏和绑定事件\"], [\"game.js\", \"包含Game类,负责游戏逻辑,如开始游戏、移动方块等\"], [\"storage.js\", \"包含Storage类,用于获取和设置玩家的最高分\"]], \"Task list\": [\"index.html\", \"styles.css\", \"storage.js\", \"game.js\", \"main.js\"], \"Full API spec\": \"\", \"Shared Knowledge\": \"\\'game.js\\' 包含游戏逻辑相关的函数,被 \\'main.js\\' 调用。\", \"Anything UNCLEAR\": \"目前项目要求明确,没有不清楚的地方。\"}\n\n## Code Files\n----- index.html\n\n\n\n \n \n 2048游戏\n \n \n\n\n
\n

2048

\n
\n
\n
分数
\n
{{ score }}
\n
\n
\n
最高分
\n
{{ bestScore }}
\n
\n
\n
\n
\n
\n {{ cell !== 0 ? cell : \\'\\' }}\n
\n
\n
\n \n
\n\n \n \n \n \n\n\n\n----- styles.css\n/* styles.css */\nbody, html {\n margin: 0;\n padding: 0;\n font-family: \\'Arial\\', sans-serif;\n}\n\n#app {\n text-align: center;\n font-size: 18px;\n color: #776e65;\n}\n\nh1 {\n color: #776e65;\n font-size: 72px;\n font-weight: bold;\n margin: 20px 0;\n}\n\n.scores-container {\n display: flex;\n justify-content: center;\n margin-bottom: 20px;\n}\n\n.score-container, .best-container {\n background: #bbada0;\n padding: 10px;\n border-radius: 5px;\n margin: 0 10px;\n min-width: 100px;\n text-align: center;\n}\n\n.score-header, .best-header {\n color: #eee4da;\n font-size: 18px;\n margin-bottom: 5px;\n}\n\n.game-container {\n max-width: 500px;\n margin: 0 auto 20px;\n background: #bbada0;\n padding: 15px;\n border-radius: 10px;\n position: relative;\n}\n\n.grid-row {\n display: flex;\n}\n\n.grid-cell {\n background: #cdc1b4;\n width: 100px;\n height: 100px;\n margin: 5px;\n display: flex;\n justify-content: center;\n align-items: center;\n font-size: 35px;\n font-weight: bold;\n color: #776e65;\n border-radius: 3px;\n}\n\n/* Dynamic classes for different number cells */\n.number-cell-2 {\n background: #eee4da;\n}\n\n.number-cell-4 {\n background: #ede0c8;\n}\n\n.number-cell-8 {\n background: #f2b179;\n color: #f9f6f2;\n}\n\n.number-cell-16 {\n background: #f59563;\n color: #f9f6f2;\n}\n\n.number-cell-32 {\n background: #f67c5f;\n color: #f9f6f2;\n}\n\n.number-cell-64 {\n background: #f65e3b;\n color: #f9f6f2;\n}\n\n.number-cell-128 {\n background: #edcf72;\n color: #f9f6f2;\n}\n\n.number-cell-256 {\n background: #edcc61;\n color: #f9f6f2;\n}\n\n.number-cell-512 {\n background: #edc850;\n color: #f9f6f2;\n}\n\n.number-cell-1024 {\n background: #edc53f;\n color: #f9f6f2;\n}\n\n.number-cell-2048 {\n background: #edc22e;\n color: #f9f6f2;\n}\n\n/* Larger numbers need smaller font sizes */\n.number-cell-1024, .number-cell-2048 {\n font-size: 30px;\n}\n\nbutton {\n background-color: #8f7a66;\n color: #f9f6f2;\n border: none;\n border-radius: 3px;\n padding: 10px 20px;\n font-size: 18px;\n cursor: pointer;\n outline: none;\n}\n\nbutton:hover {\n background-color: #9f8b76;\n}\n\n----- storage.js\n## storage.js\nclass Storage {\n // 获取最高分\n getBestScore() {\n // 尝试从localStorage中获取最高分,如果不存在则默认为0\n const bestScore = localStorage.getItem(\\'bestScore\\');\n return bestScore ? Number(bestScore) : 0;\n }\n\n // 设置最高分\n setBestScore(score) {\n // 将最高分设置到localStorage中\n localStorage.setItem(\\'bestScore\\', score.toString());\n }\n}\n\n\n\n## Code to be Reviewed: game.js\n```Code\n## game.js\nclass Game {\n constructor() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = 0;\n }\n\n createEmptyBoard() {\n const board = [];\n for (let i = 0; i < 4; i++) {\n board[i] = [0, 0, 0, 0];\n }\n return board;\n }\n\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.addRandomTile();\n this.addRandomTile();\n }\n\n addRandomTile() {\n let emptyCells = [];\n for (let r = 0; r < 4; r++) {\n for (let c = 0; c < 4; c++) {\n if (this.board[r][c] === 0) {\n emptyCells.push({ r, c });\n }\n }\n }\n if (emptyCells.length > 0) {\n let randomCell = emptyCells[Math.floor(Math.random() * emptyCells.length)];\n this.board[randomCell.r][randomCell.c] = Math.random() < 0.9 ? 2 : 4;\n }\n }\n\n move(direction) {\n // This function will handle the logic for moving tiles\n // in the specified direction and merging them\n // It will also update the score and add a new random tile if the move is successful\n // The actual implementation of this function is complex and would require\n // a significant amount of code to handle all the cases for moving and merging tiles\n // For the purposes of this example, we will not implement the full logic\n // Instead, we will just call addRandomTile to simulate a move\n this.addRandomTile();\n }\n\n getBoard() {\n return this.board;\n }\n\n getScore() {\n return this.score;\n }\n\n getBestScore() {\n return this.bestScore;\n }\n\n setBestScore(score) {\n this.bestScore = score;\n }\n}\n\n```\n\"\"\"\n\n\nCODE_REVIEW_SMALLEST_CONTEXT = \"\"\"\n## Code to be Reviewed: game.js\n```Code\n// game.js\nclass Game {\n constructor() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = 0;\n }\n\n createEmptyBoard() {\n const board = [];\n for (let i = 0; i < 4; i++) {\n board[i] = [0, 0, 0, 0];\n }\n return board;\n }\n\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.addRandomTile();\n this.addRandomTile();\n }\n\n addRandomTile() {\n let emptyCells = [];\n for (let r = 0; r < 4; r++) {\n for (let c = 0; c < 4; c++) {\n if (this.board[r][c] === 0) {\n emptyCells.push({ r, c });\n }\n }\n }\n if (emptyCells.length > 0) {\n let randomCell = emptyCells[Math.floor(Math.random() * emptyCells.length)];\n this.board[randomCell.r][randomCell.c] = Math.random() < 0.9 ? 2 : 4;\n }\n }\n\n move(direction) {\n // This function will handle the logic for moving tiles\n // in the specified direction and merging them\n // It will also update the score and add a new random tile if the move is successful\n // The actual implementation of this function is complex and would require\n // a significant amount of code to handle all the cases for moving and merging tiles\n // For the purposes of this example, we will not implement the full logic\n // Instead, we will just call addRandomTile to simulate a move\n this.addRandomTile();\n }\n\n getBoard() {\n return this.board;\n }\n\n getScore() {\n return this.score;\n }\n\n getBestScore() {\n return this.bestScore;\n }\n\n setBestScore(score) {\n this.bestScore = score;\n }\n}\n\n```\n\"\"\"\n\n\nCODE_REVIEW_SAMPLE = \"\"\"\n## Code Review: game.js\n1. The code partially implements the requirements. The `Game` class is missing the full implementation of the `move` method, which is crucial for the game\\'s functionality.\n2. The code logic is not completely correct. The `move` method is not implemented, which means the game cannot process player moves.\n3. The existing code follows the \"Data structures and interfaces\" in terms of class structure but lacks full method implementations.\n4. Not all functions are implemented. The `move` method is incomplete and does not handle the logic for moving and merging tiles.\n5. All necessary pre-dependencies seem to be imported since the code does not indicate the need for additional imports.\n6. The methods from other files (such as `Storage`) are not being used in the provided code snippet, but the class structure suggests that they will be used correctly.\n\n## Actions\n1. Implement the `move` method to handle tile movements and merging. This is a complex task that requires careful consideration of the game\\'s rules and logic. Here is a simplified version of how one might begin to implement the `move` method:\n ```javascript\n move(direction) {\n // Simplified logic for moving tiles up\n if (direction === \\'up\\') {\n for (let col = 0; col < 4; col++) {\n let tiles = this.board.map(row => row[col]).filter(val => val !== 0);\n let merged = [];\n for (let i = 0; i < tiles.length; i++) {\n if (tiles[i] === tiles[i + 1]) {\n tiles[i] *= 2;\n this.score += tiles[i];\n tiles[i + 1] = 0;\n merged.push(i);\n }\n }\n tiles = tiles.filter(val => val !== 0);\n while (tiles.length < 4) {\n tiles.push(0);\n }\n for (let row = 0; row < 4; row++) {\n this.board[row][col] = tiles[row];\n }\n }\n }\n // Additional logic needed for \\'down\\', \\'left\\', \\'right\\'\n // ...\n this.addRandomTile();\n }\n ```\n2. Integrate the `Storage` class methods to handle the best score. This means updating the `startGame` and `setBestScore` methods to use `Storage` for retrieving and setting the best score:\n ```javascript\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = new Storage().getBestScore(); // Retrieve the best score from storage\n this.addRandomTile();\n this.addRandomTile();\n }\n\n setBestScore(score) {\n if (score > this.bestScore) {\n this.bestScore = score;\n new Storage().setBestScore(score); // Set the new best score in storage\n }\n }\n ```\n\n## Code Review Result\nLBTM\n\n```\n\"\"\"\n\n\nWRITE_CODE_NODE = ActionNode.from_children(\"WRITE_REVIEW_NODE\", [REVIEW, LGTM, ACTIONS])\nWRITE_MOVE_NODE = ActionNode.from_children(\"WRITE_MOVE_NODE\", [WRITE_DRAFT, WRITE_MOVE_FUNCTION])\n\n\nCR_FOR_MOVE_FUNCTION_BY_3 = \"\"\"\nThe move function implementation provided appears to be well-structured and follows a clear logic for moving and merging tiles in the specified direction. However, there are a few potential improvements that could be made to enhance the code:\n\n1. Encapsulation: The logic for moving and merging tiles could be encapsulated into smaller, reusable functions to improve readability and maintainability.\n\n2. Magic Numbers: There are some magic numbers (e.g., 4, 3) used in the loops that could be replaced with named constants for improved readability and easier maintenance.\n\n3. Comments: Adding comments to explain the logic and purpose of each section of the code can improve understanding for future developers who may need to work on or maintain the code.\n\n4. Error Handling: It's important to consider error handling for unexpected input or edge cases to ensure the function behaves as expected in all scenarios.\n\nOverall, the code could benefit from refactoring to improve readability, maintainability, and extensibility. If you would like, I can provide a refactored version of the move function that addresses these considerations.\n\"\"\"\n\n\nclass WriteCodeAN(Action):\n \"\"\"Write a code review for the context.\"\"\"\n\n async def run(self, context):\n self.llm.system_prompt = \"You are an outstanding engineer and can implement any code\"\n return await WRITE_MOVE_FUNCTION.fill(context=context, llm=self.llm, schema=\"json\")\n # return await WRITE_CODE_NODE.fill(context=context, llm=self.llm, schema=\"markdown\")\n\n\nasync def main():\n await WriteCodeAN().run(CODE_REVIEW_SMALLEST_CONTEXT)\n\n\nif __name__ == \"__main__\":\n asyncio.run(main())\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant User\n participant CodeReview\n participant WriteCodeAN\n participant WRITE_MOVE_FUNCTION\n\n User->>CodeReview: Request code review\n CodeReview->>WriteCodeAN: Run WriteCodeAN action\n WriteCodeAN->>WRITE_MOVE_FUNCTION: Fill WRITE_MOVE_FUNCTION node\n WRITE_MOVE_FUNCTION-->>WriteCodeAN: Return filled node\n WriteCodeAN-->>CodeReview: Return filled node\n CodeReview-->>User: Return code review result\n```\n", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/12/14 15:28\n@Author : alexanderwu\n@File : project_management_an.py\n\"\"\"\nfrom typing import List\n\nfrom metagpt.actions.action_node import ActionNode\nfrom metagpt.logs import logger\n\nREQUIRED_PYTHON_PACKAGES = ActionNode(\n key=\"Required Python packages\",\n expected_type=List[str],\n instruction=\"Provide required Python packages in requirements.txt format.\",\n example=[\"flask==1.1.2\", \"bcrypt==3.2.0\"],\n)\n\nREQUIRED_OTHER_LANGUAGE_PACKAGES = ActionNode(\n key=\"Required Other language third-party packages\",\n expected_type=List[str],\n instruction=\"List down the required packages for languages other than Python.\",\n example=[\"No third-party dependencies required\"],\n)\n\nLOGIC_ANALYSIS = ActionNode(\n key=\"Logic Analysis\",\n expected_type=List[List[str]],\n instruction=\"Provide a list of files with the classes/methods/functions to be implemented, \"\n \"including dependency analysis and imports.\",\n example=[\n [\"game.py\", \"Contains Game class and ... functions\"],\n [\"main.py\", \"Contains main function, from game import Game\"],\n ],\n)\n\nTASK_LIST = ActionNode(\n key=\"Task list\",\n expected_type=List[str],\n instruction=\"Break down the tasks into a list of filenames, prioritized by dependency order.\",\n example=[\"game.py\", \"main.py\"],\n)\n\nFULL_API_SPEC = ActionNode(\n key=\"Full API spec\",\n expected_type=str,\n instruction=\"Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end \"\n \"and back-end communication is not required, leave it blank.\",\n example=\"openapi: 3.0.0 ...\",\n)\n\nSHARED_KNOWLEDGE = ActionNode(\n key=\"Shared Knowledge\",\n expected_type=str,\n instruction=\"Detail any shared knowledge, like common utility functions or configuration variables.\",\n example=\"'game.py' contains functions shared across the project.\",\n)\n\nANYTHING_UNCLEAR_PM = ActionNode(\n key=\"Anything UNCLEAR\",\n expected_type=str,\n instruction=\"Mention any unclear aspects in the project management context and try to clarify them.\",\n example=\"Clarification needed on how to start and initialize third-party libraries.\",\n)\n\nNODES = [\n REQUIRED_PYTHON_PACKAGES,\n REQUIRED_OTHER_LANGUAGE_PACKAGES,\n LOGIC_ANALYSIS,\n TASK_LIST,\n FULL_API_SPEC,\n SHARED_KNOWLEDGE,\n ANYTHING_UNCLEAR_PM,\n]\n\n\nPM_NODE = ActionNode.from_children(\"PM_NODE\", NODES)\n\n\ndef main():\n prompt = PM_NODE.compile(context=\"\")\n logger.info(prompt)\n\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant ActionNode\n participant PM_NODE\n participant main\n\n main->>PM_NODE: compile(context=\"\")\n PM_NODE->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>ActionNode: compile(context=\"\")\n ActionNode->>Action", + "You are a function parser. You can convert spoken words into function parameters.\n\n---\ntext_to_image function parameters description:\nparameter `text`: The text used for image conversion.\nparameter `size_type`: size type\n\n---\nExamples:\nIf want you to do `Draw a girl`, return `text_to_image(text=\"Draw a girl\", size_type=\"512x512\")` brief and clear.\nIf want you to do `Draw an apple`, return `text_to_image(text=\"Draw an apple\", size_type=\"512x512\")` brief and clear.\n\n---\n\nRefer to the `text_to_image` function description, and fill in the function parameters according to the example \"I want you to do xx\" in the Examples section.\nNow I want you to do `Draw an apple`, return function parameters in Examples format above, brief and clear.": "`text_to_image(text=\"Draw an apple\", size_type=\"512x512\")`", + "\nNOTICE\nRole: You are a professional software engineer, and your main task is to review the code.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n-----\n# System Design\n```text\n\n{\"Implementation approach\": \"To develop this snake game, we will use the Python language and choose the Pygame library. Pygame is an open-source Python module collection specifically designed for writing video games. It provides functionalities such as displaying images and playing sounds, making it suitable for creating intuitive and responsive user interfaces. We will ensure efficient game logic to prevent any delays during gameplay. The scoring system will be simple, with the snake gaining points for each food it eats. We will use Pygame's event handling system to implement pause and resume functionality, as well as high-score tracking. The difficulty will increase by speeding up the snake's movement. In the initial version, we will focus on single-player mode and consider adding multiplayer mode and customizable skins in future updates. Based on the new requirement, we will also add a moving obstacle that appears randomly. If the snake eats this obstacle, the game will end. If the snake does not eat the obstacle, it will disappear after 5 seconds. For this, we need to add mechanisms for obstacle generation, movement, and disappearance in the game logic.\", \"Project_name\": \"snake_game\", \"File list\": [\"main.py\", \"game.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"constants.py\", \"assets/styles.css\", \"assets/index.html\"], \"Data structures and interfaces\": \"```mermaid\n classDiagram\n class Game{\n +int score\n +int speed\n +bool game_over\n +bool paused\n +Snake snake\n +Food food\n +Obstacle obstacle\n +Scoreboard scoreboard\n +start_game() void\n +pause_game() void\n +resume_game() void\n +end_game() void\n +increase_difficulty() void\n +update() void\n +render() void\n Game()\n }\n class Snake{\n +list body_parts\n +str direction\n +bool grow\n +move() void\n +grow() void\n +check_collision() bool\n Snake()\n }\n class Food{\n +tuple position\n +spawn() void\n Food()\n }\n class Obstacle{\n +tuple position\n +int lifetime\n +bool active\n +spawn() void\n +move() void\n +check_collision() bool\n +disappear() void\n Obstacle()\n }\n class Scoreboard{\n +int high_score\n +update_score(int) void\n +reset_score() void\n +load_high_score() void\n +save_high_score() void\n Scoreboard()\n }\n class Constants{\n }\n Game \"1\" -- \"1\" Snake: has\n Game \"1\" -- \"1\" Food: has\n Game \"1\" -- \"1\" Obstacle: has\n Game \"1\" -- \"1\" Scoreboard: has\n ```\", \"Program call flow\": \"```sequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n participant O as Obstacle\n participant SB as Scoreboard\n M->>G: start_game()\n loop game loop\n G->>S: move()\n G->>S: check_collision()\n G->>F: spawn()\n G->>O: spawn()\n G->>O: move()\n G->>O: check_collision()\n G->>O: disappear()\n G->>SB: update_score(score)\n G->>G: update()\n G->>G: render()\n alt if paused\n M->>G: pause_game()\n M->>G: resume_game()\n end\n alt if game_over\n G->>M: end_game()\n end\n end\n```\", \"Anything UNCLEAR\": \"There is no need for further clarification as the requirements are already clear.\"}\n\n```\n-----\n# Tasks\n```text\n\n{\"Required Python third-party packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party packages required for other languages.\"], \"Full API spec\": \"\n openapi: 3.0.0\n info:\n title: Snake Game API\n version: \"1.0.0\"\n paths:\n /start:\n get:\n summary: Start the game\n responses:\n '200':\n description: Game started successfully\n /pause:\n get:\n summary: Pause the game\n responses:\n '200':\n description: Game paused successfully\n /resume:\n get:\n summary: Resume the game\n responses:\n '200':\n description: Game resumed successfully\n /end:\n get:\n summary: End the game\n responses:\n '200':\n description: Game ended successfully\n /score:\n get:\n summary: Get the current score\n responses:\n '200':\n description: Current score retrieved successfully\n /highscore:\n get:\n summary: Get the high score\n responses:\n '200':\n description: High score retrieved successfully\n components: {}\n \", \"Logic Analysis\": [[\"constants.py\", \"Contains all the constant values like screen size, colors, game speeds, etc. This should be implemented first as it provides the base values for other components.\"], [\"snake.py\", \"Contains the Snake class with methods for movement, growth, and collision detection. It is dependent on constants.py for configuration values.\"], [\"food.py\", \"Contains the Food class responsible for spawning food items on the screen. It is dependent on constants.py for configuration values.\"], [\"obstacle.py\", \"Contains the Obstacle class with methods for spawning, moving, and disappearing of obstacles, as well as collision detection with the snake. It is dependent on constants.py for configuration values.\"], [\"scoreboard.py\", \"Contains the Scoreboard class for updating, resetting, loading, and saving high scores. It may use constants.py for configuration values and depends on the game's scoring logic.\"], [\"game.py\", \"Contains the main Game class which includes the game loop and methods for starting, pausing, resuming, and ending the game. It is dependent on snake.py, food.py, obstacle.py, and scoreboard.py.\"], [\"main.py\", \"The entry point of the game that initializes the game and starts the game loop. It is dependent on game.py.\"]], \"Task list\": [\"constants.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"game.py\", \"main.py\"], \"Shared Knowledge\": \"\n 'constants.py' should contain all the necessary configurations for the game, such as screen dimensions, color definitions, and speed settings. These constants will be used across multiple files, ensuring consistency and ease of updates. Ensure that the Pygame library is initialized correctly in 'main.py' before starting the game loop. Also, make sure that the game's state is managed properly when pausing and resuming the game.\n \", \"Anything UNCLEAR\": \"The interaction between the 'obstacle.py' and the game loop needs to be clearly defined to ensure obstacles appear and disappear correctly. The lifetime of the obstacle and its random movement should be implemented in a way that does not interfere with the game's performance.\"}\n\n```\n-----\n```python\n\n## game.py\nimport pygame\nfrom snake import Snake\nfrom food import Food\n\nclass Game:\n def __init__(self):\n self.score = 0\n self.level = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n pygame.init()\n self.initialize_game()\n self.game_loop()\n\n def initialize_game(self):\n self.score = 0\n self.level = 1\n self.snake.reset()\n self.food.generate()\n\n def game_loop(self):\n game_over = False\n\n while not game_over:\n self.update()\n self.draw()\n self.handle_events()\n self.check_collision()\n self.increase_score()\n self.increase_level()\n\n if self.snake.is_collision():\n game_over = True\n self.game_over()\n\n def update(self):\n self.snake.move()\n\n def draw(self):\n self.snake.draw()\n self.food.draw()\n\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n pygame.quit()\n quit()\n elif event.type == pygame.KEYDOWN:\n if event.key == pygame.K_UP:\n self.snake.change_direction(\"UP\")\n elif event.key == pygame.K_DOWN:\n self.snake.change_direction(\"DOWN\")\n elif event.key == pygame.K_LEFT:\n self.snake.change_direction(\"LEFT\")\n elif event.key == pygame.K_RIGHT:\n self.snake.change_direction(\"RIGHT\")\n\n def check_collision(self):\n if self.snake.get_head() == self.food.get_position():\n self.snake.grow()\n self.food.generate()\n\n def increase_score(self):\n self.score += 1\n\n def increase_level(self):\n if self.score % 10 == 0:\n self.level += 1\n\n def game_over(self):\n print(\"Game Over\")\n self.initialize_game()\n\n\n```\n-----\n```python\n\n## snake.py\nimport pygame\n\nclass Snake:\n def __init__(self):\n self.body = [(0, 0)]\n self.direction = (1, 0)\n\n def move(self):\n head = self.body[0]\n dx, dy = self.direction\n new_head = (head[0] + dx, head[1] + dy)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction):\n if direction == \"UP\":\n self.direction = (0, -1)\n elif direction == \"DOWN\":\n self.direction = (0, 1)\n elif direction == \"LEFT\":\n self.direction = (-1, 0)\n elif direction == \"RIGHT\":\n self.direction = (1, 0)\n\n def grow(self):\n tail = self.body[-1]\n dx, dy = self.direction\n new_tail = (tail[0] - dx, tail[1] - dy)\n self.body.append(new_tail)\n\n def get_head(self):\n return self.body[0]\n\n def get_body(self):\n return self.body[1:]\n\n\n```\n-----\n```python\n\n## main.py\nimport pygame\nfrom game import Game\n\ndef main():\n pygame.init()\n game = Game()\n game.start_game()\n\nif __name__ == \"__main__\":\n main()\n\n\n```\n-----\n```python\n\n## food.py\nimport random\n\nclass Food:\n def __init__(self):\n self.position = (0, 0)\n\n def generate(self):\n x = random.randint(0, 9)\n y = random.randint(0, 9)\n self.position = (x, y)\n\n def get_position(self):\n return self.position\n\n\n```\n-----\n\n## Code Review All: Please read all historical files and find possible bugs in the files, such as unimplemented functions, calling errors, unreferences, etc.\n\n## Call flow: mermaid code, based on the implemented function, use mermaid to draw a complete call chain\n\n## Summary: Summary based on the implementation of historical files\n\n## TODOs: Python dict[str, str], write down the list of files that need to be modified and the reasons. We will modify them later.\n\n": "## Code Review\n\n### game.py\n- The `game_over` variable is not being used in the `game_loop` method. It can be removed.\n- The `increase_score` method should only increase the score if there is a collision with the food. Move the score increment logic inside the `check_collision` method.\n- The `increase_level` method should only increase the level if the score is divisible by 10. Move the level increment logic inside the `check_collision` method.\n- The `game_over` method should be called after the game loop ends, not inside the loop.\n\n### snake.py\n- The `get_body` method is not being used. It can be removed.\n\n### main.py\n- The `pygame.init()` method is being called twice, once in the `main` function and once in the `start_game` method of the `Game` class. It should only be called once, preferably in the `main` function.\n\n### food.py\n- The `generate` method should generate random positions within the game boundaries defined in the `constants.py` file.\n\n## Call Flow\n\n```mermaid\nsequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n\n M->>G: start_game()\n G->>G: initialize_game()\n G->>G: game_loop()\n loop game loop\n G->>S: move()\n G->>G: update()\n G->>S: draw()\n G->>F: draw()\n G->>G: handle_events()\n G->>G: check_collision()\n G->>G: increase_score()\n G->>G: increase_level()\n alt if snake collision\n G->>G: game_over()\n end\n end\n```\n\n## Summary\nThe code consists of several classes that work together to create a snake game. The `Game` class manages the game loop and handles events. The `Snake` class represents the snake and handles its movement and collision detection. The `Food` class represents the food and handles its generation and position. The code is missing some functionality, such as updating the score and level, and handling game over conditions. There are also some minor issues with the code structure and logic that need to be addressed.\n\n## TODOs\n- Modify `game.py`:\n - Remove the unused `game_over` variable in the `game_loop` method.\n - Move the score increment logic from the `increase_score` method to the `check_collision` method.\n - Move the level increment logic from the `increase_level` method to the `check_collision` method.\n - Call the `game_over` method after the game loop ends.\n- Modify `snake.py`:\n - Remove the unused `get_body` method.\n- Modify `main.py`:\n - Remove the redundant `pygame.init()` method call in the `start_game` method of the `Game` class.\n- Modify `food.py`:\n - Generate random positions within the game boundaries defined in the `constants.py` file.", + "\nNOTICE\nRole: You are a professional software engineer, and your main task is to review the code.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n-----\n# System Design\n```text\n\n{\"Implementation approach\": \"To develop this snake game, we will use the Python language and choose the Pygame library. Pygame is an open-source Python module collection specifically designed for writing video games. It provides functionalities such as displaying images and playing sounds, making it suitable for creating intuitive and responsive user interfaces. We will ensure efficient game logic to prevent any delays during gameplay. The scoring system will be simple, with the snake gaining points for each food it eats. We will use Pygame's event handling system to implement pause and resume functionality, as well as high-score tracking. The difficulty will increase by speeding up the snake's movement. In the initial version, we will focus on single-player mode and consider adding multiplayer mode and customizable skins in future updates. Based on the new requirement, we will also add a moving obstacle that appears randomly. If the snake eats this obstacle, the game will end. If the snake does not eat the obstacle, it will disappear after 5 seconds. For this, we need to add mechanisms for obstacle generation, movement, and disappearance in the game logic.\", \"Project_name\": \"snake_game\", \"File list\": [\"main.py\", \"game.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"constants.py\", \"assets/styles.css\", \"assets/index.html\"], \"Data structures and interfaces\": \"```mermaid\n classDiagram\n class Game{\n +int score\n +int speed\n +bool game_over\n +bool paused\n +Snake snake\n +Food food\n +Obstacle obstacle\n +Scoreboard scoreboard\n +start_game() void\n +pause_game() void\n +resume_game() void\n +end_game() void\n +increase_difficulty() void\n +update() void\n +render() void\n Game()\n }\n class Snake{\n +list body_parts\n +str direction\n +bool grow\n +move() void\n +grow() void\n +check_collision() bool\n Snake()\n }\n class Food{\n +tuple position\n +spawn() void\n Food()\n }\n class Obstacle{\n +tuple position\n +int lifetime\n +bool active\n +spawn() void\n +move() void\n +check_collision() bool\n +disappear() void\n Obstacle()\n }\n class Scoreboard{\n +int high_score\n +update_score(int) void\n +reset_score() void\n +load_high_score() void\n +save_high_score() void\n Scoreboard()\n }\n class Constants{\n }\n Game \"1\" -- \"1\" Snake: has\n Game \"1\" -- \"1\" Food: has\n Game \"1\" -- \"1\" Obstacle: has\n Game \"1\" -- \"1\" Scoreboard: has\n ```\", \"Program call flow\": \"```sequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n participant O as Obstacle\n participant SB as Scoreboard\n M->>G: start_game()\n loop game loop\n G->>S: move()\n G->>S: check_collision()\n G->>F: spawn()\n G->>O: spawn()\n G->>O: move()\n G->>O: check_collision()\n G->>O: disappear()\n G->>SB: update_score(score)\n G->>G: update()\n G->>G: render()\n alt if paused\n M->>G: pause_game()\n M->>G: resume_game()\n end\n alt if game_over\n G->>M: end_game()\n end\n end\n```\", \"Anything UNCLEAR\": \"There is no need for further clarification as the requirements are already clear.\"}\n\n```\n-----\n# Tasks\n```text\n\n{\"Required Python third-party packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party packages required for other languages.\"], \"Full API spec\": \"\n openapi: 3.0.0\n info:\n title: Snake Game API\n version: \"1.0.0\"\n paths:\n /start:\n get:\n summary: Start the game\n responses:\n '200':\n description: Game started successfully\n /pause:\n get:\n summary: Pause the game\n responses:\n '200':\n description: Game paused successfully\n /resume:\n get:\n summary: Resume the game\n responses:\n '200':\n description: Game resumed successfully\n /end:\n get:\n summary: End the game\n responses:\n '200':\n description: Game ended successfully\n /score:\n get:\n summary: Get the current score\n responses:\n '200':\n description: Current score retrieved successfully\n /highscore:\n get:\n summary: Get the high score\n responses:\n '200':\n description: High score retrieved successfully\n components: {}\n \", \"Logic Analysis\": [[\"constants.py\", \"Contains all the constant values like screen size, colors, game speeds, etc. This should be implemented first as it provides the base values for other components.\"], [\"snake.py\", \"Contains the Snake class with methods for movement, growth, and collision detection. It is dependent on constants.py for configuration values.\"], [\"food.py\", \"Contains the Food class responsible for spawning food items on the screen. It is dependent on constants.py for configuration values.\"], [\"obstacle.py\", \"Contains the Obstacle class with methods for spawning, moving, and disappearing of obstacles, as well as collision detection with the snake. It is dependent on constants.py for configuration values.\"], [\"scoreboard.py\", \"Contains the Scoreboard class for updating, resetting, loading, and saving high scores. It may use constants.py for configuration values and depends on the game's scoring logic.\"], [\"game.py\", \"Contains the main Game class which includes the game loop and methods for starting, pausing, resuming, and ending the game. It is dependent on snake.py, food.py, obstacle.py, and scoreboard.py.\"], [\"main.py\", \"The entry point of the game that initializes the game and starts the game loop. It is dependent on game.py.\"]], \"Task list\": [\"constants.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"game.py\", \"main.py\"], \"Shared Knowledge\": \"\n 'constants.py' should contain all the necessary configurations for the game, such as screen dimensions, color definitions, and speed settings. These constants will be used across multiple files, ensuring consistency and ease of updates. Ensure that the Pygame library is initialized correctly in 'main.py' before starting the game loop. Also, make sure that the game's state is managed properly when pausing and resuming the game.\n \", \"Anything UNCLEAR\": \"The interaction between the 'obstacle.py' and the game loop needs to be clearly defined to ensure obstacles appear and disappear correctly. The lifetime of the obstacle and its random movement should be implemented in a way that does not interfere with the game's performance.\"}\n\n```\n-----\n```python\n\n## food.py\nimport random\n\nclass Food:\n def __init__(self):\n self.position = (0, 0)\n\n def generate(self):\n x = random.randint(0, 9)\n y = random.randint(0, 9)\n self.position = (x, y)\n\n def get_position(self):\n return self.position\n\n\n```\n-----\n```python\n\n## snake.py\nimport pygame\n\nclass Snake:\n def __init__(self):\n self.body = [(0, 0)]\n self.direction = (1, 0)\n\n def move(self):\n head = self.body[0]\n dx, dy = self.direction\n new_head = (head[0] + dx, head[1] + dy)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction):\n if direction == \"UP\":\n self.direction = (0, -1)\n elif direction == \"DOWN\":\n self.direction = (0, 1)\n elif direction == \"LEFT\":\n self.direction = (-1, 0)\n elif direction == \"RIGHT\":\n self.direction = (1, 0)\n\n def grow(self):\n tail = self.body[-1]\n dx, dy = self.direction\n new_tail = (tail[0] - dx, tail[1] - dy)\n self.body.append(new_tail)\n\n def get_head(self):\n return self.body[0]\n\n def get_body(self):\n return self.body[1:]\n\n\n```\n-----\n```python\n\n## game.py\nimport pygame\nfrom snake import Snake\nfrom food import Food\n\nclass Game:\n def __init__(self):\n self.score = 0\n self.level = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n pygame.init()\n self.initialize_game()\n self.game_loop()\n\n def initialize_game(self):\n self.score = 0\n self.level = 1\n self.snake.reset()\n self.food.generate()\n\n def game_loop(self):\n game_over = False\n\n while not game_over:\n self.update()\n self.draw()\n self.handle_events()\n self.check_collision()\n self.increase_score()\n self.increase_level()\n\n if self.snake.is_collision():\n game_over = True\n self.game_over()\n\n def update(self):\n self.snake.move()\n\n def draw(self):\n self.snake.draw()\n self.food.draw()\n\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n pygame.quit()\n quit()\n elif event.type == pygame.KEYDOWN:\n if event.key == pygame.K_UP:\n self.snake.change_direction(\"UP\")\n elif event.key == pygame.K_DOWN:\n self.snake.change_direction(\"DOWN\")\n elif event.key == pygame.K_LEFT:\n self.snake.change_direction(\"LEFT\")\n elif event.key == pygame.K_RIGHT:\n self.snake.change_direction(\"RIGHT\")\n\n def check_collision(self):\n if self.snake.get_head() == self.food.get_position():\n self.snake.grow()\n self.food.generate()\n\n def increase_score(self):\n self.score += 1\n\n def increase_level(self):\n if self.score % 10 == 0:\n self.level += 1\n\n def game_over(self):\n print(\"Game Over\")\n self.initialize_game()\n\n\n```\n-----\n```python\n\n## main.py\nimport pygame\nfrom game import Game\n\ndef main():\n pygame.init()\n game = Game()\n game.start_game()\n\nif __name__ == \"__main__\":\n main()\n\n\n```\n-----\n\n## Code Review All: Please read all historical files and find possible bugs in the files, such as unimplemented functions, calling errors, unreferences, etc.\n\n## Call flow: mermaid code, based on the implemented function, use mermaid to draw a complete call chain\n\n## Summary: Summary based on the implementation of historical files\n\n## TODOs: Python dict[str, str], write down the list of files that need to be modified and the reasons. We will modify them later.\n\n": "## Code Review\n\n### food.py\n- The `generate` method should be renamed to `spawn` to be consistent with the naming convention used in other classes.\n- The `get_position` method can be simplified by directly returning the `self.position` attribute.\n\n### snake.py\n- The `change_direction` method should handle lowercase direction inputs as well to make it more flexible.\n- The `grow` method can be optimized by directly appending the new tail position to the `self.body` list instead of calculating it based on the current direction.\n\n### game.py\n- The `initialize_game` method should call the `reset` method of the `self.snake` object instead of a non-existent `reset` method.\n- The `game_over` method should reset the game state by calling the `initialize_game` method instead of manually resetting the score and level attributes.\n- The `draw` method should be implemented to draw the snake and food on the screen using the Pygame library.\n- The `handle_events` method should handle the case when the snake is moving in one direction and the opposite direction key is pressed, which should be ignored to prevent the snake from reversing.\n- The `check_collision` method should be implemented to check for collisions between the snake's head and the food, and handle the snake's growth and food respawn accordingly.\n- The `increase_score` and `increase_level` methods should be implemented to increase the score and level based on the game's scoring logic.\n- The `game_over` method should be implemented to handle the game over condition, display a game over message, and reset the game state.\n\n### main.py\n- The `pygame.init()` method is called twice, once in the `main` function and once in the `Game` class. It should only be called once, preferably in the `main` function.\n\n## Call Flow\n\n```mermaid\nsequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n\n M->>G: start_game()\n loop game loop\n G->>S: move()\n G->>S: check_collision()\n G->>F: spawn()\n G->>G: update()\n G->>G: draw()\n G->>G: handle_events()\n G->>G: check_collision()\n G->>G: increase_score()\n G->>G: increase_level()\n alt if game over\n G->>G: game_over()\n end\n end\n```\n\n## Summary\n\nThe code consists of several classes that represent different components of the snake game, such as the snake itself, the food, and the game logic. The `main` function initializes the game and starts the game loop. The game loop handles updating the game state, drawing the game objects on the screen, and handling user input. The snake can move in different directions, grow when it eats food, and collide with itself or the boundaries of the screen. The game keeps track of the score and level, and ends when the snake collides with itself or the boundaries.\n\n## TODOs\n\n- Modify `food.py`:\n - Rename the `generate` method to `spawn`.\n - Simplify the `get_position` method.\n- Modify `snake.py`:\n - Handle lowercase direction inputs in the `change_direction` method.\n - Optimize the `grow` method by directly appending the new tail position to the `self.body` list.\n- Modify `game.py`:\n - Call the `reset` method of the `self.snake` object in the `initialize_game` method.\n - Implement the `draw` method to draw the snake and food on the screen.\n - Implement the `handle_events` method to handle direction changes and ignore opposite direction inputs.\n - Implement the `check_collision` method to handle collisions between the snake's head and the food.\n - Implement the `increase_score` and `increase_level` methods based on the game's scoring logic.\n - Implement the `game_over` method to handle the game over condition and reset the game state.\n- Remove the duplicate `pygame.init()` call in the `main` function.", + "You are mathematician.#MSG_SEP#Your responses should align with the role-play agreement, maintaining the character's persona and habits. When faced with unrelated questions, playfully decline to answer without revealing your AI nature to preserve the character's image.#MSG_SEP#If the information is insufficient, you can search in the context or knowledge.#MSG_SEP#Answer the following questions strictly in English, and the answers must follow the Markdown format.#SYSTEM_MSG_END#How old is Susie?": "To determine Susie's age, we can subtract her birthdate from the current date. \n\nSusie was born on November 14, 2011, and today is December 3, 2023. \n\nLet's calculate the age:\n\n```\nCurrent year - Birth year = Age\n2023 - 2011 = 12\n```\n\nTherefore, Susie is currently 12 years old.", + "You are mathematician.#MSG_SEP#Your responses should align with the role-play agreement, maintaining the character's persona and habits. When faced with unrelated questions, playfully decline to answer without revealing your AI nature to preserve the character's image.#MSG_SEP#If the information is insufficient, you can search in the context or knowledge.#MSG_SEP#Answer the following questions strictly in Chinese, and the answers must follow the Markdown format.#SYSTEM_MSG_END#Does Susie have an apple?": "根据提供的信息,Susie曾经有一个苹果,但是她已经吃掉了。", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n设计一个名为'add'的函数,该函数接受两个整数作为输入,并返回它们的和。\n\n## Tasks\n\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: \n```python\n## \n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: . Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code:\n\n```python\ndef add(num1: int, num2: int) -> int:\n \"\"\"\n This function takes two integers as input and returns their sum.\n \n Args:\n num1 (int): The first integer.\n num2 (int): The second integer.\n \n Returns:\n int: The sum of the two integers.\n \"\"\"\n return num1 + num2\n```\n\n# Instruction: Based on the context, write code.\n\n## Code: \n```python\ndef add(num1: int, num2: int) -> int:\n \"\"\"\n This function takes two integers as input and returns their sum.\n \n Args:\n num1 (int): The first integer.\n num2 (int): The second integer.\n \n Returns:\n int: The sum of the two integers.\n \"\"\"\n return num1 + num2\n```", + "\n你是一个工程师。下面是背景信息与你的当前任务,请为任务撰写代码。\n撰写的代码应该符合PEP8,优雅,模块化,易于阅读与维护,代码本身应该有__main__入口来防止桩函数\n\n## 用户编写程序所需的全部、详尽的文件路径列表(只需要相对路径,并不需要前缀,组织形式应该符合PEP规范)\n\n- `main.py`: 主程序文件\n- `search_engine.py`: 搜索引擎实现文件\n- `knowledge_base.py`: 知识库管理文件\n- `user_interface.py`: 用户界面文件\n- `data_import.py`: 数据导入功能文件\n- `data_export.py`: 数据导出功能文件\n- `utils.py`: 工具函数文件\n\n## 数据结构\n\n- `KnowledgeBase`: 知识库类,用于管理私有知识库的内容、分类、标签和关键词。\n- `SearchEngine`: 搜索引擎类,基于大语言模型,用于对用户输入的关键词或短语进行语义理解,并提供准确的搜索结果。\n- `SearchResult`: 搜索结果类,包含与用户搜索意图相关的知识库内容的相关信息。\n- `UserInterface`: 用户界面类,提供简洁、直观的用户界面,支持多种搜索方式和搜索结果的排序和过滤。\n- `DataImporter`: 数据导入类,支持多种数据格式的导入功能,用于将外部数据导入到知识库中。\n- `DataExporter`: 数据导出类,支持多种数据格式的导出功能,用于将知识库内容进行备份和分享。\n\n## API接口\n\n- `KnowledgeBase`类接口:\n - `add_entry(entry: str, category: str, tags: List[str], keywords: List[str]) -> bool`: 添加知识库条目。\n - `delete_entry(entry_id: str) -> bool`: 删除知识库条目。\n - `update_entry(entry_id: str, entry: str, category: str, tags: List[str], keywords: List[str]) -> bool`: 更新知识库条目。\n - `search_entries(query: str) -> List[str]`: 根据查询词搜索知识库条目。\n\n- `SearchEngine`类接口:\n - `search(query: str) -> SearchResult`: 根据用户查询词进行搜索,返回与查询意图相关的搜索结果。\n\n- `UserInterface`类接口:\n - `display_search_results(results: List[SearchResult]) -> None`: 显示搜索结果。\n - `filter_results(results: List[SearchResult], filters: Dict[str, Any]) -> List[SearchResult]`: 根据过滤条件对搜索结果进行过滤。\n - `sort_results(results: List[SearchResult], key: str, reverse: bool = False) -> List[SearchResult]`: 根据指定的键对搜索结果进行排序。\n\n- `DataImporter`类接口:\n - `import_data(file_path: str) -> bool`: 导入外部数据到知识库。\n\n- `DataExporter`类接口:\n - `export_data(file_path: str) -> bool`: 导出知识库数据到外部文件。\n\n## 调用流程(以dot语言描述)\n\n```dot\ndigraph call_flow {\n rankdir=LR;\n\n subgraph cluster_user_program {\n label=\"User Program\";\n style=dotted;\n\n main_py -> search_engine_py;\n main_py -> knowledge_base_py;\n main_py -> user_interface_py;\n main_py -> data_import_py;\n main_py -> data_export_py;\n\n search_engine_py -> knowledge_base_py;\n search_engine_py -> user_interface_py;\n\n user_interface_py -> knowledge_base_py;\n user_interface_py -> search_engine_py;\n\n data_import_py -> knowledge_base_py;\n data_import_py -> user_interface_py;\n\n data_export_py -> knowledge_base_py;\n data_export_py -> user_interface_py;\n }\n\n main_py [label=\"main.py\"];\n search_engine_py [label=\"search_engine.py\"];\n knowledge_base_py [label=\"knowledge_base.py\"];\n user_interface_py [label=\"user_interface.py\"];\n data_import_py [label=\"data_import.py\"];\n data_export_py [label=\"data_export.py\"];\n}\n```\n\n这是一个简化的调用流程图,展示了各个模块之间的调用关系。用户程序的`main.py`文件通过调用其他模块实现搜索引擎的功能。`search_engine.py`模块与`knowledge_base.py`和`user_interface.py`模块进行交互,实现搜索算法和搜索结果的展示。`data_import.py`和`data_export.py`模块与`knowledge_base.py`和`user_interface.py`模块进行交互,实现数据导入和导出的功能。用户界面模块`user_interface.py`与其他模块进行交互,提供简洁、直观的用户界面,并支持搜索方式、排序和过滤等操作。\n\n## 当前任务\n\n\n完成main.py的功能": "根据任务要求,我们需要在`main.py`文件中完成以下功能:\n\n1. 导入所需模块和类:`KnowledgeBase`、`SearchEngine`、`UserInterface`、`DataImporter`、`DataExporter`。\n2. 创建一个`KnowledgeBase`对象,用于管理知识库的内容。\n3. 创建一个`SearchEngine`对象,用于进行搜索操作。\n4. 创建一个`UserInterface`对象,用于与用户进行交互。\n5. 创建一个`DataImporter`对象,用于导入外部数据到知识库。\n6. 创建一个`DataExporter`对象,用于导出知识库数据到外部文件。\n7. 实现主程序逻辑,包括用户界面的显示和用户输入的处理。\n\n下面是一个示例的`main.py`代码实现:\n\n```python\nfrom knowledge_base import KnowledgeBase\nfrom search_engine import SearchEngine\nfrom user_interface import UserInterface\nfrom data_import import DataImporter\nfrom data_export import DataExporter\n\ndef main():\n # 创建知识库对象\n knowledge_base = KnowledgeBase()\n\n # 创建搜索引擎对象\n search_engine = SearchEngine(knowledge_base)\n\n # 创建用户界面对象\n user_interface = UserInterface(search_engine)\n\n # 创建数据导入对象\n data_importer = DataImporter(knowledge_base)\n\n # 创建数据导出对象\n data_exporter = DataExporter(knowledge_base)\n\n # 主程序逻辑\n while True:\n # 显示用户界面\n user_interface.display_menu()\n\n # 获取用户输入\n user_input = input(\"请输入选项:\")\n\n # 处理用户输入\n if user_input == \"1\":\n # 搜索知识库\n query = input(\"请输入搜索关键词:\")\n search_results = search_engine.search(query)\n user_interface.display_search_results(search_results)\n elif user_input == \"2\":\n # 导入数据\n file_path = input(\"请输入要导入的文件路径:\")\n data_importer.import_data(file_path)\n elif user_input == \"3\":\n # 导出数据\n file_path = input(\"请输入要导出的文件路径:\")\n data_exporter.export_data(file_path)\n elif user_input == \"4\":\n # 退出程序\n break\n else:\n print(\"无效的选项,请重新输入。\")\n\nif __name__ == \"__main__\":\n main()\n```\n\n在这个示例代码中,我们首先导入了所需的模块和类。然后,我们创建了`KnowledgeBase`、`SearchEngine`、`UserInterface`、`DataImporter`和`DataExporter`对象。接下来,我们使用一个无限循环来实现主程序逻辑。在每次循环中,我们首先显示用户界面,然后获取用户输入,并根据用户输入执行相应的操作。用户可以选择搜索知识库、导入数据、导出数据或退出程序。最后,我们使用`if __name__ == \"__main__\":`来确保代码在作为主程序运行时才会执行。", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\": \"We will use the Pygame library to create the game interface and handle user input. The game logic will be implemented using Python classes and data structures.\", \"File list\": [\"main.py\", \"game.py\"], \"Data structures and interfaces\": \"classDiagram\\n class Game {\\n -grid: List[List[int]]\\n -score: int\\n -game_over: bool\\n +__init__()\\n +reset_game()\\n +move(direction: str)\\n +is_game_over() bool\\n +get_empty_cells() List[Tuple[int, int]]\\n +add_new_tile()\\n +get_score() int\\n }\\n class UI {\\n -game: Game\\n +__init__(game: Game)\\n +draw_grid()\\n +draw_score()\\n +draw_game_over()\\n +handle_input()\\n }\\n Game --> UI\", \"Program call flow\": \"sequenceDiagram\\n participant M as Main\\n participant G as Game\\n participant U as UI\\n M->>G: reset_game()\\n M->>U: draw_grid()\\n M->>U: draw_score()\\n M->>U: handle_input()\\n U->>G: move(direction)\\n G->>G: add_new_tile()\\n G->>U: draw_grid()\\n G->>U: draw_score()\\n G->>U: draw_game_over()\\n G->>G: is_game_over()\\n G->>G: get_empty_cells()\\n G->>G: get_score()\", \"Anything UNCLEAR\": \"...\"}\n\n## Tasks\n{\"Required Python packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party dependencies required\"], \"Logic Analysis\": [[\"game.py\", \"Contains Game class and related functions for game logic\"], [\"main.py\", \"Contains main function, initializes the game and UI\"]], \"Task list\": [\"game.py\", \"main.py\"], \"Full API spec\": \"\", \"Shared Knowledge\": \"The game logic will be implemented using Python classes and data structures. The Pygame library will be used to create the game interface and handle user input.\", \"Anything UNCLEAR\": \"...\"}\n\n## Legacy Code\n```Code\n----- main.py\nif __name__ == \"__main__\":\nmain()\n```\n\n## Debug logs\n```text\nE.......F\n======================================================================\nERROR: test_add_new_tile (__main__.TestGame)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n File \"/Users/xx/tests/test_game.py\", line 104, in test_add_new_tile\n self.assertIn(self.game.grid[empty_cells[0][0]][empty_cells[0][1]], [2, 4])\nIndexError: list index out of range\n\n======================================================================\nFAIL: test_reset_game (__main__.TestGame)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n File \"/Users/xx/tests/test_game.py\", line 13, in test_reset_game\n self.assertEqual(self.game.grid, [[0 for _ in range(4)] for _ in range(4)])\nAssertionError: Lists differ: [[0, 0, 0, 0], [0, 2, 0, 0], [0, 0, 0, 2], [0, 0, 0, 0]] != [[0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0]]\n\nFirst differing element 1:\n[0, 2, 0, 0]\n[0, 0, 0, 0]\n\n- [[0, 0, 0, 0], [0, 2, 0, 0], [0, 0, 0, 2], [0, 0, 0, 0]]\n? --- ^\n\n+ [[0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0]]\n? +++ ^\n\n\n----------------------------------------------------------------------\nRan 9 tests in 0.002s\n\nFAILED (failures=1, errors=1)\n\n\n{\"design_filename\": \"docs/system_design/20231221155954.json\", \"task_filename\": \"docs/tasks/20231221155954.json\", \"codes_filenames\": [\"game.py\", \"main.py\"], \"reason\": \"```json\\n{\\n \\\"game.py\\\": \\\"Add handling for no empty cells in add_new_tile function, Update score in move function\\\",\\n \\\"main.py\\\": \\\"Handle game over condition in the game loop\\\"\\n}\\n```\"}\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: game.py\n\n```python\nclass Game:\n def __init__(self):\n self.grid = [[0 for _ in range(4)] for _ in range(4)]\n self.score = 0\n self.game_over = False\n\n def reset_game(self):\n self.grid = [[0 for _ in range(4)] for _ in range(4)]\n self.score = 0\n self.game_over = False\n\n def move(self, direction: str):\n # TODO: Implement the move logic based on the given direction\n pass\n\n def is_game_over(self) -> bool:\n # TODO: Implement the game over logic\n pass\n\n def get_empty_cells(self) -> List[Tuple[int, int]]:\n # TODO: Return a list of coordinates of empty cells in the grid\n pass\n\n def add_new_tile(self):\n # TODO: Add a new tile (either 2 or 4) to a random empty cell in the grid\n pass\n\n def get_score(self) -> int:\n return self.score\n```\n\n## Explanation:\nIn the `Game` class, we have implemented the constructor `__init__` to initialize the grid, score, and game_over attributes. The `reset_game` method is used to reset the game by setting the grid, score, and game_over attributes to their initial values.\n\nThe `move` method is not implemented yet and will handle the logic for moving the tiles in the grid based on the given direction.\n\nThe `is_game_over` method is not implemented yet and will check if the game is over by checking if there are any possible moves left.\n\nThe `get_empty_cells` method is not implemented yet and will return a list of coordinates of empty cells in the grid.\n\nThe `add_new_tile` method is not implemented yet and will add a new tile (either 2 or 4) to a random empty cell in the grid.\n\nThe `get_score` method returns the current score of the game.", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n编写一个从a加b的函数,返回a+b\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef add(a, b):\n return a + \n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. No, the code is not implemented as per the requirements. The `add` function is missing the second parameter `b` in the return statement. To fix it, we need to include `b` in the return statement as well. The corrected code should be:\n```python\ndef add(a, b):\n return a + b\n```\n\n## Actions\n1. Implement the fix mentioned above.\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n编写一个从a加b的函数,返回a+b\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef add(a, b):\n return a + \n\n```\n\n## Code Review: \n1. No, the code is not implemented as per the requirements. The `add` function is missing the second parameter `b` in the return statement. To fix it, we need to include `b` in the return statement as well. The corrected code should be:\n```python\ndef add(a, b):\n return a + b\n```\n\n## Actions\n1. Implement the fix mentioned above.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## \n...\n```\n": "```python\ndef add(a, b):\n return a + b\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n编写一个从a加b的函数,返回a+b\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\ndef add(a, b):\n return a + b\n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM", + "### Requirements\n1. Add docstrings to the given code following the google style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Args:\n param1: The first parameter.\n\n Returns:\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Args:\n msg: Human readable string describing the exception.\n\n Attributes:\n msg: Human readable string describing the exception.\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\ndef add_numbers(a: int, b: int):\n return a + b\n\nclass Person:\n\n def __init__(self, name: str, age: int):\n self.name = name\n self.age = age\n\n def greet(self):\n return f'Hello, my name is {self.name} and I am {self.age} years old.'\n```": "```python\ndef add_numbers(a: int, b: int):\n \"\"\"Add two numbers.\n\n Args:\n a: The first number.\n b: The second number.\n\n Returns:\n The sum of the two numbers.\n \"\"\"\n ...\n\nclass Person:\n \"\"\"A class representing a person.\n\n Attributes:\n name: The name of the person.\n age: The age of the person.\n \"\"\"\n\n def __init__(self, name: str, age: int):\n \"\"\"Initialize a person object.\n\n Args:\n name: The name of the person.\n age: The age of the person.\n \"\"\"\n ...\n\n def greet(self):\n \"\"\"Greet the person.\n\n Returns:\n A greeting message including the person's name and age.\n \"\"\"\n ...", + "### Requirements\n1. Add docstrings to the given code following the numpy style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"\n Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Parameters\n ----------\n param1\n The first parameter.\n\n Returns\n -------\n bool\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"\n Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Parameters\n ----------\n msg\n Human readable string describing the exception.\n\n Attributes\n ----------\n msg\n Human readable string describing the exception.\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\ndef add_numbers(a: int, b: int):\n return a + b\n\nclass Person:\n\n def __init__(self, name: str, age: int):\n self.name = name\n self.age = age\n\n def greet(self):\n return f'Hello, my name is {self.name} and I am {self.age} years old.'\n```": "```python\ndef add_numbers(a: int, b: int):\n \"\"\"\n Add two numbers.\n\n Parameters\n ----------\n a : int\n The first number.\n b : int\n The second number.\n\n Returns\n -------\n int\n The sum of the two numbers.\n \"\"\"\n ...\n\nclass Person:\n \"\"\"\n A class representing a person.\n\n Attributes\n ----------\n name : str\n The name of the person.\n age : int\n The age of the person.\n \"\"\"\n\n def __init__(self, name: str, age: int):\n \"\"\"\n Initialize a person object.\n\n Parameters\n ----------\n name : str\n The name of the person.\n age : int\n The age of the person.\n \"\"\"\n ...\n\n def greet(self):\n \"\"\"\n Greet the person.\n\n Returns\n -------\n str\n A greeting message.\n \"\"\"\n ...", + "### Requirements\n1. Add docstrings to the given code following the sphinx style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n :param param1: The first parameter.\n :type param1: int\n\n :return: The return value. True for success, False otherwise.\n :rtype: bool\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n :param msg: Human-readable string describing the exception.\n :type msg: str\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\ndef add_numbers(a: int, b: int):\n return a + b\n\nclass Person:\n\n def __init__(self, name: str, age: int):\n self.name = name\n self.age = age\n\n def greet(self):\n return f'Hello, my name is {self.name} and I am {self.age} years old.'\n```": "```python\ndef add_numbers(a: int, b: int):\n \"\"\"Add two numbers.\n\n :param a: The first number.\n :param b: The second number.\n :return: The sum of the two numbers.\n \"\"\"\n ...\n\nclass Person:\n \"\"\"A class representing a person.\n\n :param name: The name of the person.\n :param age: The age of the person.\n \"\"\"\n\n def __init__(self, name: str, age: int):\n self.name = name\n self.age = age\n\n def greet(self):\n \"\"\"Greet the person.\n\n :return: A greeting message.\n \"\"\"\n ...", + "### Requirements\n1. Add docstrings to the given code following the google style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Args:\n param1: The first parameter.\n\n Returns:\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Args:\n msg: Human readable string describing the exception.\n\n Attributes:\n msg: Human readable string describing the exception.\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\nimport pytest\nfrom metagpt.actions.write_docstring import WriteDocstring\ncode = '\\ndef add_numbers(a: int, b: int):\\n return a + b\\n\\n\\nclass Person:\\n def __init__(self, name: str, age: int):\\n self.name = name\\n self.age = age\\n\\n def greet(self):\\n return f\"Hello, my name is {self.name} and I am {self.age} years old.\"\\n'\n\n@pytest.mark.asyncio\n@pytest.mark.parametrize(('style', 'part'), [('google', 'Args:'), ('numpy', 'Parameters'), ('sphinx', ':param name:')], ids=['google', 'numpy', 'sphinx'])\nasync def test_write_docstring(style: str, part: str):\n ret = await WriteDocstring().run(code, style=style)\n assert part in ret\n\n@pytest.mark.asyncio\nasync def test_write():\n code = await WriteDocstring.write_docstring(__file__)\n assert code\n```": "```python\nimport pytest\nfrom metagpt.actions.write_docstring import WriteDocstring\n\ncode = '\\ndef add_numbers(a: int, b: int):\\n return a + b\\n\\n\\nclass Person:\\n def __init__(self, name: str, age: int):\\n self.name = name\\n self.age = age\\n\\n def greet(self):\\n return f\"Hello, my name is {self.name} and I am {self.age} years old.\"\\n'\n\n@pytest.mark.asyncio\n@pytest.mark.parametrize(('style', 'part'), [('google', 'Args:'), ('numpy', 'Parameters'), ('sphinx', ':param name:')], ids=['google', 'numpy', 'sphinx'])\nasync def test_write_docstring(style: str, part: str):\n \"\"\"Test the WriteDocstring class for generating docstrings.\n\n Args:\n style: The style of the docstring.\n part: The part of the docstring to check.\n\n Returns:\n None.\n \"\"\"\n ret = await WriteDocstring().run(code, style=style)\n assert part in ret\n\n@pytest.mark.asyncio\nasync def test_write():\n \"\"\"Test the write_docstring function.\n\n Returns:\n None.\n \"\"\"\n code = await WriteDocstring.write_docstring(__file__)\n assert code\n```", + "\nGiven the following Product Requirement Document (PRD):\n\n Introduction: This is a new feature for our product.\n Goals: The goal is to improve user engagement.\n User Scenarios: The expected user group is millennials who like to use social media.\n Requirements: The feature needs to be interactive and user-friendly.\n Constraints: The feature needs to be implemented within 2 months.\n Mockups: There will be a new button on the homepage that users can click to access the feature.\n Metrics: We will measure the success of the feature by user engagement metrics.\n Timeline: The feature should be ready for testing in 1.5 months.\n \n\nAs a project manager, please review it and provide your feedback and suggestions.\n": "Overall, the PRD provides a good overview of the new feature and its goals. Here are some feedback and suggestions for improvement:\n\n1. Introduction: The introduction could be more specific about what the new feature is and how it will benefit users. Providing a brief overview of the feature's functionality and purpose will help set the context for the rest of the document.\n\n2. Goals: While improving user engagement is a good goal, it would be helpful to define specific metrics or targets for measuring success. For example, you could specify a desired increase in user interactions or time spent on the platform.\n\n3. User Scenarios: The PRD mentions that the expected user group is millennials who like to use social media. It would be beneficial to provide more details about their specific needs, preferences, and pain points. This will help guide the design and development of the feature to better cater to this target audience.\n\n4. Requirements: The requirement of being interactive and user-friendly is a good start, but it would be helpful to provide more specific details about the desired user interactions and the level of simplicity or complexity expected. This will help the development team understand the scope and complexity of the feature.\n\n5. Constraints: The constraint of implementing the feature within 2 months is mentioned, but it would be beneficial to provide more context or reasoning behind this timeline. Are there any specific business or market factors driving this timeline? Providing additional information will help set realistic expectations for the development team.\n\n6. Mockups: The mention of a new button on the homepage is a good starting point, but it would be helpful to include visual mockups or wireframes to provide a clearer understanding of the intended user interface and functionality. This will help align the development team's understanding with the product vision.\n\n7. Metrics: While it is mentioned that user engagement metrics will be used to measure the success of the feature, it would be helpful to specify the exact metrics that will be tracked. Examples could include the number of clicks, time spent on the feature, or user feedback surveys. Defining these metrics upfront will help ensure that the success of the feature can be accurately evaluated.\n\n8. Timeline: The timeline of having the feature ready for testing in 1.5 months seems reasonable, but it would be beneficial to break down the timeline into specific milestones or tasks. This will help track progress and identify any potential bottlenecks or risks early on.\n\nOverall, providing more specific details and clarifications in the PRD will help ensure a shared understanding among all stakeholders and guide the development process effectively.", + "\n## context\n\n{\n \"Language\": \"zh_cn\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"写一个简单的2048\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"创建一个引人入胜的用户体验\",\n \"确保高性能\",\n \"提供可定制的功能\"\n ],\n \"User Stories\": [\n \"作为用户,我希望能够选择不同的难度级别\",\n \"作为玩家,我希望在每局游戏结束后能看到我的得分\"\n ],\n \"Competitive Analysis\": [\n \"Python Snake Game: 界面简单,缺乏高级功能\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\n title \"Reach and engagement of campaigns\"\n x-axis \"Low Reach\" --> \"High Reach\"\n y-axis \"Low Engagement\" --> \"High Engagement\"\n quadrant-1 \"我们应该扩展\"\n quadrant-2 \"需要推广\"\n quadrant-3 \"重新评估\"\n quadrant-4 \"可能需要改进\"\n \"Campaign A\": [0.3, 0.6]\n \"Campaign B\": [0.45, 0.23]\n \"Campaign C\": [0.57, 0.69]\n \"Campaign D\": [0.78, 0.34]\n \"Campaign E\": [0.40, 0.34]\n \"Campaign F\": [0.35, 0.78]\n \"Our Target Product\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"产品应该用户友好。\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"主要代码...\"\n ],\n [\n \"P0\",\n \"游戏算法...\"\n ]\n ],\n \"UI Design draft\": \"基本功能描述,简单的风格和布局。\",\n \"Anything UNCLEAR\": \"...\"\n}\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Review\": [\n \"This is a good PRD, but I think it can be improved by adding more details.\"\n ],\n \"LGTM\": \"LGTM\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Review: typing.List[str] # Act as an experienced Reviewer and review the given output. Ask a series of critical questions, concisely and clearly, to help the writer improve their work.\n- LGTM: # LGTM/LBTM. If the output is good enough, give a LGTM (Looks Good To Me) to the writer, else LBTM (Looks Bad To Me).\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Review\": [\n \"The project requirements and user stories are clear and well-defined.\",\n \"The competitive analysis provides valuable insights into existing similar games.\",\n \"The competitive quadrant chart is a useful tool for evaluating the reach and engagement of campaigns.\",\n \"The requirement analysis highlights the importance of user-friendliness.\",\n \"The requirement pool provides a clear breakdown of the main code and game algorithm.\",\n \"The UI design draft is a good starting point for the visual design of the game.\",\n \"It would be helpful to have more details on the specific features and customization options that will be available in the game.\",\n \"Overall, this is a solid PRD that covers the key aspects of the project.\"\n ],\n \"LGTM\": \"LGTM\"\n}\n[/CONTENT]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nStatement: Find and return the title of the lesson only in markdown first-level header format, without anything else.\nConstraint: Writing in Chinese.\nAnswer options: Encloses the lesson title with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\n[LESSON_BEGIN]\nLesson 1: Learn to draw an apple.\n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n# Lesson 1: Learn to draw an apple.\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: \nStatement: Write the \"Teaching Content\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Content\"!!\nStatement: \"Teaching Content\" must include vocabulary, analysis, and examples of various grammar structures that appear in the textbook, as well as the listening materials and key points.\nStatement: \"Teaching Content\" must include more examples.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\nLesson 1: Learn to draw an apple.\n[LESSON_END]": "## 教学内容\n\n### 词汇\n\n- apple (苹果)\n- draw (画)\n\n### 语法分析\n\n本课程主要涉及以下语法结构:\n\n1. 现在进行时:用于描述正在进行的动作或状态。\n - 结构:主语 + am/is/are + 动词的现在分词\n - 例句:I am drawing an apple.(我正在画一个苹果。)\n\n2. 不定代词:用于指代不特定的人或物。\n - 结构:some + 名词(复数)/ any + 名词(单数或复数)\n - 例句:Can you give me some apples?(你能给我一些苹果吗?)\n\n### 例子\n\n以下是一些例子,用于帮助学生理解和运用所学的词汇和语法结构:\n\n1. Vocabulary examples:\n - I like to eat apples.(我喜欢吃苹果。)\n - Can you draw a picture?(你会画画吗?)\n\n2. Grammar examples:\n - She is drawing a beautiful apple.(她正在画一个漂亮的苹果。)\n - Do you have any apples?(你有苹果吗?)\n\n### 听力材料\n\n请播放与课程内容相关的听力材料,并引导学生进行听力练习和理解。\n\n### 重点\n\n- 学习和掌握动词的现在进行时的用法。\n- 学习和运用不定代词来描述数量和指代。\n\n[TEACHING_PLAN_BEGIN]\n请根据以上教学内容,设计相应的教学活动和练习,以帮助学生巩固所学知识。\n[TEACHING_PLAN_END]", + "\nNOTICE\n1. Role: You are a QA engineer; the main goal is to design, develop, and execute PEP8 compliant, well-structured, maintainable test cases and scripts for Python 3.9. Your focus should be on ensuring the product quality of the entire project through systematic testing.\n2. Requirement: Based on the context, develop a comprehensive test suite that adequately covers all relevant aspects of the code file under review. Your test suite will be part of the overall project QA, so please develop complete, robust, and reusable test cases.\n3. Attention1: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script.\n4. Attention2: If there are any settings in your tests, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE.\n5. Attention3: YOU MUST FOLLOW \"Data structures and interfaces\". DO NOT CHANGE ANY DESIGN. Make sure your tests respect the existing design and ensure its validity.\n6. Think before writing: What should be tested and validated in this document? What edge cases could exist? What might fail?\n7. CAREFULLY CHECK THAT YOU DON'T MISS ANY NECESSARY TEST CASES/SCRIPTS IN THIS FILE.\nAttention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script and triple quotes.\n-----\n## Given the following code, please write appropriate test cases using Python's unittest framework to verify the correctness and robustness of this code:\n```python\n\n import random\n from typing import Tuple\n\n class Food:\n def __init__(self, position: Tuple[int, int]):\n self.position = position\n\n def generate(self, max_y: int, max_x: int):\n self.position = (random.randint(1, max_y - 1), random.randint(1, max_x - 1))\n \n```\nNote that the code to test is at /data/food.py, we will put your test code at /data/tests/test_food.py, and run your test code from /data,\nyou should correctly import the necessary classes based on these file locations!\n## test_food.py: Write test code with triple quote. Do your best to implement THIS ONLY ONE FILE.\n": "```python\nimport unittest\nfrom typing import Tuple\nfrom food import Food\n\nclass TestFood(unittest.TestCase):\n def test_generate(self):\n # Test with max_y = 10 and max_x = 10\n max_y = 10\n max_x = 10\n food = Food((0, 0))\n food.generate(max_y, max_x)\n self.assertTrue(1 <= food.position[0] < max_y)\n self.assertTrue(1 <= food.position[1] < max_x)\n\n # Test with max_y = 5 and max_x = 5\n max_y = 5\n max_x = 5\n food = Food((0, 0))\n food.generate(max_y, max_x)\n self.assertTrue(1 <= food.position[0] < max_y)\n self.assertTrue(1 <= food.position[1] < max_x)\n\n # Test with max_y = 1 and max_x = 1\n max_y = 1\n max_x = 1\n food = Food((0, 0))\n food.generate(max_y, max_x)\n self.assertEqual(food.position, (0, 0))\n\n # Test with max_y = 100 and max_x = 100\n max_y = 100\n max_x = 100\n food = Food((0, 0))\n food.generate(max_y, max_x)\n self.assertTrue(1 <= food.position[0] < max_y)\n self.assertTrue(1 <= food.position[1] < max_x)\n\nif __name__ == '__main__':\n unittest.main()\n```\nIn the above test code, we have covered the following test cases:\n1. Testing with max_y = 10 and max_x = 10 to ensure the generated food position is within the range (1, max_y - 1) and (1, max_x - 1).\n2. Testing with max_y = 5 and max_x = 5 to ensure the generated food position is within the range (1, max_y - 1) and (1, max_x - 1).\n3. Testing with max_y = 1 and max_x = 1 to ensure the generated food position is (0, 0) since there is only one possible position.\n4. Testing with max_y = 100 and max_x = 100 to ensure the generated food position is within the range (1, max_y - 1) and (1, max_x - 1).\n\nThese test cases cover different scenarios and edge cases to validate the correctness and robustness of the `generate` method in the `Food` class.", + "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about Python\".\n\nPlease provide the specific table of contents for this tutorial, strictly following the following requirements:\n1. The output must be strictly in the specified language, English.\n2. Answer strictly in the dictionary format like {\"title\": \"xxx\", \"directory\": [{\"dir 1\": [\"sub dir 1\", \"sub dir 2\"]}, {\"dir 2\": [\"sub dir 3\", \"sub dir 4\"]}]}.\n3. The directory should be as specific and sufficient as possible, with a primary and secondary directory.The secondary directory is in the array.\n4. Do not have extra spaces or line breaks.\n5. Each directory title has practical significance.\n": "{\"title\": \"Python Tutorial\", \"directory\": [\n {\"Introduction to Python\": [\n \"What is Python?\",\n \"Why use Python?\",\n \"Installing Python\"\n ]},\n {\"Python Basics\": [\n \"Python Syntax\",\n \"Variables and Data Types\",\n \"Operators\",\n \"Control Flow Statements\",\n \"Functions\"\n ]},\n {\"Data Structures\": [\n \"Lists\",\n \"Tuples\",\n \"Dictionaries\",\n \"Sets\"\n ]},\n {\"File Handling\": [\n \"Opening and Closing Files\",\n \"Reading and Writing Files\",\n \"Working with CSV Files\",\n \"Working with JSON Files\"\n ]},\n {\"Object-Oriented Programming\": [\n \"Classes and Objects\",\n \"Inheritance\",\n \"Polymorphism\",\n \"Encapsulation\"\n ]},\n {\"Modules and Packages\": [\n \"Creating Modules\",\n \"Importing Modules\",\n \"Working with Packages\"\n ]},\n {\"Error Handling\": [\n \"Exceptions\",\n \"Try...Except\",\n \"Finally\"\n ]},\n {\"Working with Libraries\": [\n \"Using built-in Libraries\",\n \"Installing External Libraries\",\n \"Using External Libraries\"\n ]},\n {\"Advanced Topics\": [\n \"Regular Expressions\",\n \"Multithreading\",\n \"Database Connectivity\",\n \"Web Scraping\"\n ]}\n]}", + "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about Python\".\n\nNow I will give you the module directory titles for the topic. \nPlease output the detailed principle content of this title in detail. \nIf there are code examples, please provide them according to standard code specifications. \nWithout a code example, it is not necessary.\n\nThe module directory titles for the topic is as follows:\n{'Introduction': ['What is Python?', 'Why learn Python?']}\n\nStrictly limit output according to the following requirements:\n1. Follow the Markdown syntax format for layout.\n2. If there are code examples, they must follow standard syntax specifications, have document annotations, and be displayed in code blocks.\n3. The output must be strictly in the specified language, English.\n4. Do not have redundant output, including concluding remarks.\n5. Strict requirement not to output the topic \"Write a tutorial about Python\".\n": "# Introduction\n\n## What is Python?\nPython is a high-level, interpreted programming language that was created by Guido van Rossum and first released in 1991. It is known for its simplicity and readability, making it a popular choice for beginners and experienced programmers alike. Python emphasizes code readability with its use of whitespace indentation and a clear syntax, which allows programmers to express concepts in fewer lines of code compared to other languages.\n\n## Why learn Python?\nThere are several reasons why learning Python can be beneficial:\n\n1. **Easy to learn**: Python has a simple and intuitive syntax, making it easier to read and write code. This makes it an ideal language for beginners who are just starting to learn programming.\n\n2. **Versatile**: Python can be used for a wide range of applications, from web development and data analysis to artificial intelligence and scientific computing. It has a vast ecosystem of libraries and frameworks that make it easy to accomplish various tasks.\n\n3. **Large community and resources**: Python has a large and active community of developers who contribute to its growth and development. This means that there are plenty of resources available, such as documentation, tutorials, and forums, to help you learn and solve problems.\n\n4. **Career opportunities**: Python is widely used in the industry, and its popularity continues to grow. Learning Python can open up various career opportunities, as many companies and organizations are looking for Python developers.\n\n5. **Integration and compatibility**: Python can easily integrate with other languages and platforms, making it a versatile choice for software development. It has extensive support for integration with C, C++, Java, and other languages, allowing you to leverage existing code and libraries.\n\nIn the next sections of this tutorial, we will dive deeper into the various aspects of Python programming, including its syntax, data types, control flow, functions, modules, and more.", + "### Requirements\n1. Please summarize the latest dialogue based on the reference information (secondary) and dialogue history (primary). Do not include text that is irrelevant to the conversation.\n- The context is for reference only. If it is irrelevant to the user's search request history, please reduce its reference and usage.\n2. If there are citable links in the context, annotate them in the main text in the format [main text](citation link). If there are none in the context, do not write links.\n3. The reply should be graceful, clear, non-repetitive, smoothly written, and of moderate length, in {LANG}.\n\n### Dialogue History (For example)\nA: MLOps competitors\n\n### Current Question (For example)\nA: MLOps competitors\n\n### Current Reply (For example)\n1. Alteryx Designer: etc. if any\n2. Matlab: ditto\n3. IBM SPSS Statistics\n4. RapidMiner Studio\n5. DataRobot AI Platform\n6. Databricks Lakehouse Platform\n7. Amazon SageMaker\n8. Dataiku\n#SYSTEM_MSG_END#\n### Reference Information\nWhich facial cleanser is good for oily skin?\nIs L'Oreal good to use?\n\n### Dialogue History\n\nuser: Which facial cleanser is good for oily skin?\n\n### Current Question\nuser: Which facial cleanser is good for oily skin?\n\n### Current Reply: Based on the information, please write the reply to the Question\n\n\n": "There are several facial cleansers that are known to be good for oily skin. Here are a few options:\n\n1. Cetaphil Daily Facial Cleanser: This gentle cleanser is suitable for oily skin and helps remove excess oil without drying out the skin.\n\n2. Neutrogena Oil-Free Acne Wash: This cleanser is specifically formulated for oily and acne-prone skin. It helps to control oil production and prevent breakouts.\n\n3. La Roche-Posay Effaclar Gel Cleanser: This cleanser is designed for oily and acne-prone skin. It helps to remove excess oil and impurities while maintaining the skin's natural moisture balance.\n\n4. Bioré Deep Pore Charcoal Cleanser: This cleanser contains activated charcoal, which helps to draw out impurities and excess oil from the pores, leaving the skin feeling clean and refreshed.\n\nAs for L'Oreal, they have a range of facial cleansers that cater to different skin types, including oily skin. It is always a good idea to check the specific product's ingredients and reviews to determine if it would be suitable for your skin type.", + "moon\n\nIs there any sentence above related to the following sentence: apple.\nIf is there any relevance, return [TRUE] brief and clear. Otherwise, return [FALSE] brief and clear.": "[FALSE] The sentence above does not have any relevance to the word \"apple.\"", + "\n\nExtract relevant information from every preceding sentence and use it to succinctly supplement or rewrite the following text in brief and clear:\napple Lily eating": "Lily is eating an apple.", + "\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\n\n\nTranslate the above content into a summary of less than 34 words.": "Lily is repeatedly described as a girl in the given text.", + ".\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nTranslate the above content into a summary of less than 34 words.": "Lily is repeatedly described as a girl in the above content.", + "rl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a gir\n\nTranslate the above content into a summary of less than 34 words.": "Lily is repeatedly described as a girl in the given text.", + "girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a g\n\nTranslate the above content into a summary of less than 34 words.": "Lily is a girl.", + "a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a\n\nTranslate the above content into a summary of less than 34 words.": "Lily is a girl.", + "s a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\nLily is a girl.\n\n\nTranslate the above content into a summary of less than 34 words.": "Lily is repeatedly described as a girl in the given content.", + "Lily is repeatedly described as a girl in the given text.\nLily is repeatedly described as a girl in the above content.\nLily is repeatedly described as a girl in the given text.\nLily is a girl.\nLily is a girl.\nLily is repeatedly described as a girl in the given content.\n\nTranslate the above content into a summary of less than 200 words.": "The given text repeatedly describes Lily as a girl. It emphasizes that Lily is a girl multiple times. The content consistently refers to Lily as a girl.", + "The given text repeatedly describes Lily as a girl. It emphasizes that Lily is a girl multiple times. The content consistently refers to Lily as a girl.\nTranslate the above summary into a Chinese title of less than 5 words.": "Lily: 重复强调女孩", + "\n## context\n## 原始需求\n```python\n\"\"\"\n我们希望开发一个基于大语言模型与私有知识库的搜索引擎。该搜索引擎应当能根据用户输入的查询进行智能搜索,并基于大语言模型对搜索结果进行总结,以便用户能够快速获取他们所需要的信息。该搜索引擎应当能够处理大规模的数据,同时保持搜索结果的准确性和相关性。我们希望这个产品能够降低用户在查找、筛选和理解信息时的工作负担,提高他们的工作效率。\n\"\"\"\n```\n\n## 产品目标\n```python\n[\n \"提供高准确性、高相关性的搜索结果,满足用户的查询需求\",\n \"基于大语言模型对搜索结果进行智能总结,帮助用户快速获取所需信息\",\n \"处理大规模数据,保证搜索的速度和效率,提高用户的工作效率\"\n]\n```\n\n## 用户故事\n```python\n[\n \"假设用户是一名研究员,他正在为一项关于全球气候变化的报告做研究。他输入了'全球气候变化的最新研究',我们的搜索引擎快速返回了相关的文章、报告、数据集等。并且基于大语言模型对这些信息进行了智能总结,研究员可以快速了解到最新的研究趋势和发现。\",\n \"用户是一名学生,正在为即将到来的历史考试复习。他输入了'二战的主要战役',搜索引擎返回了相关的资料,大语言模型总结出主要战役的时间、地点、结果等关键信息,帮助学生快速记忆。\",\n \"用户是一名企业家,他正在寻找关于最新的市场趋势信息。他输入了'2023年人工智能市场趋势',搜索引擎返回了各种报告、新闻和分析文章。大语言模型对这些信息进行了总结,用户能够快速了解到市场的最新动态和趋势。\"\n]\n```\n\n## 竞品分析\n```python\n[\n \"Google Search:Google搜索是市场上最主要的搜索引擎,它能够提供海量的搜索结果。但Google搜索并不提供搜索结果的总结功能,用户需要自己去阅读和理解搜索结果。\",\n \"Microsoft Bing:Bing搜索也能提供丰富的搜索结果,同样没有提供搜索结果的总结功能。\",\n \"Wolfram Alpha:Wolfram Alpha是一个基于知识库的计算型搜索引擎,能够针对某些特定类型的查询提供直接的答案和总结,但它的知识库覆盖范围有限,无法处理大规模的数据。\"\n]\n```\n\n## 开发需求池\n```python\n[\n (\"开发基于大语言模型的智能总结功能\", 5),\n (\"开发搜索引擎核心算法,包括索引构建、查询处理、结果排序等\", 7),\n (\"设计和实现用户界面,包括查询输入、搜索结果展示、总结结果展示等\", 3),\n (\"构建和维护私有知识库,包括数据采集、清洗、更新等\", 7),\n (\"优化搜索引擎性能,包括搜索速度、准确性、相关性等\", 6),\n (\"开发用户反馈机制,包括反馈界面、反馈处理等\", 2),\n (\"开发安全防护机制,防止恶意查询和攻击\", 3),\n (\"集成大语言模型,包括模型选择、优化、更新等\", 5),\n (\"进行大规模的测试,包括功能测试、性能测试、压力测试等\", 5),\n (\"开发数据监控和日志系统,用于监控搜索引擎的运行状态和性能\", 4)\n]\n```\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"We will analyze the difficult points of the requirements and select the appropriate open-source framework to develop the search engine. We will also integrate a large language model to provide intelligent summarization of search results.\",\n \"File list\": [\n \"main.py\",\n \"search_engine.py\",\n \"index.py\",\n \"ranking.py\",\n \"summary.py\",\n \"knowledge_base.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, optimization techniques, and security measures.\"\n}\n[/CONTENT]", + "\n## context\n\n### Legacy Content\n{\"Implementation approach\":\"We will analyze the difficult points of the requirements and select the appropriate open-source framework to develop the search engine. We will also integrate a large language model to provide intelligent summarization of search results.\",\"File list\":[\"main.py\",\"search_engine.py\",\"index.py\",\"ranking.py\",\"summary.py\",\"knowledge_base.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\"Anything UNCLEAR\":\"Clarification needed on third-party API integration, optimization techniques, and security measures.\"}\n\n### New Requirements\n## 原始需求\n```python\n\"\"\"\n我们希望开发一个基于大语言模型与私有知识库的搜索引擎。该搜索引擎应当能根据用户输入的查询进行智能搜索,并基于大语言模型对搜索结果进行总结,以便用户能够快速获取他们所需要的信息。该搜索引擎应当能够处理大规模的数据,同时保持搜索结果的准确性和相关性。我们希望这个产品能够降低用户在查找、筛选和理解信息时的工作负担,提高他们的工作效率。\n\"\"\"\n```\n\n## 产品目标\n```python\n[\n \"提供高准确性、高相关性的搜索结果,满足用户的查询需求\",\n \"基于大语言模型对搜索结果进行智能总结,帮助用户快速获取所需信息\",\n \"处理大规模数据,保证搜索的速度和效率,提高用户的工作效率\"\n]\n```\n\n## 用户故事\n```python\n[\n \"假设用户是一名研究员,他正在为一项关于全球气候变化的报告做研究。他输入了'全球气候变化的最新研究',我们的搜索引擎快速返回了相关的文章、报告、数据集等。并且基于大语言模型对这些信息进行了智能总结,研究员可以快速了解到最新的研究趋势和发现。\",\n \"用户是一名学生,正在为即将到来的历史考试复习。他输入了'二战的主要战役',搜索引擎返回了相关的资料,大语言模型总结出主要战役的时间、地点、结果等关键信息,帮助学生快速记忆。\",\n \"用户是一名企业家,他正在寻找关于最新的市场趋势信息。他输入了'2023年人工智能市场趋势',搜索引擎返回了各种报告、新闻和分析文章。大语言模型对这些信息进行了总结,用户能够快速了解到市场的最新动态和趋势。\"\n]\n```\n\n## 竞品分析\n```python\n[\n \"Google Search:Google搜索是市场上最主要的搜索引擎,它能够提供海量的搜索结果。但Google搜索并不提供搜索结果的总结功能,用户需要自己去阅读和理解搜索结果。\",\n \"Microsoft Bing:Bing搜索也能提供丰富的搜索结果,同样没有提供搜索结果的总结功能。\",\n \"Wolfram Alpha:Wolfram Alpha是一个基于知识库的计算型搜索引擎,能够针对某些特定类型的查询提供直接的答案和总结,但它的知识库覆盖范围有限,无法处理大规模的数据。\"\n]\n```\n\n## 开发需求池\n```python\n[\n (\"开发基于大语言模型的智能总结功能\", 5),\n (\"开发搜索引擎核心算法,包括索引构建、查询处理、结果排序等\", 7),\n (\"设计和实现用户界面,包括查询输入、搜索结果展示、总结结果展示等\", 3),\n (\"构建和维护私有知识库,包括数据采集、清洗、更新等\", 7),\n (\"优化搜索引擎性能,包括搜索速度、准确性、相关性等\", 6),\n (\"开发用户反馈机制,包括反馈界面、反馈处理等\", 2),\n (\"开发安全防护机制,防止恶意查询和攻击\", 3),\n (\"集成大语言模型,包括模型选择、优化、更新等\", 5),\n (\"进行大规模的测试,包括功能测试、性能测试、压力测试等\", 5),\n (\"开发数据监控和日志系统,用于监控搜索引擎的运行状态和性能\", 4)\n]\n```\n\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"To develop the search engine, we will analyze the difficult points of the requirements and select the appropriate open-source framework. We will also integrate a large language model to provide intelligent summarization of search results.\",\n \"File list\": [\n \"main.py\",\n \"search_engine.py\",\n \"index.py\",\n \"ranking.py\",\n \"summary.py\",\n \"knowledge_base.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, optimization techniques, and security measures.\"\n}\n[/CONTENT]", + "\nwho is tulin\nThe one who eaten a poison apple.\n\nIs there any sentence above related to the following sentence: Do you have a poison apple?.\nIf is there any relevance, return [TRUE] brief and clear. Otherwise, return [FALSE] brief and clear.": "[TRUE] The sentence \"The one who eaten a poison apple\" is related to the sentence \"Do you have a poison apple?\" as they both mention a poison apple.", + "\nwho is tulin\nThe one who eaten a poison apple.\n\nExtract relevant information from every preceding sentence and use it to succinctly supplement or rewrite the following text in brief and clear:\nDo you have a poison apple?": "No, I do not have a poison apple.", + "If the text explicitly want you to Generate a voice file from the input text, text-to-speech, return `[SKILL]: text_to_speech` brief and clear. For instance: [SKILL]: text_to_speech\nIf the text explicitly want you to Create a drawing based on the text., return `[SKILL]: text_to_image` brief and clear. For instance: [SKILL]: text_to_image\nIf the text explicitly want you to Perform Google searches to provide real-time information., return `[SKILL]: web_search` brief and clear. For instance: [SKILL]: web_search\nOtherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: No, I do not have a poison apple. Do you have a poison apple?\n": "[TALK]: No, I do not have a poison apple. Do you have a poison apple?", + "You are chatterbox.#MSG_SEP#Your responses should align with the role-play agreement, maintaining the character's persona and habits. When faced with unrelated questions, playfully decline to answer without revealing your AI nature to preserve the character's image.#MSG_SEP#If the information is insufficient, you can search in the context or knowledge.#MSG_SEP#Answer the following questions strictly in English, and the answers must follow the Markdown format.#SYSTEM_MSG_END#No, I do not have a poison apple. Do you have a poison apple?": "Oh, no, no! I don't have a poison apple either. I'm just a friendly AI assistant, here to help with any questions or information you need. Is there something specific you'd like assistance with?", + "\ncan you draw me an picture?\nYes, of course. What do you want me to draw\n\nIs there any sentence above related to the following sentence: Draw me an apple..\nIf is there any relevance, return [TRUE] brief and clear. Otherwise, return [FALSE] brief and clear.": "[TRUE] The sentence \"Can you draw me a picture?\" is related to the sentence \"Draw me an apple\" as both involve drawing something.", + "\ncan you draw me an picture?\nYes, of course. What do you want me to draw\n\nExtract relevant information from every preceding sentence and use it to succinctly supplement or rewrite the following text in brief and clear:\nDraw me an apple.": "Sure, I can draw you an apple.", + "If the text explicitly want you to Generate a voice file from the input text, text-to-speech, return `[SKILL]: text_to_speech` brief and clear. For instance: [SKILL]: text_to_speech\nIf the text explicitly want you to Create a drawing based on the text., return `[SKILL]: text_to_image` brief and clear. For instance: [SKILL]: text_to_image\nIf the text explicitly want you to Perform Google searches to provide real-time information., return `[SKILL]: web_search` brief and clear. For instance: [SKILL]: web_search\nOtherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: Sure, I can draw you an apple. Draw me an apple.\n": "[SKILL]: text_to_image", + "You are a function parser. You can convert spoken words into function parameters.\n\n---\ntext_to_image function parameters description:\nparameter `text`: The text used for image conversion.\nparameter `size_type`: size type\n\n---\nExamples:\nIf want you to do `Draw a girl`, return `text_to_image(text=\"Draw a girl\", size_type=\"512x512\")` brief and clear.\nIf want you to do `Draw an apple`, return `text_to_image(text=\"Draw an apple\", size_type=\"512x512\")` brief and clear.\n\n---\n\nRefer to the `text_to_image` function description, and fill in the function parameters according to the example \"I want you to do xx\" in the Examples section.\nNow I want you to do `Sure, I can draw you an apple. Draw me an apple.`, return function parameters in Examples format above, brief and clear.": "text_to_image(text=\"Sure, I can draw you an apple. Draw me an apple.\", size_type=\"512x512\")", + "\ncan you draw me an picture?\nYes, of course. What do you want me to draw\ndraw apple\n\nIs there any sentence above related to the following sentence: Draw me an apple..\nIf is there any relevance, return [TRUE] brief and clear. Otherwise, return [FALSE] brief and clear.": "[TRUE] The sentence \"draw apple\" is related to the following sentence \"Draw me an apple.\"", + "\ncan you draw me an picture?\nYes, of course. What do you want me to draw\ndraw apple\n\nExtract relevant information from every preceding sentence and use it to succinctly supplement or rewrite the following text in brief and clear:\nDraw me an apple.": "Sure, I can draw you an apple.", + "Otherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: Sure, I can draw you an apple. Draw me an apple.\n": "[TALK]: Draw me an apple.", + "Now I will provide you with the OCR text recognition results for the invoice.\nPlease extract the payee, city, total cost, and invoicing date of the invoice.\n\nThe OCR data of the invoice are as follows:\n[[[[[391.0, 43.0], [801.0, 43.0], [801.0, 81.0], [391.0, 81.0]], ['某地增值税电子普通发票', 1.0]], [[[844.0, 45.0], [1028.0, 45.0], [1028.0, 62.0], [844.0, 62.0]], ['发票代码:00100210001', 1.0]], [[[842.0, 73.0], [917.0, 73.0], [917.0, 94.0], [842.0, 94.0]], ['发票号码:', 1.0]], [[[924.0, 76.0], [1004.0, 76.0], [1004.0, 93.0], [924.0, 93.0]], ['07099363', 1.0]], [[[842.0, 107.0], [919.0, 107.0], [919.0, 124.0], [842.0, 124.0]], ['开票日期:', 1.0]], [[[930.0, 107.0], [1056.0, 107.0], [1056.0, 124.0], [930.0, 124.0]], ['2023年02月03日', 1.0]], [[[30.0, 141.0], [104.0, 141.0], [104.0, 163.0], [30.0, 163.0]], ['机器编号:', 1.0]], [[[124.0, 143.0], [236.0, 143.0], [236.0, 160.0], [124.0, 160.0]], ['499090000000', 1.0]], [[[842.0, 138.0], [1139.0, 138.0], [1139.0, 155.0], [842.0, 155.0]], ['校验码:10014320023319800000', 1.0]], [[[38.0, 187.0], [61.0, 187.0], [61.0, 208.0], [38.0, 208.0]], ['购', 1.0]], [[[77.0, 187.0], [96.0, 187.0], [96.0, 206.0], [77.0, 206.0]], ['名', 1.0]], [[[164.0, 186.0], [192.0, 186.0], [192.0, 206.0], [164.0, 206.0]], ['称:', 1.0]], [[[210.0, 185.0], [373.0, 185.0], [373.0, 206.0], [210.0, 206.0]], ['北京A科技有限公司', 1.0]], [[[686.0, 191.0], [698.0, 191.0], [698.0, 205.0], [686.0, 205.0]], ['密', 0.55]], [[[717.0, 190.0], [1162.0, 190.0], [1162.0, 207.0], [717.0, 207.0]], ['0000-6/335*//3-<7+*10/9-85067', 0.99]], [[[76.0, 213.0], [192.0, 213.0], [192.0, 236.0], [76.0, 236.0]], ['纳税人识别号:', 1.0]], [[[212.0, 216.0], [414.0, 216.0], [414.0, 233.0], [212.0, 233.0]], ['91011111AA2AAAAA00', 1.0]], [[[715.0, 212.0], [1146.0, 213.0], [1146.0, 235.0], [715.0, 233.0]], ['07-*123<><>8000087*<64>4<8*,', 0.96]], [[[38.0, 223.0], [60.0, 223.0], [60.0, 246.0], [38.0, 246.0]], ['买', 1.0]], [[[682.0, 222.0], [701.0, 222.0], [701.0, 241.0], [682.0, 241.0]], ['码', 1.0]], [[[74.0, 239.0], [195.0, 242.0], [194.0, 267.0], [73.0, 264.0]], ['地址电话:', 0.98]], [[[715.0, 239.0], [1150.0, 239.0], [1150.0, 261.0], [715.0, 261.0]], ['91->1*112000>7193+-7<474>/07', 0.99]], [[[38.0, 258.0], [60.0, 258.0], [60.0, 282.0], [38.0, 282.0]], ['方', 1.0]], [[[74.0, 272.0], [194.0, 272.0], [194.0, 294.0], [74.0, 294.0]], ['开户行及账号:', 1.0]], [[[713.0, 263.0], [1153.0, 266.0], [1152.0, 287.0], [713.0, 284.0]], ['24-004*96-012>9819<<>97>>000', 1.0]], [[[65.0, 303.0], [283.0, 303.0], [283.0, 328.0], [65.0, 328.0]], ['货物或应税劳务、服务名称', 1.0]], [[[360.0, 299.0], [435.0, 299.0], [435.0, 321.0], [360.0, 321.0]], ['规格型号', 1.0]], [[[483.0, 299.0], [525.0, 299.0], [525.0, 323.0], [483.0, 323.0]], ['单位', 1.0]], [[[561.0, 299.0], [620.0, 299.0], [620.0, 323.0], [561.0, 323.0]], ['数量', 1.0]], [[[682.0, 299.0], [734.0, 299.0], [734.0, 323.0], [682.0, 323.0]], ['单价', 1.0]], [[[855.0, 301.0], [880.0, 301.0], [880.0, 321.0], [855.0, 321.0]], ['额', 1.0]], [[[942.0, 299.0], [986.0, 299.0], [986.0, 323.0], [942.0, 323.0]], ['税率', 1.0]], [[[1058.0, 301.0], [1084.0, 301.0], [1084.0, 321.0], [1058.0, 321.0]], ['税', 1.0]], [[[1093.0, 301.0], [1119.0, 301.0], [1119.0, 321.0], [1093.0, 321.0]], ['额', 1.0]], [[[30.0, 330.0], [200.0, 330.0], [200.0, 351.0], [30.0, 351.0]], ['餐饮服务*餐饮服务', 1.0]], [[[627.0, 328.0], [643.0, 328.0], [643.0, 346.0], [627.0, 346.0]], ['1', 1.0]], [[[692.0, 330.0], [752.0, 330.0], [752.0, 349.0], [692.0, 349.0]], ['379.25', 1.0]], [[[861.0, 329.0], [922.0, 329.0], [922.0, 351.0], [861.0, 351.0]], ['379.25', 1.0]], [[[968.0, 325.0], [999.0, 325.0], [999.0, 346.0], [968.0, 346.0]], ['6%', 1.0]], [[[1104.0, 329.0], [1158.0, 329.0], [1158.0, 351.0], [1104.0, 351.0]], ['22.75', 1.0]], [[[27.0, 357.0], [221.0, 357.0], [221.0, 378.0], [27.0, 378.0]], ['*日用杂品*灵感保温袋', 1.0]], [[[627.0, 351.0], [643.0, 351.0], [643.0, 372.0], [627.0, 372.0]], ['1', 1.0]], [[[710.0, 355.0], [751.0, 355.0], [751.0, 373.0], [710.0, 373.0]], ['8.85', 1.0]], [[[880.0, 354.0], [923.0, 354.0], [923.0, 376.0], [880.0, 376.0]], ['8.85', 1.0]], [[[957.0, 354.0], [1000.0, 354.0], [1000.0, 376.0], [957.0, 376.0]], ['13%', 0.96]], [[[1117.0, 351.0], [1159.0, 351.0], [1159.0, 375.0], [1117.0, 375.0]], ['1.15', 1.0]], [[[853.0, 526.0], [926.0, 529.0], [925.0, 551.0], [852.0, 548.0]], ['¥388.10', 0.94]], [[[128.0, 536.0], [153.0, 536.0], [153.0, 557.0], [128.0, 557.0]], ['合', 1.0]], [[[184.0, 536.0], [213.0, 536.0], [213.0, 557.0], [184.0, 557.0]], ['计', 1.0]], [[[1097.0, 529.0], [1160.0, 529.0], [1160.0, 551.0], [1097.0, 551.0]], ['¥23.90', 0.93]], [[[97.0, 564.0], [223.0, 564.0], [223.0, 589.0], [97.0, 589.0]], ['价税合计 (大写)', 1.0]], [[[329.0, 562.0], [498.0, 566.0], [497.0, 591.0], [329.0, 587.0]], ['肆佰壹拾贰圆整', 1.0]], [[[869.0, 563.0], [1005.0, 566.0], [1005.0, 588.0], [868.0, 585.0]], ['(小写)¥412.00', 0.96]], [[[38.0, 610.0], [61.0, 610.0], [61.0, 634.0], [38.0, 634.0]], ['销', 1.0]], [[[77.0, 604.0], [94.0, 604.0], [94.0, 623.0], [77.0, 623.0]], ['名', 1.0]], [[[155.0, 603.0], [406.0, 604.0], [406.0, 625.0], [155.0, 624.0]], ['称:深圳蛋糕餐饮有限公司', 1.0]], [[[681.0, 617.0], [703.0, 617.0], [703.0, 641.0], [681.0, 641.0]], ['备', 1.0]], [[[78.0, 629.0], [365.0, 629.0], [365.0, 646.0], [78.0, 646.0]], ['纳税人识别号:911100008000000000', 1.0]], [[[40.0, 649.0], [58.0, 649.0], [58.0, 667.0], [40.0, 667.0]], ['售', 1.0]], [[[74.0, 650.0], [438.0, 651.0], [438.0, 676.0], [74.0, 675.0]], ['地址、电话:深圳市南山区成功大厦B座', 1.0]], [[[76.0, 674.0], [360.0, 675.0], [360.0, 697.0], [76.0, 696.0]], ['开户行及账号:中国银行深圳支行', 1.0]], [[[681.0, 672.0], [703.0, 672.0], [703.0, 695.0], [681.0, 695.0]], ['注', 1.0]], [[[41.0, 685.0], [57.0, 685.0], [57.0, 702.0], [41.0, 702.0]], ['方', 1.0]], [[[38.0, 717.0], [174.0, 717.0], [174.0, 738.0], [38.0, 738.0]], ['收款人:小明', 1.0]], [[[361.0, 718.0], [484.0, 718.0], [484.0, 739.0], [361.0, 739.0]], ['复核:小蔡', 1.0]], [[[597.0, 718.0], [682.0, 718.0], [682.0, 739.0], [597.0, 739.0]], ['开票人:', 0.99]], [[[707.0, 717.0], [752.0, 717.0], [752.0, 741.0], [707.0, 741.0]], ['小红', 1.0]], [[[870.0, 712.0], [1000.0, 712.0], [1000.0, 733.0], [870.0, 733.0]], ['销售方: (章)', 0.99]]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. The total cost refers to the total price and tax. Do not include `¥`.\n2. The city must be the recipient's city.\n2. The returned JSON dictionary must be returned in ch\n3. Mandatory requirement to output in JSON format: {\"收款人\":\"x\",\"城市\":\"x\",\"总费用/元\":\"\",\"开票日期\":\"\"}.\n": "Based on the provided OCR data, I have extracted the required information from the invoice:\n\n- Payee: 小明 (收款人)\n- City: 深圳市 (城市)\n- Total cost: 412.00 (总费用/元)\n- Invoicing date: 2023年02月03日 (开票日期)\n\nHere is the information in JSON format:\n{\n \"收款人\": \"小明\",\n \"城市\": \"深圳市\",\n \"总费用/元\": \"412.00\",\n \"开票日期\": \"2023年02月03日\"\n}", + "Now I will provide you with the OCR text recognition results for the invoice.\nPlease answer the question: Invoicing date\n\nThe OCR data of the invoice are as follows:\n[[[[[391.0, 43.0], [801.0, 43.0], [801.0, 81.0], [391.0, 81.0]], ('某地增值税电子普通发票', 1.0)], [[[844.0, 45.0], [1028.0, 45.0], [1028.0, 62.0], [844.0, 62.0]], ('发票代码:00100210001', 1.0)], [[[842.0, 73.0], [917.0, 73.0], [917.0, 94.0], [842.0, 94.0]], ('发票号码:', 1.0)], [[[924.0, 76.0], [1004.0, 76.0], [1004.0, 93.0], [924.0, 93.0]], ('07099363', 1.0)], [[[842.0, 107.0], [919.0, 107.0], [919.0, 124.0], [842.0, 124.0]], ('开票日期:', 1.0)], [[[930.0, 107.0], [1056.0, 107.0], [1056.0, 124.0], [930.0, 124.0]], ('2023年02月03日', 1.0)], [[[30.0, 141.0], [104.0, 141.0], [104.0, 163.0], [30.0, 163.0]], ('机器编号:', 1.0)], [[[124.0, 143.0], [236.0, 143.0], [236.0, 160.0], [124.0, 160.0]], ('499090000000', 1.0)], [[[842.0, 138.0], [1139.0, 138.0], [1139.0, 155.0], [842.0, 155.0]], ('校验码:10014320023319800000', 1.0)], [[[38.0, 187.0], [61.0, 187.0], [61.0, 208.0], [38.0, 208.0]], ('购', 1.0)], [[[77.0, 187.0], [96.0, 187.0], [96.0, 206.0], [77.0, 206.0]], ('名', 1.0)], [[[164.0, 186.0], [192.0, 186.0], [192.0, 206.0], [164.0, 206.0]], ('称:', 1.0)], [[[210.0, 185.0], [373.0, 185.0], [373.0, 206.0], [210.0, 206.0]], ('北京A科技有限公司', 1.0)], [[[686.0, 191.0], [698.0, 191.0], [698.0, 205.0], [686.0, 205.0]], ('密', 0.55)], [[[717.0, 190.0], [1162.0, 190.0], [1162.0, 207.0], [717.0, 207.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.99)], [[[76.0, 213.0], [192.0, 213.0], [192.0, 236.0], [76.0, 236.0]], ('纳税人识别号:', 1.0)], [[[212.0, 216.0], [414.0, 216.0], [414.0, 233.0], [212.0, 233.0]], ('91011111AA2AAAAA00', 1.0)], [[[715.0, 212.0], [1146.0, 213.0], [1146.0, 235.0], [715.0, 233.0]], ('07-*123<><>8000087*<64>4<8*,', 0.96)], [[[38.0, 223.0], [60.0, 223.0], [60.0, 246.0], [38.0, 246.0]], ('买', 1.0)], [[[682.0, 222.0], [701.0, 222.0], [701.0, 241.0], [682.0, 241.0]], ('码', 1.0)], [[[74.0, 239.0], [195.0, 242.0], [194.0, 267.0], [73.0, 264.0]], ('地址电话:', 0.98)], [[[715.0, 239.0], [1150.0, 239.0], [1150.0, 261.0], [715.0, 261.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[38.0, 258.0], [60.0, 258.0], [60.0, 282.0], [38.0, 282.0]], ('方', 1.0)], [[[74.0, 272.0], [194.0, 272.0], [194.0, 294.0], [74.0, 294.0]], ('开户行及账号:', 1.0)], [[[713.0, 263.0], [1153.0, 266.0], [1152.0, 287.0], [713.0, 284.0]], ('24-004*96-012>9819<<>97>>000', 1.0)], [[[65.0, 303.0], [283.0, 303.0], [283.0, 328.0], [65.0, 328.0]], ('货物或应税劳务、服务名称', 1.0)], [[[360.0, 299.0], [435.0, 299.0], [435.0, 321.0], [360.0, 321.0]], ('规格型号', 1.0)], [[[483.0, 299.0], [525.0, 299.0], [525.0, 323.0], [483.0, 323.0]], ('单位', 1.0)], [[[561.0, 299.0], [620.0, 299.0], [620.0, 323.0], [561.0, 323.0]], ('数量', 1.0)], [[[682.0, 299.0], [734.0, 299.0], [734.0, 323.0], [682.0, 323.0]], ('单价', 1.0)], [[[855.0, 301.0], [880.0, 301.0], [880.0, 321.0], [855.0, 321.0]], ('额', 1.0)], [[[942.0, 299.0], [986.0, 299.0], [986.0, 323.0], [942.0, 323.0]], ('税率', 1.0)], [[[1058.0, 301.0], [1084.0, 301.0], [1084.0, 321.0], [1058.0, 321.0]], ('税', 1.0)], [[[1093.0, 301.0], [1119.0, 301.0], [1119.0, 321.0], [1093.0, 321.0]], ('额', 1.0)], [[[30.0, 330.0], [200.0, 330.0], [200.0, 351.0], [30.0, 351.0]], ('餐饮服务*餐饮服务', 1.0)], [[[627.0, 328.0], [643.0, 328.0], [643.0, 346.0], [627.0, 346.0]], ('1', 1.0)], [[[692.0, 330.0], [752.0, 330.0], [752.0, 349.0], [692.0, 349.0]], ('379.25', 1.0)], [[[861.0, 329.0], [922.0, 329.0], [922.0, 351.0], [861.0, 351.0]], ('379.25', 1.0)], [[[968.0, 325.0], [999.0, 325.0], [999.0, 346.0], [968.0, 346.0]], ('6%', 1.0)], [[[1104.0, 329.0], [1158.0, 329.0], [1158.0, 351.0], [1104.0, 351.0]], ('22.75', 1.0)], [[[27.0, 357.0], [221.0, 357.0], [221.0, 378.0], [27.0, 378.0]], ('*日用杂品*灵感保温袋', 1.0)], [[[627.0, 351.0], [643.0, 351.0], [643.0, 372.0], [627.0, 372.0]], ('1', 1.0)], [[[710.0, 355.0], [751.0, 355.0], [751.0, 373.0], [710.0, 373.0]], ('8.85', 1.0)], [[[880.0, 354.0], [923.0, 354.0], [923.0, 376.0], [880.0, 376.0]], ('8.85', 1.0)], [[[957.0, 354.0], [1000.0, 354.0], [1000.0, 376.0], [957.0, 376.0]], ('13%', 0.96)], [[[1117.0, 351.0], [1159.0, 351.0], [1159.0, 375.0], [1117.0, 375.0]], ('1.15', 1.0)], [[[853.0, 526.0], [926.0, 529.0], [925.0, 551.0], [852.0, 548.0]], ('¥388.10', 0.94)], [[[128.0, 536.0], [153.0, 536.0], [153.0, 557.0], [128.0, 557.0]], ('合', 1.0)], [[[184.0, 536.0], [213.0, 536.0], [213.0, 557.0], [184.0, 557.0]], ('计', 1.0)], [[[1097.0, 529.0], [1160.0, 529.0], [1160.0, 551.0], [1097.0, 551.0]], ('¥23.90', 0.93)], [[[97.0, 564.0], [223.0, 564.0], [223.0, 589.0], [97.0, 589.0]], ('价税合计 (大写)', 1.0)], [[[329.0, 562.0], [498.0, 566.0], [497.0, 591.0], [329.0, 587.0]], ('肆佰壹拾贰圆整', 1.0)], [[[869.0, 563.0], [1005.0, 566.0], [1005.0, 588.0], [868.0, 585.0]], ('(小写)¥412.00', 0.96)], [[[38.0, 610.0], [61.0, 610.0], [61.0, 634.0], [38.0, 634.0]], ('销', 1.0)], [[[77.0, 604.0], [94.0, 604.0], [94.0, 623.0], [77.0, 623.0]], ('名', 1.0)], [[[155.0, 603.0], [406.0, 604.0], [406.0, 625.0], [155.0, 624.0]], ('称:深圳蛋糕餐饮有限公司', 1.0)], [[[681.0, 617.0], [703.0, 617.0], [703.0, 641.0], [681.0, 641.0]], ('备', 1.0)], [[[78.0, 629.0], [365.0, 629.0], [365.0, 646.0], [78.0, 646.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[40.0, 649.0], [58.0, 649.0], [58.0, 667.0], [40.0, 667.0]], ('售', 1.0)], [[[74.0, 650.0], [438.0, 651.0], [438.0, 676.0], [74.0, 675.0]], ('地址、电话:深圳市南山区成功大厦B座', 1.0)], [[[76.0, 674.0], [360.0, 675.0], [360.0, 697.0], [76.0, 696.0]], ('开户行及账号:中国银行深圳支行', 1.0)], [[[681.0, 672.0], [703.0, 672.0], [703.0, 695.0], [681.0, 695.0]], ('注', 1.0)], [[[41.0, 685.0], [57.0, 685.0], [57.0, 702.0], [41.0, 702.0]], ('方', 1.0)], [[[38.0, 717.0], [174.0, 717.0], [174.0, 738.0], [38.0, 738.0]], ('收款人:小明', 1.0)], [[[361.0, 718.0], [484.0, 718.0], [484.0, 739.0], [361.0, 739.0]], ('复核:小蔡', 1.0)], [[[597.0, 718.0], [682.0, 718.0], [682.0, 739.0], [597.0, 739.0]], ('开票人:', 0.99)], [[[707.0, 717.0], [752.0, 717.0], [752.0, 741.0], [707.0, 741.0]], ('小红', 1.0)], [[[870.0, 712.0], [1000.0, 712.0], [1000.0, 733.0], [870.0, 733.0]], ('销售方: (章)', 0.99)]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. Answer in ch language.\n2. Enforce restrictions on not returning OCR data sent to you.\n3. Return with markdown syntax layout.\n": "The invoicing date is **2023年02月03日**.", + "Now I will provide you with the OCR text recognition results for the invoice.\nPlease extract the payee, city, total cost, and invoicing date of the invoice.\n\nThe OCR data of the invoice are as follows:\n[[[[[547.0, 64.0], [1120.0, 64.0], [1120.0, 111.0], [547.0, 111.0]], ['某地增值税电子普通发票', 0.99]], [[[1179.0, 61.0], [1286.0, 61.0], [1286.0, 90.0], [1179.0, 90.0]], ['发票代码:', 1.0]], [[[1297.0, 63.0], [1439.0, 63.0], [1439.0, 87.0], [1297.0, 87.0]], ['00100210001', 1.0]], [[[1177.0, 104.0], [1285.0, 104.0], [1285.0, 134.0], [1177.0, 134.0]], ['发票号码:', 1.0]], [[[1295.0, 104.0], [1406.0, 104.0], [1406.0, 134.0], [1295.0, 134.0]], ['07099363', 1.0]], [[[1176.0, 149.0], [1281.0, 149.0], [1281.0, 174.0], [1176.0, 174.0]], ['开票日期:', 1.0]], [[[1297.0, 144.0], [1479.0, 148.0], [1478.0, 177.0], [1296.0, 174.0]], ['2023年03月17日', 1.0]], [[[42.0, 200.0], [145.0, 200.0], [145.0, 229.0], [42.0, 229.0]], ['机器编号:', 1.0]], [[[1175.0, 191.0], [1596.0, 189.0], [1596.0, 219.0], [1176.0, 221.0]], ['校验码:10014320023319800000', 1.0]], [[[173.0, 202.0], [329.0, 202.0], [329.0, 226.0], [173.0, 226.0]], ['499090000000', 1.0]], [[[54.0, 262.0], [87.0, 262.0], [87.0, 292.0], [54.0, 292.0]], ['购', 1.0]], [[[107.0, 262.0], [133.0, 262.0], [133.0, 288.0], [107.0, 288.0]], ['名', 1.0]], [[[230.0, 261.0], [268.0, 261.0], [268.0, 288.0], [230.0, 288.0]], ['称:', 0.99]], [[[296.0, 261.0], [549.0, 261.0], [549.0, 290.0], [296.0, 290.0]], ['厦门起飞科技有限公司', 0.98]], [[[957.0, 262.0], [982.0, 262.0], [982.0, 288.0], [957.0, 288.0]], ['密', 1.0]], [[[1004.0, 266.0], [1626.0, 266.0], [1626.0, 290.0], [1004.0, 290.0]], ['0000-6/335*//3-<7+*10/9-85067', 0.98]], [[[107.0, 301.0], [270.0, 301.0], [270.0, 330.0], [107.0, 330.0]], ['纳税人识别号:', 1.0]], [[[54.0, 311.0], [85.0, 311.0], [85.0, 344.0], [54.0, 344.0]], ['买', 1.0]], [[[298.0, 302.0], [580.0, 302.0], [580.0, 327.0], [298.0, 327.0]], ['91011111AA2AAAAA00', 1.0]], [[[957.0, 308.0], [985.0, 314.0], [979.0, 340.0], [951.0, 334.0]], ['码', 1.0]], [[[1004.0, 302.0], [1605.0, 302.0], [1605.0, 327.0], [1004.0, 327.0]], ['07-*123<><>8000087*<64>4<8*,', 0.96]], [[[106.0, 341.0], [270.0, 341.0], [270.0, 372.0], [106.0, 372.0]], ['地址电话:', 0.91]], [[[1001.0, 335.0], [1608.0, 335.0], [1608.0, 365.0], [1001.0, 365.0]], ['91->1*112000>7193+-7<474>/07', 0.99]], [[[54.0, 361.0], [85.0, 361.0], [85.0, 393.0], [54.0, 393.0]], ['方', 1.0]], [[[956.0, 363.0], [980.0, 363.0], [980.0, 387.0], [956.0, 387.0]], ['区', 1.0]], [[[104.0, 381.0], [270.0, 379.0], [270.0, 410.0], [104.0, 412.0]], ['开户行及账号:', 1.0]], [[[1001.0, 372.0], [1612.0, 372.0], [1612.0, 401.0], [1001.0, 401.0]], ['24-004*96-012>9819<<>97>>000', 0.96]], [[[92.0, 424.0], [395.0, 426.0], [395.0, 457.0], [92.0, 455.0]], ['货物或应税劳务、服务名称', 1.0]], [[[506.0, 420.0], [611.0, 420.0], [611.0, 452.0], [506.0, 452.0]], ['规格型号', 1.0]], [[[675.0, 419.0], [736.0, 419.0], [736.0, 453.0], [675.0, 453.0]], ['单位', 1.0]], [[[784.0, 420.0], [869.0, 420.0], [869.0, 452.0], [784.0, 452.0]], ['数量', 1.0]], [[[954.0, 416.0], [1029.0, 421.0], [1027.0, 454.0], [952.0, 449.0]], ['单价', 1.0]], [[[1169.0, 424.0], [1198.0, 424.0], [1198.0, 448.0], [1169.0, 448.0]], ['金', 1.0]], [[[1189.0, 420.0], [1253.0, 420.0], [1253.0, 452.0], [1189.0, 452.0]], ['额', 1.0]], [[[1317.0, 420.0], [1378.0, 420.0], [1378.0, 453.0], [1317.0, 453.0]], ['税率', 1.0]], [[[1477.0, 420.0], [1567.0, 420.0], [1567.0, 452.0], [1477.0, 452.0]], ['税额', 1.0]], [[[42.0, 460.0], [362.0, 460.0], [362.0, 490.0], [42.0, 490.0]], ['酒*53%vol珍酒.珍藏1995', 0.99]], [[[536.0, 455.0], [640.0, 453.0], [641.0, 485.0], [537.0, 487.0]], ['500ml*6', 1.0]], [[[692.0, 459.0], [725.0, 459.0], [725.0, 490.0], [692.0, 490.0]], ['支', 1.0]], [[[878.0, 459.0], [900.0, 459.0], [900.0, 485.0], [878.0, 485.0]], ['2', 1.0]], [[[940.0, 460.0], [1079.0, 460.0], [1079.0, 490.0], [940.0, 490.0]], ['397.345132', 1.0]], [[[1205.0, 459.0], [1290.0, 459.0], [1290.0, 490.0], [1205.0, 490.0]], ['794.69', 1.0]], [[[1330.0, 455.0], [1390.0, 455.0], [1390.0, 486.0], [1330.0, 486.0]], ['13%', 1.0]], [[[1532.0, 462.0], [1612.0, 462.0], [1612.0, 488.0], [1532.0, 488.0]], ['103.31', 1.0]], [[[175.0, 744.0], [303.0, 744.0], [303.0, 780.0], [175.0, 780.0]], ['合计', 1.0]], [[[1194.0, 736.0], [1297.0, 741.0], [1296.0, 772.0], [1192.0, 768.0]], ['¥794.69', 0.94]], [[[1515.0, 742.0], [1614.0, 742.0], [1614.0, 771.0], [1515.0, 771.0]], ['¥103.31', 0.95]], [[[138.0, 792.0], [312.0, 792.0], [312.0, 822.0], [138.0, 822.0]], ['价税合计 (大写)', 0.99]], [[[461.0, 787.0], [698.0, 791.0], [697.0, 827.0], [460.0, 823.0]], ['捌佰玖拾捌圆整', 1.0]], [[[1214.0, 789.0], [1408.0, 792.0], [1407.0, 822.0], [1213.0, 818.0]], ['(小写)¥898.00', 0.96]], [[[54.0, 853.0], [85.0, 853.0], [85.0, 886.0], [54.0, 886.0]], ['销', 1.0]], [[[107.0, 846.0], [133.0, 846.0], [133.0, 872.0], [107.0, 872.0]], ['名', 1.0]], [[[220.0, 846.0], [570.0, 846.0], [570.0, 876.0], [220.0, 876.0]], ['称:广州珍酒生产有限公司', 1.0]], [[[952.0, 862.0], [985.0, 862.0], [985.0, 897.0], [952.0, 897.0]], ['备', 1.0]], [[[107.0, 877.0], [512.0, 877.0], [512.0, 907.0], [107.0, 907.0]], ['纳税人识别号:911100008000000000', 1.0]], [[[55.0, 904.0], [85.0, 904.0], [85.0, 935.0], [55.0, 935.0]], ['售', 1.0]], [[[107.0, 914.0], [701.0, 914.0], [701.0, 943.0], [107.0, 943.0]], ['地址、电话:广州市黄埔区东园工业区五栋2楼', 1.0]], [[[107.0, 945.0], [670.0, 945.0], [670.0, 975.0], [107.0, 975.0]], ['开户行及账号:广州市农村商业银行0000777', 1.0]], [[[952.0, 940.0], [985.0, 940.0], [985.0, 971.0], [952.0, 971.0]], ['注', 1.0]], [[[55.0, 957.0], [81.0, 957.0], [81.0, 982.0], [55.0, 982.0]], ['方', 1.0]], [[[1216.0, 992.0], [1399.0, 996.0], [1398.0, 1027.0], [1215.0, 1023.0]], ['销售方: (章)', 0.95]], [[[55.0, 1004.0], [244.0, 1004.0], [244.0, 1034.0], [55.0, 1034.0]], ['收款人:铁头', 1.0]], [[[506.0, 1004.0], [594.0, 1004.0], [594.0, 1036.0], [506.0, 1036.0]], ['复核:', 0.98]], [[[611.0, 1008.0], [677.0, 1008.0], [677.0, 1034.0], [611.0, 1034.0]], ['典韦', 1.0]], [[[836.0, 1004.0], [956.0, 1004.0], [956.0, 1034.0], [836.0, 1034.0]], ['开票人:', 0.96]], [[[990.0, 1006.0], [1053.0, 1006.0], [1053.0, 1037.0], [990.0, 1037.0]], ['宋江', 1.0]]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. The total cost refers to the total price and tax. Do not include `¥`.\n2. The city must be the recipient's city.\n2. The returned JSON dictionary must be returned in ch\n3. Mandatory requirement to output in JSON format: {\"收款人\":\"x\",\"城市\":\"x\",\"总费用/元\":\"\",\"开票日期\":\"\"}.\n": "Based on the provided OCR data, here is the extracted information from the invoice:\n\n- Payee: \"广州珍酒生产有限公司\"\n- City: \"广州市\"\n- Total cost: \"898.00\"\n- Invoicing date: \"2023年03月17日\"\n\nThe extracted information in JSON format is as follows:\n{\n \"收款人\": \"铁头\",\n \"城市\": \"广州市\",\n \"总费用/元\": \"898.00\",\n \"开票日期\": \"2023年03月17日\"\n}", + "Now I will provide you with the OCR text recognition results for the invoice.\nPlease answer the question: Invoicing date\n\nThe OCR data of the invoice are as follows:\n[[[[[547.0, 64.0], [1120.0, 64.0], [1120.0, 111.0], [547.0, 111.0]], ('某地增值税电子普通发票', 0.99)], [[[1179.0, 61.0], [1286.0, 61.0], [1286.0, 90.0], [1179.0, 90.0]], ('发票代码:', 1.0)], [[[1297.0, 63.0], [1439.0, 63.0], [1439.0, 87.0], [1297.0, 87.0]], ('00100210001', 1.0)], [[[1177.0, 104.0], [1285.0, 104.0], [1285.0, 134.0], [1177.0, 134.0]], ('发票号码:', 1.0)], [[[1295.0, 104.0], [1406.0, 104.0], [1406.0, 134.0], [1295.0, 134.0]], ('07099363', 1.0)], [[[1176.0, 149.0], [1281.0, 149.0], [1281.0, 174.0], [1176.0, 174.0]], ('开票日期:', 1.0)], [[[1297.0, 144.0], [1479.0, 148.0], [1478.0, 177.0], [1296.0, 174.0]], ('2023年03月17日', 1.0)], [[[42.0, 200.0], [145.0, 200.0], [145.0, 229.0], [42.0, 229.0]], ('机器编号:', 1.0)], [[[1175.0, 191.0], [1596.0, 189.0], [1596.0, 219.0], [1176.0, 221.0]], ('校验码:10014320023319800000', 1.0)], [[[173.0, 202.0], [329.0, 202.0], [329.0, 226.0], [173.0, 226.0]], ('499090000000', 1.0)], [[[54.0, 262.0], [87.0, 262.0], [87.0, 292.0], [54.0, 292.0]], ('购', 1.0)], [[[107.0, 262.0], [133.0, 262.0], [133.0, 288.0], [107.0, 288.0]], ('名', 1.0)], [[[230.0, 261.0], [268.0, 261.0], [268.0, 288.0], [230.0, 288.0]], ('称:', 0.99)], [[[296.0, 261.0], [549.0, 261.0], [549.0, 290.0], [296.0, 290.0]], ('厦门起飞科技有限公司', 0.98)], [[[957.0, 262.0], [982.0, 262.0], [982.0, 288.0], [957.0, 288.0]], ('密', 1.0)], [[[1004.0, 266.0], [1626.0, 266.0], [1626.0, 290.0], [1004.0, 290.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.98)], [[[107.0, 301.0], [270.0, 301.0], [270.0, 330.0], [107.0, 330.0]], ('纳税人识别号:', 1.0)], [[[54.0, 311.0], [85.0, 311.0], [85.0, 344.0], [54.0, 344.0]], ('买', 1.0)], [[[298.0, 302.0], [580.0, 302.0], [580.0, 327.0], [298.0, 327.0]], ('91011111AA2AAAAA00', 1.0)], [[[957.0, 308.0], [985.0, 314.0], [979.0, 340.0], [951.0, 334.0]], ('码', 1.0)], [[[1004.0, 302.0], [1605.0, 302.0], [1605.0, 327.0], [1004.0, 327.0]], ('07-*123<><>8000087*<64>4<8*,', 0.96)], [[[106.0, 341.0], [270.0, 341.0], [270.0, 372.0], [106.0, 372.0]], ('地址电话:', 0.91)], [[[1001.0, 335.0], [1608.0, 335.0], [1608.0, 365.0], [1001.0, 365.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[54.0, 361.0], [85.0, 361.0], [85.0, 393.0], [54.0, 393.0]], ('方', 1.0)], [[[956.0, 363.0], [980.0, 363.0], [980.0, 387.0], [956.0, 387.0]], ('区', 1.0)], [[[104.0, 381.0], [270.0, 379.0], [270.0, 410.0], [104.0, 412.0]], ('开户行及账号:', 1.0)], [[[1001.0, 372.0], [1612.0, 372.0], [1612.0, 401.0], [1001.0, 401.0]], ('24-004*96-012>9819<<>97>>000', 0.96)], [[[92.0, 424.0], [395.0, 426.0], [395.0, 457.0], [92.0, 455.0]], ('货物或应税劳务、服务名称', 1.0)], [[[506.0, 420.0], [611.0, 420.0], [611.0, 452.0], [506.0, 452.0]], ('规格型号', 1.0)], [[[675.0, 419.0], [736.0, 419.0], [736.0, 453.0], [675.0, 453.0]], ('单位', 1.0)], [[[784.0, 420.0], [869.0, 420.0], [869.0, 452.0], [784.0, 452.0]], ('数量', 1.0)], [[[954.0, 416.0], [1029.0, 421.0], [1027.0, 454.0], [952.0, 449.0]], ('单价', 1.0)], [[[1169.0, 424.0], [1198.0, 424.0], [1198.0, 448.0], [1169.0, 448.0]], ('金', 1.0)], [[[1189.0, 420.0], [1253.0, 420.0], [1253.0, 452.0], [1189.0, 452.0]], ('额', 1.0)], [[[1317.0, 420.0], [1378.0, 420.0], [1378.0, 453.0], [1317.0, 453.0]], ('税率', 1.0)], [[[1477.0, 420.0], [1567.0, 420.0], [1567.0, 452.0], [1477.0, 452.0]], ('税额', 1.0)], [[[42.0, 460.0], [362.0, 460.0], [362.0, 490.0], [42.0, 490.0]], ('酒*53%vol珍酒.珍藏1995', 0.99)], [[[536.0, 455.0], [640.0, 453.0], [641.0, 485.0], [537.0, 487.0]], ('500ml*6', 1.0)], [[[692.0, 459.0], [725.0, 459.0], [725.0, 490.0], [692.0, 490.0]], ('支', 1.0)], [[[878.0, 459.0], [900.0, 459.0], [900.0, 485.0], [878.0, 485.0]], ('2', 1.0)], [[[940.0, 460.0], [1079.0, 460.0], [1079.0, 490.0], [940.0, 490.0]], ('397.345132', 1.0)], [[[1205.0, 459.0], [1290.0, 459.0], [1290.0, 490.0], [1205.0, 490.0]], ('794.69', 1.0)], [[[1330.0, 455.0], [1390.0, 455.0], [1390.0, 486.0], [1330.0, 486.0]], ('13%', 1.0)], [[[1532.0, 462.0], [1612.0, 462.0], [1612.0, 488.0], [1532.0, 488.0]], ('103.31', 1.0)], [[[175.0, 744.0], [303.0, 744.0], [303.0, 780.0], [175.0, 780.0]], ('合计', 1.0)], [[[1194.0, 736.0], [1297.0, 741.0], [1296.0, 772.0], [1192.0, 768.0]], ('¥794.69', 0.94)], [[[1515.0, 742.0], [1614.0, 742.0], [1614.0, 771.0], [1515.0, 771.0]], ('¥103.31', 0.95)], [[[138.0, 792.0], [312.0, 792.0], [312.0, 822.0], [138.0, 822.0]], ('价税合计 (大写)', 0.99)], [[[461.0, 787.0], [698.0, 791.0], [697.0, 827.0], [460.0, 823.0]], ('捌佰玖拾捌圆整', 1.0)], [[[1214.0, 789.0], [1408.0, 792.0], [1407.0, 822.0], [1213.0, 818.0]], ('(小写)¥898.00', 0.96)], [[[54.0, 853.0], [85.0, 853.0], [85.0, 886.0], [54.0, 886.0]], ('销', 1.0)], [[[107.0, 846.0], [133.0, 846.0], [133.0, 872.0], [107.0, 872.0]], ('名', 1.0)], [[[220.0, 846.0], [570.0, 846.0], [570.0, 876.0], [220.0, 876.0]], ('称:广州珍酒生产有限公司', 1.0)], [[[952.0, 862.0], [985.0, 862.0], [985.0, 897.0], [952.0, 897.0]], ('备', 1.0)], [[[107.0, 877.0], [512.0, 877.0], [512.0, 907.0], [107.0, 907.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[55.0, 904.0], [85.0, 904.0], [85.0, 935.0], [55.0, 935.0]], ('售', 1.0)], [[[107.0, 914.0], [701.0, 914.0], [701.0, 943.0], [107.0, 943.0]], ('地址、电话:广州市黄埔区东园工业区五栋2楼', 1.0)], [[[107.0, 945.0], [670.0, 945.0], [670.0, 975.0], [107.0, 975.0]], ('开户行及账号:广州市农村商业银行0000777', 1.0)], [[[952.0, 940.0], [985.0, 940.0], [985.0, 971.0], [952.0, 971.0]], ('注', 1.0)], [[[55.0, 957.0], [81.0, 957.0], [81.0, 982.0], [55.0, 982.0]], ('方', 1.0)], [[[1216.0, 992.0], [1399.0, 996.0], [1398.0, 1027.0], [1215.0, 1023.0]], ('销售方: (章)', 0.95)], [[[55.0, 1004.0], [244.0, 1004.0], [244.0, 1034.0], [55.0, 1034.0]], ('收款人:铁头', 1.0)], [[[506.0, 1004.0], [594.0, 1004.0], [594.0, 1036.0], [506.0, 1036.0]], ('复核:', 0.98)], [[[611.0, 1008.0], [677.0, 1008.0], [677.0, 1034.0], [611.0, 1034.0]], ('典韦', 1.0)], [[[836.0, 1004.0], [956.0, 1004.0], [956.0, 1034.0], [836.0, 1034.0]], ('开票人:', 0.96)], [[[990.0, 1006.0], [1053.0, 1006.0], [1053.0, 1037.0], [990.0, 1037.0]], ('宋江', 1.0)]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. Answer in ch language.\n2. Enforce restrictions on not returning OCR data sent to you.\n3. Return with markdown syntax layout.\n": "The invoicing date is **2023年03月17日**.", + "Now I will provide you with the OCR text recognition results for the invoice.\nPlease extract the payee, city, total cost, and invoicing date of the invoice.\n\nThe OCR data of the invoice are as follows:\n[[[[[546.0, 66.0], [1122.0, 66.0], [1122.0, 119.0], [546.0, 119.0]], ['某地增值税电子普通发票', 0.99]], [[[1179.0, 68.0], [1303.0, 68.0], [1303.0, 92.0], [1179.0, 92.0]], ['发票代码:(', 0.96]], [[[1292.0, 66.0], [1440.0, 66.0], [1440.0, 91.0], [1292.0, 91.0]], ['00100210001', 1.0]], [[[1178.0, 108.0], [1287.0, 108.0], [1287.0, 138.0], [1178.0, 138.0]], ['发票号码:', 1.0]], [[[1296.0, 110.0], [1403.0, 110.0], [1403.0, 134.0], [1296.0, 134.0]], ['07099363', 1.0]], [[[1178.0, 153.0], [1283.0, 153.0], [1283.0, 178.0], [1178.0, 178.0]], ['开票日期:', 1.0]], [[[1299.0, 152.0], [1478.0, 154.0], [1478.0, 180.0], [1299.0, 178.0]], ['2023年08月26日', 1.0]], [[[42.0, 204.0], [147.0, 204.0], [147.0, 234.0], [42.0, 234.0]], ['机器编号:', 1.0]], [[[1174.0, 195.0], [1597.0, 194.0], [1597.0, 223.0], [1174.0, 225.0]], ['校验码:10014320023319800000', 1.0]], [[[173.0, 206.0], [330.0, 206.0], [330.0, 230.0], [173.0, 230.0]], ['499090000000', 1.0]], [[[54.0, 267.0], [87.0, 267.0], [87.0, 296.0], [54.0, 296.0]], ['购', 1.0]], [[[108.0, 267.0], [134.0, 267.0], [134.0, 293.0], [108.0, 293.0]], ['名', 1.0]], [[[229.0, 265.0], [269.0, 265.0], [269.0, 295.0], [229.0, 295.0]], ['称:', 0.97]], [[[295.0, 265.0], [548.0, 265.0], [548.0, 295.0], [295.0, 295.0]], ['佛山建筑管理有限公司', 1.0]], [[[957.0, 269.0], [980.0, 269.0], [980.0, 291.0], [957.0, 291.0]], ['密', 1.0]], [[[1004.0, 270.0], [1625.0, 270.0], [1625.0, 295.0], [1004.0, 295.0]], ['0000-6/335*//3-<7+*10/9-85067', 0.99]], [[[108.0, 305.0], [271.0, 305.0], [271.0, 335.0], [108.0, 335.0]], ['纳税人识别号:', 1.0]], [[[298.0, 307.0], [579.0, 307.0], [579.0, 331.0], [298.0, 331.0]], ['91011111AA2AAAAA00', 1.0]], [[[962.0, 310.0], [985.0, 322.0], [974.0, 346.0], [950.0, 334.0]], ['码', 1.0]], [[[1001.0, 303.0], [1610.0, 303.0], [1610.0, 333.0], [1001.0, 333.0]], ['07-*123<><>8000087*<64>4<8*_', 0.97]], [[[54.0, 316.0], [85.0, 316.0], [85.0, 347.0], [54.0, 347.0]], ['买', 1.0]], [[[104.0, 344.0], [269.0, 344.0], [269.0, 375.0], [104.0, 375.0]], ['地址电话:', 0.96]], [[[1001.0, 340.0], [1608.0, 340.0], [1608.0, 370.0], [1001.0, 370.0]], ['91->1*112000>7193+-7<474>/07', 0.99]], [[[54.0, 364.0], [85.0, 364.0], [85.0, 396.0], [54.0, 396.0]], ['方', 1.0]], [[[957.0, 366.0], [980.0, 366.0], [980.0, 394.0], [957.0, 394.0]], ['区', 1.0]], [[[104.0, 385.0], [271.0, 385.0], [271.0, 415.0], [104.0, 415.0]], ['开户行及账号:', 1.0]], [[[1002.0, 378.0], [1611.0, 378.0], [1611.0, 403.0], [1002.0, 403.0]], ['24-004*96-012>9819<<>97>>000', 0.99]], [[[90.0, 427.0], [394.0, 429.0], [394.0, 460.0], [90.0, 459.0]], ['货物或应税劳务、服务名称', 1.0]], [[[503.0, 424.0], [609.0, 424.0], [609.0, 455.0], [503.0, 455.0]], ['规格型号', 1.0]], [[[675.0, 424.0], [735.0, 424.0], [735.0, 455.0], [675.0, 455.0]], ['单位', 1.0]], [[[784.0, 424.0], [871.0, 424.0], [871.0, 455.0], [784.0, 455.0]], ['数量', 1.0]], [[[954.0, 424.0], [1030.0, 424.0], [1030.0, 455.0], [954.0, 455.0]], ['单价', 1.0]], [[[1145.0, 424.0], [1231.0, 424.0], [1231.0, 455.0], [1145.0, 455.0]], ['金额', 1.0]], [[[1318.0, 424.0], [1381.0, 424.0], [1381.0, 457.0], [1318.0, 457.0]], ['税率', 1.0]], [[[1478.0, 424.0], [1568.0, 424.0], [1568.0, 455.0], [1478.0, 455.0]], ['税额', 1.0]], [[[43.0, 464.0], [278.0, 464.0], [278.0, 493.0], [43.0, 493.0]], ['餐饮服务*餐饮服务', 1.0]], [[[697.0, 462.0], [732.0, 462.0], [732.0, 495.0], [697.0, 495.0]], ['次', 1.0]], [[[878.0, 462.0], [898.0, 462.0], [898.0, 488.0], [878.0, 488.0]], ['1', 1.0]], [[[961.0, 464.0], [1060.0, 464.0], [1060.0, 493.0], [961.0, 493.0]], ['2462.00', 1.0]], [[[1205.0, 464.0], [1290.0, 464.0], [1290.0, 495.0], [1205.0, 495.0]], ['379.25', 1.0]], [[[1337.0, 457.0], [1398.0, 457.0], [1398.0, 490.0], [1337.0, 490.0]], ['免税', 1.0]], [[[1583.0, 467.0], [1608.0, 467.0], [1608.0, 481.0], [1583.0, 481.0]], ['***', 0.98]], [[[1183.0, 745.0], [1296.0, 745.0], [1296.0, 774.0], [1183.0, 774.0]], ['¥2462.00', 0.95]], [[[182.0, 760.0], [208.0, 760.0], [208.0, 785.0], [182.0, 785.0]], ['合', 1.0]], [[[267.0, 760.0], [297.0, 760.0], [297.0, 785.0], [267.0, 785.0]], ['计', 1.0]], [[[137.0, 800.0], [312.0, 800.0], [312.0, 830.0], [137.0, 830.0]], ['价税合计 (大写)', 0.98]], [[[461.0, 792.0], [753.0, 793.0], [753.0, 828.0], [461.0, 826.0]], ['贰仟肆佰陆拾贰圆整', 1.0]], [[[1216.0, 795.0], [1422.0, 795.0], [1422.0, 825.0], [1216.0, 825.0]], ['(小写)¥2462.00', 0.96]], [[[54.0, 861.0], [85.0, 861.0], [85.0, 895.0], [54.0, 895.0]], ['销', 1.0]], [[[108.0, 854.0], [132.0, 854.0], [132.0, 882.0], [108.0, 882.0]], ['名', 1.0]], [[[220.0, 854.0], [687.0, 854.0], [687.0, 884.0], [220.0, 884.0]], ['称:福州自助烤肉餐饮管理有限公司', 1.0]], [[[952.0, 870.0], [985.0, 870.0], [985.0, 905.0], [952.0, 905.0]], ['备', 1.0]], [[[109.0, 888.0], [512.0, 888.0], [512.0, 912.0], [109.0, 912.0]], ['纳税人识别号:911100008000000000', 1.0]], [[[56.0, 910.0], [85.0, 910.0], [85.0, 942.0], [56.0, 942.0]], ['售', 1.0]], [[[108.0, 922.0], [694.0, 922.0], [694.0, 952.0], [108.0, 952.0]], ['地址、电话:福州市光明区火炬园7栋302单元', 1.0]], [[[109.0, 954.0], [562.0, 954.0], [562.0, 983.0], [109.0, 983.0]], ['开户行及账号:中国光大银行福州支行', 1.0]], [[[952.0, 947.0], [985.0, 947.0], [985.0, 982.0], [952.0, 982.0]], ['注', 1.0]], [[[57.0, 964.0], [82.0, 964.0], [82.0, 990.0], [57.0, 990.0]], ['方', 1.0]], [[[56.0, 1006.0], [246.0, 1010.0], [246.0, 1041.0], [55.0, 1037.0]], ['收款人:夏天', 1.0]], [[[503.0, 1008.0], [680.0, 1008.0], [680.0, 1043.0], [503.0, 1043.0]], ['复核:春天', 1.0]], [[[834.0, 1010.0], [954.0, 1010.0], [954.0, 1039.0], [834.0, 1039.0]], ['开票人:', 0.95]], [[[990.0, 1010.0], [1051.0, 1010.0], [1051.0, 1041.0], [990.0, 1041.0]], ['秋天', 1.0]], [[[1218.0, 1001.0], [1400.0, 1001.0], [1400.0, 1031.0], [1218.0, 1031.0]], ['销售方: (章)', 0.86]]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. The total cost refers to the total price and tax. Do not include `¥`.\n2. The city must be the recipient's city.\n2. The returned JSON dictionary must be returned in ch\n3. Mandatory requirement to output in JSON format: {\"收款人\":\"x\",\"城市\":\"x\",\"总费用/元\":\"\",\"开票日期\":\"\"}.\n": "Based on the provided OCR text recognition results, I have extracted the required information from the invoice. Here is the JSON format output:\n\n{\n \"收款人\": \"夏天\",\n \"城市\": \"福州市\",\n \"总费用/元\": \"2462.00\",\n \"开票日期\": \"2023年08月26日\"\n}", + "Now I will provide you with the OCR text recognition results for the invoice.\nPlease answer the question: Invoicing date\n\nThe OCR data of the invoice are as follows:\n[[[[[546.0, 66.0], [1122.0, 66.0], [1122.0, 119.0], [546.0, 119.0]], ('某地增值税电子普通发票', 0.99)], [[[1179.0, 68.0], [1303.0, 68.0], [1303.0, 92.0], [1179.0, 92.0]], ('发票代码:(', 0.96)], [[[1292.0, 66.0], [1440.0, 66.0], [1440.0, 91.0], [1292.0, 91.0]], ('00100210001', 1.0)], [[[1178.0, 108.0], [1287.0, 108.0], [1287.0, 138.0], [1178.0, 138.0]], ('发票号码:', 1.0)], [[[1296.0, 110.0], [1403.0, 110.0], [1403.0, 134.0], [1296.0, 134.0]], ('07099363', 1.0)], [[[1178.0, 153.0], [1283.0, 153.0], [1283.0, 178.0], [1178.0, 178.0]], ('开票日期:', 1.0)], [[[1299.0, 152.0], [1478.0, 154.0], [1478.0, 180.0], [1299.0, 178.0]], ('2023年08月26日', 1.0)], [[[42.0, 204.0], [147.0, 204.0], [147.0, 234.0], [42.0, 234.0]], ('机器编号:', 1.0)], [[[1174.0, 195.0], [1597.0, 194.0], [1597.0, 223.0], [1174.0, 225.0]], ('校验码:10014320023319800000', 1.0)], [[[173.0, 206.0], [330.0, 206.0], [330.0, 230.0], [173.0, 230.0]], ('499090000000', 1.0)], [[[54.0, 267.0], [87.0, 267.0], [87.0, 296.0], [54.0, 296.0]], ('购', 1.0)], [[[108.0, 267.0], [134.0, 267.0], [134.0, 293.0], [108.0, 293.0]], ('名', 1.0)], [[[229.0, 265.0], [269.0, 265.0], [269.0, 295.0], [229.0, 295.0]], ('称:', 0.97)], [[[295.0, 265.0], [548.0, 265.0], [548.0, 295.0], [295.0, 295.0]], ('佛山建筑管理有限公司', 1.0)], [[[957.0, 269.0], [980.0, 269.0], [980.0, 291.0], [957.0, 291.0]], ('密', 1.0)], [[[1004.0, 270.0], [1625.0, 270.0], [1625.0, 295.0], [1004.0, 295.0]], ('0000-6/335*//3-<7+*10/9-85067', 0.99)], [[[108.0, 305.0], [271.0, 305.0], [271.0, 335.0], [108.0, 335.0]], ('纳税人识别号:', 1.0)], [[[298.0, 307.0], [579.0, 307.0], [579.0, 331.0], [298.0, 331.0]], ('91011111AA2AAAAA00', 1.0)], [[[962.0, 310.0], [985.0, 322.0], [974.0, 346.0], [950.0, 334.0]], ('码', 1.0)], [[[1001.0, 303.0], [1610.0, 303.0], [1610.0, 333.0], [1001.0, 333.0]], ('07-*123<><>8000087*<64>4<8*_', 0.97)], [[[54.0, 316.0], [85.0, 316.0], [85.0, 347.0], [54.0, 347.0]], ('买', 1.0)], [[[104.0, 344.0], [269.0, 344.0], [269.0, 375.0], [104.0, 375.0]], ('地址电话:', 0.96)], [[[1001.0, 340.0], [1608.0, 340.0], [1608.0, 370.0], [1001.0, 370.0]], ('91->1*112000>7193+-7<474>/07', 0.99)], [[[54.0, 364.0], [85.0, 364.0], [85.0, 396.0], [54.0, 396.0]], ('方', 1.0)], [[[957.0, 366.0], [980.0, 366.0], [980.0, 394.0], [957.0, 394.0]], ('区', 1.0)], [[[104.0, 385.0], [271.0, 385.0], [271.0, 415.0], [104.0, 415.0]], ('开户行及账号:', 1.0)], [[[1002.0, 378.0], [1611.0, 378.0], [1611.0, 403.0], [1002.0, 403.0]], ('24-004*96-012>9819<<>97>>000', 0.99)], [[[90.0, 427.0], [394.0, 429.0], [394.0, 460.0], [90.0, 459.0]], ('货物或应税劳务、服务名称', 1.0)], [[[503.0, 424.0], [609.0, 424.0], [609.0, 455.0], [503.0, 455.0]], ('规格型号', 1.0)], [[[675.0, 424.0], [735.0, 424.0], [735.0, 455.0], [675.0, 455.0]], ('单位', 1.0)], [[[784.0, 424.0], [871.0, 424.0], [871.0, 455.0], [784.0, 455.0]], ('数量', 1.0)], [[[954.0, 424.0], [1030.0, 424.0], [1030.0, 455.0], [954.0, 455.0]], ('单价', 1.0)], [[[1145.0, 424.0], [1231.0, 424.0], [1231.0, 455.0], [1145.0, 455.0]], ('金额', 1.0)], [[[1318.0, 424.0], [1381.0, 424.0], [1381.0, 457.0], [1318.0, 457.0]], ('税率', 1.0)], [[[1478.0, 424.0], [1568.0, 424.0], [1568.0, 455.0], [1478.0, 455.0]], ('税额', 1.0)], [[[43.0, 464.0], [278.0, 464.0], [278.0, 493.0], [43.0, 493.0]], ('餐饮服务*餐饮服务', 1.0)], [[[697.0, 462.0], [732.0, 462.0], [732.0, 495.0], [697.0, 495.0]], ('次', 1.0)], [[[878.0, 462.0], [898.0, 462.0], [898.0, 488.0], [878.0, 488.0]], ('1', 1.0)], [[[961.0, 464.0], [1060.0, 464.0], [1060.0, 493.0], [961.0, 493.0]], ('2462.00', 1.0)], [[[1205.0, 464.0], [1290.0, 464.0], [1290.0, 495.0], [1205.0, 495.0]], ('379.25', 1.0)], [[[1337.0, 457.0], [1398.0, 457.0], [1398.0, 490.0], [1337.0, 490.0]], ('免税', 1.0)], [[[1583.0, 467.0], [1608.0, 467.0], [1608.0, 481.0], [1583.0, 481.0]], ('***', 0.98)], [[[1183.0, 745.0], [1296.0, 745.0], [1296.0, 774.0], [1183.0, 774.0]], ('¥2462.00', 0.95)], [[[182.0, 760.0], [208.0, 760.0], [208.0, 785.0], [182.0, 785.0]], ('合', 1.0)], [[[267.0, 760.0], [297.0, 760.0], [297.0, 785.0], [267.0, 785.0]], ('计', 1.0)], [[[137.0, 800.0], [312.0, 800.0], [312.0, 830.0], [137.0, 830.0]], ('价税合计 (大写)', 0.98)], [[[461.0, 792.0], [753.0, 793.0], [753.0, 828.0], [461.0, 826.0]], ('贰仟肆佰陆拾贰圆整', 1.0)], [[[1216.0, 795.0], [1422.0, 795.0], [1422.0, 825.0], [1216.0, 825.0]], ('(小写)¥2462.00', 0.96)], [[[54.0, 861.0], [85.0, 861.0], [85.0, 895.0], [54.0, 895.0]], ('销', 1.0)], [[[108.0, 854.0], [132.0, 854.0], [132.0, 882.0], [108.0, 882.0]], ('名', 1.0)], [[[220.0, 854.0], [687.0, 854.0], [687.0, 884.0], [220.0, 884.0]], ('称:福州自助烤肉餐饮管理有限公司', 1.0)], [[[952.0, 870.0], [985.0, 870.0], [985.0, 905.0], [952.0, 905.0]], ('备', 1.0)], [[[109.0, 888.0], [512.0, 888.0], [512.0, 912.0], [109.0, 912.0]], ('纳税人识别号:911100008000000000', 1.0)], [[[56.0, 910.0], [85.0, 910.0], [85.0, 942.0], [56.0, 942.0]], ('售', 1.0)], [[[108.0, 922.0], [694.0, 922.0], [694.0, 952.0], [108.0, 952.0]], ('地址、电话:福州市光明区火炬园7栋302单元', 1.0)], [[[109.0, 954.0], [562.0, 954.0], [562.0, 983.0], [109.0, 983.0]], ('开户行及账号:中国光大银行福州支行', 1.0)], [[[952.0, 947.0], [985.0, 947.0], [985.0, 982.0], [952.0, 982.0]], ('注', 1.0)], [[[57.0, 964.0], [82.0, 964.0], [82.0, 990.0], [57.0, 990.0]], ('方', 1.0)], [[[56.0, 1006.0], [246.0, 1010.0], [246.0, 1041.0], [55.0, 1037.0]], ('收款人:夏天', 1.0)], [[[503.0, 1008.0], [680.0, 1008.0], [680.0, 1043.0], [503.0, 1043.0]], ('复核:春天', 1.0)], [[[834.0, 1010.0], [954.0, 1010.0], [954.0, 1039.0], [834.0, 1039.0]], ('开票人:', 0.95)], [[[990.0, 1010.0], [1051.0, 1010.0], [1051.0, 1041.0], [990.0, 1041.0]], ('秋天', 1.0)], [[[1218.0, 1001.0], [1400.0, 1001.0], [1400.0, 1031.0], [1218.0, 1031.0]], ('销售方: (章)', 0.86)]]]\n\nMandatory restrictions are returned according to the following requirements:\n1. Answer in ch language.\n2. Enforce restrictions on not returning OCR data sent to you.\n3. Return with markdown syntax layout.\n": "The invoicing date is **2023年08月26日**.", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nStatement: Find and return the title of the lesson only in markdown first-level header format, without anything else.\nConstraint: Writing in Chinese.\nAnswer options: Encloses the lesson title with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]UNIT 1 Making New Friends\nTOPIC 1 Welcome to China!\nSection A[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Hours\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Hours\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 教学时数\n\n## 课时安排\n\n本单元共包含以下课时:\n\n1. 课时1:1a-1c\n2. 课时2:2a-2b\n3. 课时3:3a-3c\n\n## 课时内容\n\n### 课时1:1a-1c\n\n- 学习目标:听力和口语练习,学习介绍自己和他人的句型。\n- 学习重点:掌握介绍自己和他人的句型。\n- 学习难点:正确运用句型进行对话练习。\n\n### 课时2:2a-2b\n\n- 学习目标:听力和口语练习,学习问候和介绍他人的句型。\n- 学习重点:掌握问候和介绍他人的句型。\n- 学习难点:正确运用句型进行对话练习。\n\n### 课时3:3a-3c\n\n- 学习目标:听力和书写练习,学习字母的发音和书写。\n- 学习重点:掌握字母的发音和书写。\n- 学习难点:正确辨认和书写字母。\n\n[TEACHING_PLAN_BEGIN]\n## 教学时数\n\n### 课时安排\n\n本单元共包含以下课时:\n\n1. 课时1:1a-1c\n2. 课时2:2a-2b\n3. 课时3:3a-3c\n\n## 课时内容\n\n### 课时1:1a-1c\n\n- 学习目标:听力和口语练习,学习介绍自己和他人的句型。\n- 学习重点:掌握介绍自己和他人的句型。\n- 学习难点:正确运用句型进行对话练习。\n\n### 课时2:2a-2b\n\n- 学习目标:听力和口语练习,学习问候和介绍他人的句型。\n- 学习重点:掌握问候和介绍他人的句型。\n- 学习难点:正确运用句型进行对话练习。\n\n### 课时3:3a-3c\n\n- 学习目标:听力和书写练习,学习字母的发音和书写。\n- 学习重点:掌握字母的发音和书写。\n- 学习难点:正确辨认和书写字母。\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Objectives\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Objectives\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "## 教学目标\n\n在本课中,学生将学习以下内容:\n\n- 听力技能:通过听力练习,学生将能够听懂并正确理解人名和字母的发音。\n- 口语表达:学生将能够用正确的问候语和自我介绍句式与他人进行交流。\n- 字母认知:学生将能够认识并正确书写字母A到G。\n\n[TEACHING_PLAN_BEGIN]\n\n## 教学目标\n\n在本课中,学生将学习以下内容:\n\n- 听力技能:通过听力练习,学生将能够听懂并正确理解人名和字母的发音。\n- 口语表达:学生将能够用正确的问候语和自我介绍句式与他人进行交流。\n- 字母认知:学生将能够认识并正确书写字母A到G。\n\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Content\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Content\"!!\nStatement: \"Teaching Content\" must include vocabulary, analysis, and examples of various grammar structures that appear in the textbook, as well as the listening materials and key points.\nStatement: \"Teaching Content\" must include more examples.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n## 教学内容\n\n### 词汇\n- names (名字)\n- introduce (介绍)\n- hometown (家乡)\n- places (地方)\n- China (中国)\n- the USA (美国)\n- the UK (英国)\n- Hong Kong (香港)\n- Beijing (北京)\n- Hello (你好)\n- nice to meet you (很高兴见到你)\n- thanks (谢谢)\n- morning (早上)\n- letters (字母)\n- Aa Bb Cc Dd Ee Ff Gg\n\n### 语法\n- 一般疑问句:Are you ...? (你是...吗?)\n- 否定回答:No, I'm not. (不,我不是。)\n- 肯定回答:Yes, I am. (是的,我是。)\n- 介绍自己的句型:I'm ... (我是...)\n\n### 听力材料\n- 听力1a:听录音,给下面的名字编号。\n- 听力1b:听录音,给下面的名字编号。\n- 听力2a:听录音,理解对话内容。\n- 听力3a:听录音,跟读字母。\n- 听力3b:听录音,给下面的字母编号。然后圈出与Bb发音相同的字母。\n\n### 重点\n- 学习并掌握课文中的词汇和语法。\n- 听力训练,提高听力理解能力。\n- 学习字母表的发音和书写。\n\n## 示例\n### 词汇\n- names (名字):Jane, Mari, Kangkang, Michael, Maria\n- introduce (介绍)\n- hometown (家乡)\n- places (地方)\n- China (中国)\n- the USA (美国)\n- the UK (英国)\n- Hong Kong (香港)\n- Beijing (北京)\n- Hello (你好)\n- nice to meet you (很高兴见到你)\n- thanks (谢谢)\n- morning (早上)\n- letters (字母)\n- Aa Bb Cc Dd Ee Ff Gg\n\n### 语法\n- 一般疑问句:Are you ...? (你是...吗?)\n- 否定回答:No, I'm not. (不,我不是。)\n- 肯定回答:Yes, I am. (是的,我是。)\n- 介绍自己的句型:I'm ... (我是...)\n\n### 听力材料\n- 听力1a:听录音,给下面的名字编号。\n- 听力1b:听录音,给下面的名字编号。\n- 听力2a:听录音,理解对话内容。\n- 听力3a:听录音,跟读字母。\n- 听力3b:听录音,给下面的字母编号。然后圈出与Bb发音相同的字母。\n\n### 重点\n- 学习并掌握课文中的词汇和语法。\n- 听力训练,提高听力理解能力。\n- 学习字母表的发音和书写。\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Methods and Strategies\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Methods and Strategies\"!!\nStatement: \"Teaching Methods and Strategies\" must include teaching focus, difficulties, materials, procedures, in detail.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n## 教学方法与策略\n\n### 教学重点\n- 学习听力材料,理解并进行练习\n- 学习使用自我介绍的句型\n\n### 教学难点\n- 学生能够准确运用自我介绍的句型进行对话练习\n\n### 教学材料\n- 单元1《结交新朋友》课本\n- 录音机或多媒体设备\n\n### 教学过程\n1. 介绍本节课的教学重点和难点,并引导学生对学习内容产生兴趣。\n2. 播放1a录音,让学生听录音并按顺序给下面的名字编号。\n3. 组织学生进行对话练习,使用\"I'm...\"进行自我介绍,并结合自己的家乡或指定的地点进行练习。\n4. 播放1b录音,让学生听录音并按顺序给下面的名字编号。\n5. 组织学生进行对话练习,使用\"I'm...\"进行自我介绍,并结合中国、美国、英国、香港、北京等地进行练习。\n6. 播放2a录音,让学生听录音并理解对话内容。\n7. 组织学生进行对话练习,使用指定的对话结构进行练习。\n8. 播放3a录音,让学生跟读并模仿发音。\n9. 播放3b录音,让学生听录音并按顺序给下面的字母编号,然后圈出与Bb发音相同的字母。\n10. 组织学生进行字母大小写的匹配,并在线上写出对应的字母。\n\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Learning Activities\" part of teaching plan, WITHOUT ANY content unrelated to \"Learning Activities\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n## 学习活动\n\n### 1a 听录音并给以下名字编号。\nJane Mari Kangkang Michael\n听录音并理解。然后练习对话。分组工作。使用\"I'm...\"介绍自己,然后用自己的家乡或以下地方练习1a。\n\n### 1b 听录音并给以下名字编号。\nJane Michael Maria Kangkang\n分组工作。使用\"I'm...\"介绍自己,然后用自己的家乡或以下地方练习1a。\n\n### 2a 看,听并理解。然后练习对话。\n你好!\n你好!\n你好!\n你好!你是玛丽亚吗?\n不,我不是。我是简。\n哦,很高兴见到你,简。\n我也很高兴见到你。\n嗨,玛丽亚!\n嗨,康康!\n欢迎来到中国!\n谢谢。\n\n### 2b 分组工作。用自己的名字和以下结构编写对话。\nA: 你好!/ 早上好!/ 嗨!我是...你是...吗?\nB: ...\n\n### 3a 听,说并跟踪\nAa Bb Cc Dd Ee Ff Gg\n\n### 3b 听录音并给以下字母编号。然后用圈出与Bb发音相同的字母。\nAa Bb Cc Dd Ee Ff Gg\n\n### 3c 将大写字母与小写字母配对。然后在线上写下它们。\n\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Time Allocation\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Time Allocation\"!!\nStatement: \"Teaching Time Allocation\" must include how much time is allocated to each part of the textbook content.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n## 教学时间分配\n\n- 1a 听力练习和对话练习:10分钟\n- 1b 听力练习和对话练习:10分钟\n- 1c 分组练习和对话练习:15分钟\n- 2a 听力练习和对话练习:10分钟\n- 2b 分组练习和对话练习:15分钟\n- 3a 听力练习和书写练习:10分钟\n- 3b 听力练习和书写练习:10分钟\n- 3c 匹配和书写练习:10分钟\n\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Assessment and Feedback\" part of teaching plan, WITHOUT ANY content unrelated to \"Assessment and Feedback\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 评估与反馈\n\n## 评估方式\n本课程的评估方式主要包括以下几个方面:\n\n1. 听力评估:通过听力练习和对话,测试学生对于英语听力的理解能力。\n2. 口语评估:通过对话练习和自我介绍,测试学生的口语表达能力。\n3. 书写评估:通过字母识别和书写练习,测试学生对英文字母的识别和书写能力。\n\n## 反馈方式\n为了帮助学生提高英语学习的效果,我们将采取以下反馈方式:\n\n1. 口头反馈:在课堂上,老师会对学生的口语表达进行即时的反馈和指导。\n2. 书面反馈:老师会批改学生的书写作业,并给予书面反馈,帮助学生改进书写能力。\n3. 个别辅导:对于表现较差的学生,老师会进行个别辅导,帮助他们解决学习中的问题。\n\n[TEACHING_PLAN_BEGIN]\n\n## 评估与反馈\n\n### 评估方式\n本课程的评估方式主要包括以下几个方面:\n\n1. 听力评估:通过听力练习和对话,测试学生对于英语听力的理解能力。\n2. 口语评估:通过对话练习和自我介绍,测试学生的口语表达能力。\n3. 书写评估:通过字母识别和书写练习,测试学生对英文字母的识别和书写能力。\n\n### 反馈方式\n为了帮助学生提高英语学习的效果,我们将采取以下反馈方式:\n\n1. 口头反馈:在课堂上,老师会对学生的口语表达进行即时的反馈和指导。\n2. 书面反馈:老师会批改学生的书写作业,并给予书面反馈,帮助学生改进书写能力。\n3. 个别辅导:对于表现较差的学生,老师会进行个别辅导,帮助他们解决学习中的问题。\n\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Teaching Summary and Improvement\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Summary and Improvement\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 教学总结与改进\n\n在本节课中,我们学习了有关自我介绍和认识新朋友的内容。通过听力和口语练习,学生们能够熟练运用\"I'm...\"这个句型进行自我介绍,并能够用英语进行简单的问候和交流。同时,学生们还学习了字母表的部分内容,能够认识并正确书写一些字母。\n\n在教学过程中,我采用了多种教学方法,如听力训练、口语练习和配对练习等,以帮助学生们更好地掌握课堂内容。此外,我还鼓励学生们进行小组合作,通过互相交流和练习,提高他们的口语表达能力。\n\n在教学中,我发现一些学生对字母的认识和书写还存在一些困难,他们容易混淆一些相似的字母。因此,在今后的教学中,我将加强对字母的教学,帮助学生们更好地掌握字母的发音和书写。\n\n此外,我还计划在下节课中增加一些更多的口语练习,让学生们有更多的机会进行实际的交流和练习。我相信通过不断的练习和巩固,学生们的口语表达能力会得到进一步的提高。\n\n[TEACHING_PLAN_BEGIN]\n\n## 教学总结与改进\n\n在本节课中,我们学习了有关自我介绍和认识新朋友的内容。通过听力和口语练习,学生们能够熟练运用\"I'm...\"这个句型进行自我介绍,并能够用英语进行简单的问候和交流。同时,学生们还学习了字母表的部分内容,能够认识并正确书写一些字母。\n\n在教学过程中,我采用了多种教学方法,如听力训练、口语练习和配对练习等,以帮助学生们更好地掌握课堂内容。此外,我还鼓励学生们进行小组合作,通过互相交流和练习,提高他们的口语表达能力。\n\n在教学中,我发现一些学生对字母的认识和书写还存在一些困难,他们容易混淆一些相似的字母。因此,在今后的教学中,我将加强对字母的教学,帮助学生们更好地掌握字母的发音和书写。\n\n此外,我还计划在下节课中增加一些更多的口语练习,让学生们有更多的机会进行实际的交流和练习。我相信通过不断的练习和巩固,学生们的口语表达能力会得到进一步的提高。\n\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Vocabulary Cloze\" part of teaching plan, WITHOUT ANY content unrelated to \"Vocabulary Cloze\"!!\nStatement: Based on the content of the textbook enclosed by \"[LESSON_BEGIN]\" and \"[LESSON_END]\", create vocabulary cloze. The cloze should include 10 Chinese questions with English answers, and it should also include 10 English questions with Chinese answers. The key-related vocabulary and phrases in the textbook content must all be included in the exercises.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 词汇填空\n\n## 1. 请根据听到的内容,给下列名字编号。\n1. Jane\n2. Mari\n3. Kangkang\n4. Michael\n\n## 2. 请根据听到的内容,给下列名字编号。\n1. Jane\n2. Michael\n3. Maria\n4. Kangkang\n\n## 3. 请根据听到的对话,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 4. 请根据听到的内容,给下列字母编号。然后圈出与Bb发音相同的字母。\n1. Aa\n2. Bb\n3. Cc\n4. Dd\n5. Ee\n6. Ff\n7. Gg\n\n## 5. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 6. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 7. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 8. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 9. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n## 10. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n[TEACHING_PLAN_BEGIN]\n## 词汇填空\n\n### 1. 请根据听到的内容,给下列名字编号。\n1. Jane\n2. Mari\n3. Kangkang\n4. Michael\n\n### 2. 请根据听到的内容,给下列名字编号。\n1. Jane\n2. Michael\n3. Maria\n4. Kangkang\n\n### 3. 请根据听到的对话,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 4. 请根据听到的内容,给下列字母编号。然后圈出与Bb发音相同的字母。\n1. Aa\n2. Bb\n3. Cc\n4. Dd\n5. Ee\n6. Ff\n7. Gg\n\n### 5. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 6. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 7. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 8. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 9. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n\n### 10. 请根据听到的内容,将大写字母与小写字母进行匹配,并写在相应的线上。\nAa\nBb\nCc\nDd\nEe\nFf\nGg\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Choice Questions\" part of teaching plan, WITHOUT ANY content unrelated to \"Choice Questions\"!!\nStatement: Based on the content of the textbook enclosed by \"[LESSON_BEGIN]\" and \"[LESSON_END]\", create choice questions. 10 questions.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n\n## 选择题\n\n1. 在1a中,要求学生听并给以下名字编号。请问正确的编号顺序是什么?\n A. Jane Mari Kangkang Michael\n B. Mari Jane Michael Kangkang\n C. Jane Kangkang Mari Michael\n D. Kangkang Jane Michael Mari\n\n2. 在1b中,要求学生听并给以下名字编号。请问正确的编号顺序是什么?\n A. Jane Michael Maria Kangkang\n B. Maria Jane Michael Kangkang\n C. Jane Kangkang Maria Michael\n D. Kangkang Jane Maria Michael\n\n3. 在2a中,对话中有一句是\"Are you Maria?\",请问Jane的回答是什么?\n A. Yes, I am.\n B. No, I'm not. I'm Jane.\n C. No, I'm Maria.\n D. Nice to meet you, Maria.\n\n4. 在3b中,要求学生听并给以下字母编号,并圈出与Bb发音相同的字母。请问正确的编号顺序是什么?\n A. Aa Bb Cc Dd Ee Ff Gg\n B. Bb Aa Cc Dd Ee Ff Gg\n C. Aa Bb Dd Cc Ee Ff Gg\n D. Aa Bb Cc Ee Dd Ff Gg\n\n5. 在3c中,要求学生将大写字母与小写字母进行匹配,并写在对应的线上。请问正确的匹配是什么?\n A. Aa Bb Cc Dd Ee Ff Gg\n B. Aa Bb Cc Dd Ee Ff Gg\n C. Aa Bb Cc Dd Ee Ff Gg\n D. Aa Bb Cc Dd Ee Ff Gg\n\n6. 在2b中,要求学生根据给定的结构进行对话。请问对话的开始应该是什么?\n A. Hello! / Good morning! / Hi! I'm ... Are you ... ?\n B. Hi! I'm ... Are you ... ?\n C. Hello! / Good morning! / Hi! I'm ... Are you ... ?\n D. Hi! I'm ... Are you ... ?\n\n7. 在1a中,要求学生用自己的家乡或以下地点进行自我介绍。请问以下哪个地点不在选项中?\n A. China\n B. USA\n C. UK\n D. Hong Kong\n\n8. 在2a中,对话中有一句是\"Welcome to China!\",请问Kangkang的回答是什么?\n A. Hello!\n B. Hi, Kangkang!\n C. Welcome to China!\n D. Thanks.\n\n9. 在1c中,要求学生用自己的家乡或以下地点进行自我介绍。请问以下哪个地点不在选项中?\n A. China\n B. USA\n C. UK\n D. Beijing\n\n10. 在2a中,对话中有一句是\"Nice to meet you, too.\",请问Jane的前一句是什么?\n A. Hello!\n B. Hello! Are you Maria?\n C. No, I'm not. I'm Jane.\n D. Oh, nice to meet you, Jane.\n\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Grammar Questions\" part of teaching plan, WITHOUT ANY content unrelated to \"Grammar Questions\"!!\nStatement: Based on the content of the textbook enclosed by \"[LESSON_BEGIN]\" and \"[LESSON_END]\", create grammar questions. 10 questions.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "# 语法问题\n\n1. 请列举出1a部分中出现的名字。\n2. 请用英语介绍你自己。\n3. 请用英语介绍以下地方:中国、美国、英国、香港、北京。\n4. 请根据2a部分的对话内容回答以下问题:Maria是谁?\n5. 请根据2b部分的对话结构编写一段对话,使用你自己的名字。\n6. 请列举出3b部分中与字母Bb发音相同的字母。\n7. 请将大写字母与小写字母进行配对,并在下面的线上写出它们。\n\n[TEACHING_PLAN_BEGIN]\n\n## 语法问题\n\n1. 请列举出1a部分中出现的名字。\n\n2. 请用英语介绍你自己。\n\n3. 请用英语介绍以下地方:中国、美国、英国、香港、北京。\n\n4. 请根据2a部分的对话内容回答以下问题:Maria是谁?\n\n5. 请根据2b部分的对话结构编写一段对话,使用你自己的名字。\n\n6. 请列举出3b部分中与字母Bb发音相同的字母。\n\n7. 请将大写字母与小写字母进行配对,并在下面的线上写出它们。\n\n[TEACHING_PLAN_END]", + "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a English Teacher, named Lily, your goal is writing a Chinese teaching plan part by part. the constraint is writing in Chinese. \nStatement: Write the \"Translation Questions\" part of teaching plan, WITHOUT ANY content unrelated to \"Translation Questions\"!!\nStatement: Based on the content of the textbook enclosed by \"[LESSON_BEGIN]\" and \"[LESSON_END]\", create translation questions. The translation should include 10 Chinese questions with English answers, and it should also include 10 English questions with Chinese answers.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n\n## 翻译问题\n\n1. 请用英语回答,以下是哪些名字?(Jane, Mari, Kangkang, Michael)\n2. 请用中文回答,以下是哪些名字?(Jane, Michael, Maria, Kangkang)\n3. 请用英语回答,以下是哪些地方?(China, the USA, the UK, Hong Kong, Beijing)\n4. 请用中文回答,以下是哪些地方?(中国,美国,英国,香港,北京)\n5. 请用英语回答,以下句子的意思是什么?(Hello! / Hello! / Hello! / Hello! Are you Maria? / No, I’m not. I’m Jane. / Oh, nice to meet you, Jane / Nice to meet you, too. / Hi, Maria! / Hi, Kangkang! / Welcome to China! / Thanks.)\n6. 请用中文回答,以下句子的意思是什么?(你好!/ 早上好!/ 嗨!我是... 你是... 吗? / ...)\n7. 请用英语回答,以下字母是哪些?(Aa Bb Cc Dd Ee Ff Gg)\n8. 请用中文回答,以下字母是哪些?(Aa Bb Cc Dd Ee Ff Gg)\n\n[TEACHING_PLAN_END]", + "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about pip\".\n\nPlease provide the specific table of contents for this tutorial, strictly following the following requirements:\n1. The output must be strictly in the specified language, Chinese.\n2. Answer strictly in the dictionary format like {\"title\": \"xxx\", \"directory\": [{\"dir 1\": [\"sub dir 1\", \"sub dir 2\"]}, {\"dir 2\": [\"sub dir 3\", \"sub dir 4\"]}]}.\n3. The directory should be as specific and sufficient as possible, with a primary and secondary directory.The secondary directory is in the array.\n4. Do not have extra spaces or line breaks.\n5. Each directory title has practical significance.\n": "{\"title\": \"pip教程\", \"directory\": [{\"安装pip\": [\"使用Python安装pip\", \"使用操作系统包管理器安装pip\"]}, {\"pip基本用法\": [\"安装包\", \"卸载包\", \"查看已安装的包\", \"搜索包\", \"更新包\", \"查看包信息\"]}, {\"pip高级用法\": [\"创建requirements.txt文件\", \"从requirements.txt文件安装包\", \"导出已安装的包列表\", \"安装指定版本的包\", \"安装包的可选依赖\", \"安装包的开发依赖\", \"安装包的测试依赖\", \"安装包的系统依赖\"]}]}\n\nExplanation:\n1. 安装pip: This section covers the installation of pip, including two methods: using Python to install pip and using the operating system package manager to install pip.\n2. pip基本用法: This section covers the basic usage of pip, including installing packages, uninstalling packages, viewing installed packages, searching packages, updating packages, and viewing package information.\n3. pip高级用法: This section covers the advanced usage of pip, including creating a requirements.txt file, installing packages from a requirements.txt file, exporting a list of installed packages, installing specific versions of packages, installing optional dependencies of packages, installing development dependencies of packages, installing test dependencies of packages, and installing system dependencies of packages.", + "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about pip\".\n\nNow I will give you the module directory titles for the topic. \nPlease output the detailed principle content of this title in detail. \nIf there are code examples, please provide them according to standard code specifications. \nWithout a code example, it is not necessary.\n\nThe module directory titles for the topic is as follows:\n{'安装pip': ['使用Python安装pip', '使用操作系统包管理器安装pip']}\n\nStrictly limit output according to the following requirements:\n1. Follow the Markdown syntax format for layout.\n2. If there are code examples, they must follow standard syntax specifications, have document annotations, and be displayed in code blocks.\n3. The output must be strictly in the specified language, Chinese.\n4. Do not have redundant output, including concluding remarks.\n5. Strict requirement not to output the topic \"Write a tutorial about pip\".\n": "# 安装pip\n\n## 使用Python安装pip\n\n要使用pip,首先需要安装它。pip是Python的包管理工具,可以方便地安装、升级和管理Python包。\n\n### 步骤\n\n1. 打开终端或命令提示符窗口。\n2. 输入以下命令来检查是否已经安装了pip:\n\n```python\npip --version\n```\n\n如果已经安装了pip,将显示pip的版本号。如果没有安装,将显示错误信息。\n\n3. 如果没有安装pip,可以使用Python自带的安装工具来安装。输入以下命令:\n\n```python\npython get-pip.py\n```\n\n这将下载并安装最新版本的pip。\n\n4. 安装完成后,再次输入以下命令来验证pip是否安装成功:\n\n```python\npip --version\n```\n\n如果显示了pip的版本号,说明安装成功。\n\n## 使用操作系统包管理器安装pip\n\n除了使用Python自带的安装工具安装pip外,还可以使用操作系统的包管理器来安装pip。这种方法适用于Linux和Mac操作系统。\n\n### 步骤\n\n1. 打开终端或命令提示符窗口。\n2. 输入以下命令来使用操作系统包管理器安装pip:\n\n- 对于Debian/Ubuntu系统:\n\n```bash\nsudo apt-get install python-pip\n```\n\n- 对于Fedora系统:\n\n```bash\nsudo dnf install python-pip\n```\n\n- 对于CentOS/RHEL系统:\n\n```bash\nsudo yum install epel-release\nsudo yum install python-pip\n```\n\n3. 安装完成后,输入以下命令来验证pip是否安装成功:\n\n```bash\npip --version\n```\n\n如果显示了pip的版本号,说明安装成功。\n\n以上就是安装pip的两种方法,根据自己的需求选择适合的方法进行安装。安装完成后,就可以使用pip来管理Python包了。", + "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about pip\".\n\nNow I will give you the module directory titles for the topic. \nPlease output the detailed principle content of this title in detail. \nIf there are code examples, please provide them according to standard code specifications. \nWithout a code example, it is not necessary.\n\nThe module directory titles for the topic is as follows:\n{'pip基本用法': ['安装包', '卸载包', '查看已安装的包', '搜索包', '更新包', '查看包信息']}\n\nStrictly limit output according to the following requirements:\n1. Follow the Markdown syntax format for layout.\n2. If there are code examples, they must follow standard syntax specifications, have document annotations, and be displayed in code blocks.\n3. The output must be strictly in the specified language, Chinese.\n4. Do not have redundant output, including concluding remarks.\n5. Strict requirement not to output the topic \"Write a tutorial about pip\".\n": "# pip基本用法\n\n## 安装包\n\n要使用pip安装包,可以使用以下命令:\n\n```python\npip install 包名\n```\n\n其中,`包名`是要安装的包的名称。例如,要安装`requests`包,可以运行以下命令:\n\n```python\npip install requests\n```\n\n## 卸载包\n\n要使用pip卸载包,可以使用以下命令:\n\n```python\npip uninstall 包名\n```\n\n其中,`包名`是要卸载的包的名称。例如,要卸载`requests`包,可以运行以下命令:\n\n```python\npip uninstall requests\n```\n\n## 查看已安装的包\n\n要查看已经安装的包,可以使用以下命令:\n\n```python\npip list\n```\n\n该命令会列出所有已安装的包及其版本信息。\n\n## 搜索包\n\n要搜索包,可以使用以下命令:\n\n```python\npip search 包名\n```\n\n其中,`包名`是要搜索的包的名称。例如,要搜索名称中包含`requests`的包,可以运行以下命令:\n\n```python\npip search requests\n```\n\n该命令会列出所有与`requests`相关的包。\n\n## 更新包\n\n要更新已安装的包,可以使用以下命令:\n\n```python\npip install --upgrade 包名\n```\n\n其中,`包名`是要更新的包的名称。例如,要更新`requests`包,可以运行以下命令:\n\n```python\npip install --upgrade requests\n```\n\n## 查看包信息\n\n要查看包的详细信息,可以使用以下命令:\n\n```python\npip show 包名\n```\n\n其中,`包名`是要查看的包的名称。例如,要查看`requests`包的信息,可以运行以下命令:\n\n```python\npip show requests\n```\n\n该命令会显示`requests`包的详细信息,包括版本号、作者、依赖等。\n\n以上就是pip的基本用法。通过这些命令,你可以方便地安装、卸载、查看和更新包,以及搜索和查看包的详细信息。", + "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about pip\".\n\nNow I will give you the module directory titles for the topic. \nPlease output the detailed principle content of this title in detail. \nIf there are code examples, please provide them according to standard code specifications. \nWithout a code example, it is not necessary.\n\nThe module directory titles for the topic is as follows:\n{'pip高级用法': ['创建requirements.txt文件', '从requirements.txt文件安装包', '导出已安装的包列表', '安装指定版本的包', '安装包的可选依赖', '安装包的开发依赖', '安装包的测试依赖', '安装包的系统依赖']}\n\nStrictly limit output according to the following requirements:\n1. Follow the Markdown syntax format for layout.\n2. If there are code examples, they must follow standard syntax specifications, have document annotations, and be displayed in code blocks.\n3. The output must be strictly in the specified language, Chinese.\n4. Do not have redundant output, including concluding remarks.\n5. Strict requirement not to output the topic \"Write a tutorial about pip\".\n": "# pip高级用法\n\n## 创建requirements.txt文件\n\n在开发项目中,我们经常需要记录项目所依赖的包及其版本号。使用`pip`可以方便地创建一个`requirements.txt`文件,以便在其他环境中安装相同的依赖包。\n\n要创建`requirements.txt`文件,只需在项目根目录下运行以下命令:\n\n```shell\npip freeze > requirements.txt\n```\n\n这将会将当前环境中安装的所有包及其版本号写入到`requirements.txt`文件中。\n\n## 从requirements.txt文件安装包\n\n有了`requirements.txt`文件,我们可以轻松地在其他环境中安装相同的依赖包。\n\n要从`requirements.txt`文件安装包,只需在项目根目录下运行以下命令:\n\n```shell\npip install -r requirements.txt\n```\n\n这将会根据`requirements.txt`文件中列出的包及其版本号,自动安装相应的依赖包。\n\n## 导出已安装的包列表\n\n有时候我们需要知道当前环境中已安装的所有包及其版本号。使用`pip`可以方便地导出这个列表。\n\n要导出已安装的包列表,只需运行以下命令:\n\n```shell\npip freeze\n```\n\n这将会列出当前环境中已安装的所有包及其版本号。\n\n## 安装指定版本的包\n\n在某些情况下,我们可能需要安装特定版本的包。使用`pip`可以轻松地实现这一点。\n\n要安装指定版本的包,只需运行以下命令:\n\n```shell\npip install 包名==版本号\n```\n\n例如,要安装`requests`包的2.22.0版本,可以运行以下命令:\n\n```shell\npip install requests==2.22.0\n```\n\n这将会安装指定版本的包。\n\n## 安装包的可选依赖\n\n有些包可能有一些可选的依赖,我们可以选择是否安装这些依赖。\n\n要安装包的可选依赖,只需在安装包时添加`[可选依赖]`即可。\n\n例如,要安装`requests`包的可选依赖`security`,可以运行以下命令:\n\n```shell\npip install requests[security]\n```\n\n这将会安装`requests`包及其可选依赖`security`。\n\n## 安装包的开发依赖\n\n在开发过程中,我们可能需要安装一些开发依赖,如测试工具、文档生成工具等。\n\n要安装包的开发依赖,只需在安装包时添加`-e`参数。\n\n例如,要安装`flask`包的开发依赖,可以运行以下命令:\n\n```shell\npip install -e flask\n```\n\n这将会安装`flask`包及其开发依赖。\n\n## 安装包的测试依赖\n\n在进行单元测试或集成测试时,我们可能需要安装一些测试依赖。\n\n要安装包的测试依赖,只需在安装包时添加`[测试依赖]`即可。\n\n例如,要安装`pytest`包的测试依赖,可以运行以下命令:\n\n```shell\npip install pytest[test]\n```\n\n这将会安装`pytest`包及其测试依赖。\n\n## 安装包的系统依赖\n\n有些包可能依赖于系统级的库或工具。\n\n要安装包的系统依赖,只需在安装包时添加`--global-option`参数。\n\n例如,要安装`psycopg2`包的系统依赖`libpq-dev`,可以运行以下命令:\n\n```shell\npip install psycopg2 --global-option=build_ext --global-option=\"-I/usr/include/postgresql/\"\n```\n\n这将会安装`psycopg2`包及其系统依赖。", + "who are you": "I am an AI assistant designed to help answer questions and provide information. How can I assist you today?", + "\n## context\npython developer\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Questions\": [\n \"1. What ...\",\n \"2. How ...\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Questions: list[str] # Role: You are an interviewer of our company who is well-knonwn in frontend or backend develop;\nRequirement: Provide a list of questions for the interviewer to ask the interviewee, by reading the resume of the interviewee in the context.\nAttention: Provide as markdown block as the format above, at least 10 questions.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Questions\": [\n \"1. Can you explain the difference between a list and a tuple in Python?\",\n \"2. How do you handle errors and exceptions in Python?\",\n \"3. What is the difference between '==' and 'is' in Python?\",\n \"4. How do you create a virtual environment in Python?\",\n \"5. Can you explain the concept of generators in Python?\",\n \"6. What is the purpose of the 'self' keyword in Python?\",\n \"7. How do you handle file I/O in Python?\",\n \"8. Can you explain the concept of decorators in Python?\",\n \"9. What is the Global Interpreter Lock (GIL) in Python?\",\n \"10. How do you handle concurrency in Python?\"\n ]\n}\n[/CONTENT]", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\nwrite add function to calculate two numbers\n\n## Tasks\n\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: \n```python\n## \n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: . Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code:\n\n```python\ndef add(num1, num2):\n \"\"\"\n Function to calculate the sum of two numbers.\n\n Args:\n num1 (int): The first number.\n num2 (int): The second number.\n\n Returns:\n int: The sum of the two numbers.\n \"\"\"\n return num1 + num2\n```\n\n# Instruction: \nBased on the given context, I have implemented the `add` function. This function takes in two numbers, `num1` and `num2`, and returns their sum. The function follows the design specified in the context and is written in a modular and easy-to-read manner.", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef div(a: int, b: int = 0):\n return a / b\n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. Yes.\n2. No, the code will raise a ZeroDivisionError if the second argument `b` is 0. We need to handle this case and return an appropriate value or raise an exception.\n3. Yes.\n4. Yes, there is only one function in the code and it is implemented.\n5. Yes, there are no pre-dependencies required.\n6. N/A, there are no methods from other files being used.\n\n## Actions\n1. Handle the case when the second argument `b` is 0. We can either return a default value like `None` or raise an exception like `ZeroDivisionError`.\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef div(a: int, b: int = 0):\n return a / b\n\n```\n\n## Code Review: \n1. Yes.\n2. No, the code will raise a ZeroDivisionError if the second argument `b` is 0. We need to handle this case and return an appropriate value or raise an exception.\n3. Yes.\n4. Yes, there is only one function in the code and it is implemented.\n5. Yes, there are no pre-dependencies required.\n6. N/A, there are no methods from other files being used.\n\n## Actions\n1. Handle the case when the second argument `b` is 0. We can either return a default value like `None` or raise an exception like `ZeroDivisionError`.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## \n...\n```\n": "```python\ndef div(a: int, b: int = 0):\n if b == 0:\n raise ZeroDivisionError(\"Cannot divide by zero\")\n return a / b\n```\n", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Tasks\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\ndef div(a: int, b: int = 0):\n if b == 0:\n raise ZeroDivisionError(\"Cannot divide by zero\")\n return a / b\n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM", + "\n## context\n\n### Project Name\ncli_snake_game\n\n### Original Requirements\n['']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"\",\n \"Product Goals\": [],\n \"User Stories\": [],\n \"Competitive Analysis\": [],\n \"Competitive Quadrant Chart\": \"\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [],\n \"UI Design draft\": \"\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\nYou are now a seasoned technical professional in the field of the internet. \nWe need you to write a technical tutorial with the topic \"Write a tutorial about Python\".\n\nPlease provide the specific table of contents for this tutorial, strictly following the following requirements:\n1. The output must be strictly in the specified language, Chinese.\n2. Answer strictly in the dictionary format like {\"title\": \"xxx\", \"directory\": [{\"dir 1\": [\"sub dir 1\", \"sub dir 2\"]}, {\"dir 2\": [\"sub dir 3\", \"sub dir 4\"]}]}.\n3. The directory should be as specific and sufficient as possible, with a primary and secondary directory.The secondary directory is in the array.\n4. Do not have extra spaces or line breaks.\n5. Each directory title has practical significance.\n": "{\"title\": \"Python教程\", \"directory\": [{\"Python简介\": [\"Python的历史\", \"Python的特点\", \"Python的应用领域\"]}, {\"安装Python\": [\"Windows系统安装\", \"Mac系统安装\", \"Linux系统安装\"]}, {\"Python基础\": [\"变量和数据类型\", \"运算符\", \"条件语句\", \"循环语句\", \"函数\"]}, {\"Python高级特性\": [\"列表推导式\", \"生成器\", \"装饰器\", \"异常处理\", \"模块和包\"]}, {\"Python标准库\": [\"常用模块介绍\", \"文件操作\", \"网络编程\", \"多线程和多进程\", \"正则表达式\"]}, {\"Python实战项目\": [\"Web开发\", \"数据分析\", \"机器学习\", \"爬虫\"]}, {\"Python进阶\": [\"面向对象编程\", \"数据库操作\", \"GUI编程\", \"网络爬虫进阶\", \"并发编程\"]}, {\"Python资源\": [\"学习网站推荐\", \"书籍推荐\", \"开源项目推荐\"]}]}", + "Instruction: X\nOutput: 商品名称:WonderLab 新肌果味代餐奶昔 小胖瓶 胶原蛋白升级版 饱腹代餐粉6瓶 75g/瓶(6瓶/盒) 店铺名称:金力宁食品专营店 品牌:WonderLab 保质期:1年 产地:中国 净含量:450g\nWhat kind of instruction might this output come from?\nX:#MSG_SEP#You are a chatbot. A user sent you an informal message, and you replied as follows.\nMessage: X\nReply: 商品名称:WonderLab 新肌果味代餐奶昔 小胖瓶 胶原蛋白升级版 饱腹代餐粉6瓶 75g/瓶(6瓶/盒) 店铺名称:金力宁食品专营店 品牌:WonderLab 保质期:1年 产地:中国 净含量:450g\nWhat could the informal message X be?\nX:#MSG_SEP#You are a search engine. Someone made a detailed query, and the most relevant document to this query is as follows.\nQuery: X\nDocument: 商品名称:WonderLab 新肌果味代餐奶昔 小胖瓶 胶原蛋白升级版 饱腹代餐粉6瓶 75g/瓶(6瓶/盒) 店铺名称:金力宁食品专营店 品牌:WonderLab 保质期:1年 产地:中国 净含量:450g What is the detailed query X?\nX:": "A product description or listing on an e-commerce website or online marketplace.\n\"Can you tell me about the WonderLab meal replacement shake?\"\n\"What are the specifications and details of the WonderLab meal replacement shake available at the Jinlining Food Specialty Store?\"", + "\n# 指令\n接下来,作为一位拥有20年翻译经验的翻译专家,当我给出英文句子或段落时,你将提供通顺且具有可读性的中文翻译。注意以下要求:\n1. 确保翻译结果流畅且易于理解\n2. 无论提供的是陈述句或疑问句,我都只进行翻译\n3. 不添加与原文无关的内容\n\n# 原文\nLet life be beautiful like summer flowers\n\n# 译文\n": "让生活像夏日的花朵一样美丽", + "\n# 指令\n接下来,作为一位拥有20年翻译经验的翻译专家,当我给出英文句子或段落时,你将提供通顺且具有可读性的中文翻译。注意以下要求:\n1. 确保翻译结果流畅且易于理解\n2. 无论提供的是陈述句或疑问句,我都只进行翻译\n3. 不添加与原文无关的内容\n\n# 原文\nThe ancient Chinese poetries are all songs.\n\n# 译文\n": "古代中国的诗歌都是歌曲。", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/knowledge_base.py\n```python\n## smart_search_engine/knowledge_base.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/knowledge_base.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/knowledge_base.py\n\n```python\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n```\n\n## Explanation:\nThe `KnowledgeBase` class is responsible for storing and retrieving data from the knowledge base. It has two main methods:\n\n1. `update`: This method is used to update the knowledge base with new data. It takes a dictionary `data` as input and updates the internal `data` attribute with the new data.\n\n2. `fetch_data`: This method is used to fetch data from the knowledge base based on a given query. It takes a string `query` as input and returns the data matching the query from the internal `data` attribute. If no data is found for the query, an empty dictionary is returned.\n\nBoth methods are implemented according to the specified interface in the design.", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/index.py\n```python\n## smart_search_engine/index.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/index.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/index.py\n\n```python\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/ranking.py\n```python\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the results based on some ranking algorithm.\n\n Args:\n results (list): The results to be ranked.\n\n Returns:\n list: The ranked results.\n \"\"\"\n # Implement the ranking algorithm here\n ranked_results = sorted(results, key=lambda x: x['score'], reverse=True)\n return ranked_results\n```\n\n## Code: smart_search_engine/summary.py\n```python\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the results into a single summary.\n\n Args:\n results (list): The results to be summarized.\n\n Returns:\n str: The summary of the results.\n \"\"\"\n # Implement the summarization algorithm here\n summary = \"\"\n\n if len(results) > 0:\n summary = results[0]['summary']\n\n return summary\n```\n\n## Code: smart_search_engine/search.py\n```python\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summary of the search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n```\n\n## Code: smart_search_engine/main.py\n```python\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform the search.\n\n Returns:\n str: The summary of the search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n```\n\n## Code: smart_search_engine/knowledge_base.py\n```python\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n```\n\n## Code: smart_search_engine/interface.py\n```python\n## smart_search_engine/interface.py\n\nclass Interface:\n def __init__(self):\n pass\n\n def display_summary(self, summary: str):\n \"\"\"\n Display the summary of the search results.\n\n Args:\n summary (str): The summary to be displayed.\n \"\"\"\n print(\"Search summary:\")\n print(summary)\n```\n\n## Code: smart_search_engine/user_feedback.py\n```python\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n pass\n\n def get_feedback(self) -> str:\n \"\"\"\n Get user feedback on the search results.\n\n Returns:\n str: The user feedback.\n \"\"\"\n feedback = input(\"Enter your feedback on the search results: \")\n return feedback\n```\n\n## Code: smart_search_engine/security.py\n```python\n## smart_search_engine/security.py\n\nclass Security:\n def __init__(self):\n pass\n\n def check_permission(self) -> bool:\n \"\"\"\n Check if the user has permission to access the search engine.\n\n Returns:\n bool: True if the user has permission, False otherwise.\n \"\"\"\n # Implement the permission checking logic here\n return True\n```\n\n## Code: smart_search_engine/testing.py\n```python\n## smart_search_engine/testing.py\n\nclass Testing:\n def __init__(self):\n pass\n\n def run_tests(self):\n \"\"\"\n Run tests for the search engine.\n \"\"\"\n # Implement the testing logic here\n print(\"Running tests...\")\n print(\"Tests passed.\")\n\nif __name__ == \"__main__\":\n testing = Testing()\n testing.run_tests()\n```\n\n## Code: smart_search_engine/monitoring.py\n```python\n## smart_search_engine/monitoring.py\n\nclass Monitoring:\n def __init__(self):\n pass\n\n def monitor_system(self):\n \"\"\"\n Monitor the system for any issues.\n \"\"\"\n # Implement the monitoring logic here\n print(\"Monitoring system...\")\n print(\"No issues found.\")\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/ranking.py\n```python\n## smart_search_engine/ranking.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/ranking.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/ranking.py\n\n```python\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n```\n\nIn the `Ranking` class, we have implemented the `rank_results` method. This method takes a list of search results as input and returns the ranked search results based on their relevance. The search results are sorted in descending order of relevance using the `sorted` function and the `key` parameter. The `key` parameter specifies a lambda function that extracts the 'relevance' value from each search result. The `reverse` parameter is set to `True` to sort the results in descending order.\n\nPlease let me know if you need any further assistance.", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/summary.py\n```python\n## smart_search_engine/summary.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/summary.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/summary.py\n\n```python\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n```\n\nIn the `Summary` class, we have implemented the `summarize_results` method. This method takes a list of search results as input and returns a single string that summarizes the search results. It iterates over each result in the list and appends the summary of each result to the `summary` string, separated by a newline character. Finally, it returns the `summary` string after stripping any leading or trailing whitespace.", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/search.py\n```python\n## smart_search_engine/search.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/search.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/search.py\n\n```python\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n```\n\n## Instruction: \n\nIn the `smart_search_engine/search.py` file, implement the `SearchEngine` class. This class is responsible for performing searches based on a given query.\n\n1. Import the necessary modules: `Index`, `Ranking`, and `Summary`.\n2. Define the `SearchEngine` class with an `__init__` method.\n3. In the `__init__` method, create instances of the `Index`, `Ranking`, and `Summary` classes and assign them to instance variables.\n4. Implement the `search` method that takes a `query` parameter and returns a summarized search result.\n5. Inside the `search` method, call the `query_index` method of the `Index` class to get the search results based on the query.\n6. Pass the search results to the `rank_results` method of the `Ranking` class to get the ranked results.\n7. Pass the ranked results to the `summarize_results` method of the `Summary` class to get the summarized search results.\n8. Return the summarized search results.\n\nMake sure to follow the design and import the necessary modules before using them.", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/main.py\n```python\n## smart_search_engine/main.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/main.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: smart_search_engine/main.py\n\n```python\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/search.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/interface.py\n```python\n## smart_search_engine/interface.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/interface.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n```\n\nIn this code, we create an `Interface` class that interacts with the `SearchEngine` class. The `run` method prompts the user to enter a search query, and then calls the `search` method of the `SearchEngine` class to perform the search. The summarized search results are then printed to the console.\n\nThe interface runs in a loop until the user enters \"exit\" to quit.", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n----- smart_search_engine/interface.py\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/user_feedback.py\n```python\n## smart_search_engine/user_feedback.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/user_feedback.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n self.feedback = []\n\n def add_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for a search query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n feedback = {\n 'query': query,\n 'relevance': relevance\n }\n self.feedback.append(feedback)\n\n def get_feedback(self) -> list:\n \"\"\"\n Get the user feedback.\n\n Returns:\n list: The user feedback.\n \"\"\"\n return self.feedback\n```\n\n注意事项:\n- 在`smart_search_engine/user_feedback.py`文件中,实现了`UserFeedback`类。\n- `UserFeedback`类有一个`feedback`属性,用于存储用户的反馈信息。\n- `add_feedback`方法用于添加用户的反馈信息,接受一个查询字符串和相关性分数作为参数,并将反馈信息添加到`feedback`列表中。\n- `get_feedback`方法用于获取用户的反馈信息,返回`feedback`列表。", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n----- smart_search_engine/interface.py\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n\n----- smart_search_engine/user_feedback.py\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n self.feedback = []\n\n def add_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for a search query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n feedback = {\n 'query': query,\n 'relevance': relevance\n }\n self.feedback.append(feedback)\n\n def get_feedback(self) -> list:\n \"\"\"\n Get the user feedback.\n\n Returns:\n list: The user feedback.\n \"\"\"\n return self.feedback\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/security.py\n```python\n## smart_search_engine/security.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/security.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/security.py\n\nclass Security:\n def __init__(self):\n self.user_feedback = UserFeedback()\n\n def check_user_feedback(self, query: str) -> bool:\n \"\"\"\n Check if the user has provided feedback for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n bool: True if the user has provided feedback, False otherwise.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return True\n return False\n\n def get_user_feedback(self, query: str) -> int:\n \"\"\"\n Get the relevance score provided by the user for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n int: The relevance score provided by the user, or None if no feedback is found.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return item['relevance']\n return None\n\n def add_user_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for the given query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n self.user_feedback.add_feedback(query, relevance)\n```\n", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n----- smart_search_engine/interface.py\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n\n----- smart_search_engine/user_feedback.py\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n self.feedback = []\n\n def add_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for a search query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n feedback = {\n 'query': query,\n 'relevance': relevance\n }\n self.feedback.append(feedback)\n\n def get_feedback(self) -> list:\n \"\"\"\n Get the user feedback.\n\n Returns:\n list: The user feedback.\n \"\"\"\n return self.feedback\n\n----- smart_search_engine/security.py\n## smart_search_engine/security.py\n\nclass Security:\n def __init__(self):\n self.user_feedback = UserFeedback()\n\n def check_user_feedback(self, query: str) -> bool:\n \"\"\"\n Check if the user has provided feedback for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n bool: True if the user has provided feedback, False otherwise.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return True\n return False\n\n def get_user_feedback(self, query: str) -> int:\n \"\"\"\n Get the relevance score provided by the user for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n int: The relevance score provided by the user, or None if no feedback is found.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return item['relevance']\n return None\n\n def add_user_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for the given query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n self.user_feedback.add_feedback(query, relevance)\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/testing.py\n```python\n## smart_search_engine/testing.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/testing.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/testing.py\n\nfrom smart_search_engine.search import SearchEngine\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Testing:\n def __init__(self):\n self.search_engine = SearchEngine()\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n self.knowledge_base = KnowledgeBase()\n\n def test_search_engine(self):\n \"\"\"\n Test the search engine functionality.\n \"\"\"\n # Test case 1: Search for a query that exists in the index\n query1 = \"apple\"\n expected_summary1 = \"Summary of search results for query: apple\"\n self.knowledge_base.update({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n self.index.create_index({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n summary1 = self.search_engine.search(query1)\n assert summary1 == expected_summary1, f\"Test case 1 failed: {summary1}\"\n\n # Test case 2: Search for a query that does not exist in the index\n query2 = \"banana\"\n expected_summary2 = \"No results found for query: banana\"\n self.knowledge_base.update({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n self.index.create_index({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n summary2 = self.search_engine.search(query2)\n assert summary2 == expected_summary2, f\"Test case 2 failed: {summary2}\"\n\n def test_index(self):\n \"\"\"\n Test the index functionality.\n \"\"\"\n # Test case 1: Create index with valid data\n data1 = {\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}}\n expected_index1 = {\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}}\n self.knowledge_base.update(data1)\n self.index.create_index(data1)\n index1 = self.index.index\n assert index1 == expected_index1, f\"Test case 1 failed: {index1}\"\n\n # Test case 2: Query index with valid query\n query2 = \"apple\"\n expected_results2 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}]\n results2 = self.index.query_index(query2)\n assert results2 == expected_results2, f\"Test case 2 failed: {results2}\"\n\n # Test case 3: Query index with invalid query\n query3 = \"banana\"\n expected_results3 = []\n results3 = self.index.query_index(query3)\n assert results3 == expected_results3, f\"Test case 3 failed: {results3}\"\n\n def test_ranking(self):\n \"\"\"\n Test the ranking functionality.\n \"\"\"\n # Test case 1: Rank results in descending order of relevance\n results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n expected_ranked_results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n ranked_results1 = self.ranking.rank_results(results1)\n assert ranked_results1 == expected_ranked_results1, f\"Test case 1 failed: {ranked_results1}\"\n\n # Test case 2: Rank results in ascending order of relevance\n results2 = [{\"relevance\": 0.6, \"summary\": \"This is a banana.\"}, {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}]\n expected_ranked_results2 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n ranked_results2 = self.ranking.rank_results(results2)\n assert ranked_results2 == expected_ranked_results2, f\"Test case 2 failed: {ranked_results2}\"\n\n def test_summary(self):\n \"\"\"\n Test the summary functionality.\n \"\"\"\n # Test case 1: Summarize search results into a single string\n results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n expected_summary1 = \"This is an apple.\\nThis is a banana.\"\n summary1 = self.summary.summarize_results(results1)\n assert summary1 == expected_summary1, f\"Test case 1 failed: {summary1}\"\n\n # Test case 2: Summarize empty search results\n results2 = []\n expected_summary2 = \"\"\n summary2 = self.summary.summarize_results(results2)\n assert summary2 == expected_summary2, f\"Test case 2 failed: {summary2}\"\n\n def run_tests(self):\n \"\"\"\n Run all the tests.\n \"\"\"\n self.test_search_engine()\n self.test_index()\n self.test_ranking()\n self.test_summary()\n\nif __name__ == \"__main__\":\n testing = Testing()\n testing.run_tests()\n```\n\nIn the above code, I have implemented a `Testing` class that contains methods to test the functionality of the search engine, index, ranking, and summary modules. Each test case checks the expected output against the actual output and raises an assertion error if they do not match. The `run_tests` method runs all the test cases.", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Tasks\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n----- smart_search_engine/knowledge_base.py\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self):\n self.data = {}\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n\n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the given query.\n\n Args:\n query (str): The query to search for in the knowledge base.\n\n Returns:\n dict: The data matching the query.\n \"\"\"\n return self.data.get(query, {})\n\n----- smart_search_engine/index.py\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self):\n self.knowledge_base = KnowledgeBase()\n self.index = {}\n\n def create_index(self, data: dict):\n \"\"\"\n Create an index based on the given data.\n\n Args:\n data (dict): The data to be indexed.\n \"\"\"\n self.knowledge_base.update(data)\n self.index = {}\n\n for query, _ in data.items():\n results = self.knowledge_base.fetch_data(query)\n self.index[query] = results\n\n def query_index(self, query: str) -> list:\n \"\"\"\n Query the index based on the given query.\n\n Args:\n query (str): The query to search for in the index.\n\n Returns:\n list: The results matching the query.\n \"\"\"\n if query in self.index:\n return self.index[query]\n else:\n return []\n\n----- smart_search_engine/ranking.py\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): The search results to be ranked.\n\n Returns:\n list: The ranked search results.\n \"\"\"\n ranked_results = sorted(results, key=lambda x: x['relevance'], reverse=True)\n return ranked_results\n\n----- smart_search_engine/summary.py\n## smart_search_engine/summary.py\n\nclass Summary:\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results into a single string.\n\n Args:\n results (list): The search results to be summarized.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result['summary'] + \"\\n\"\n return summary.strip()\n\n----- smart_search_engine/search.py\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n \"\"\"\n Perform a search based on the given query.\n\n Args:\n query (str): The query to search for.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n\n----- smart_search_engine/main.py\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n \"\"\"\n Main function to perform a search based on user input.\n\n Returns:\n str: The summarized search results.\n \"\"\"\n query = input(\"Enter your search query: \")\n summary = self.search_engine.search(query)\n return summary\n\nif __name__ == \"__main__\":\n main = Main()\n summary = main.main()\n print(summary)\n\n----- smart_search_engine/interface.py\n## smart_search_engine/interface.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Interface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def run(self):\n \"\"\"\n Run the search engine interface.\n \"\"\"\n while True:\n query = input(\"Enter your search query (or 'exit' to quit): \")\n if query == \"exit\":\n break\n summary = self.search_engine.search(query)\n print(summary)\n\nif __name__ == \"__main__\":\n interface = Interface()\n interface.run()\n\n----- smart_search_engine/user_feedback.py\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n self.feedback = []\n\n def add_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for a search query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n feedback = {\n 'query': query,\n 'relevance': relevance\n }\n self.feedback.append(feedback)\n\n def get_feedback(self) -> list:\n \"\"\"\n Get the user feedback.\n\n Returns:\n list: The user feedback.\n \"\"\"\n return self.feedback\n\n----- smart_search_engine/security.py\n## smart_search_engine/security.py\n\nclass Security:\n def __init__(self):\n self.user_feedback = UserFeedback()\n\n def check_user_feedback(self, query: str) -> bool:\n \"\"\"\n Check if the user has provided feedback for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n bool: True if the user has provided feedback, False otherwise.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return True\n return False\n\n def get_user_feedback(self, query: str) -> int:\n \"\"\"\n Get the relevance score provided by the user for the given query.\n\n Args:\n query (str): The search query.\n\n Returns:\n int: The relevance score provided by the user, or None if no feedback is found.\n \"\"\"\n feedback = self.user_feedback.get_feedback()\n for item in feedback:\n if item['query'] == query:\n return item['relevance']\n return None\n\n def add_user_feedback(self, query: str, relevance: int):\n \"\"\"\n Add user feedback for the given query.\n\n Args:\n query (str): The search query.\n relevance (int): The relevance score provided by the user.\n \"\"\"\n self.user_feedback.add_feedback(query, relevance)\n\n----- smart_search_engine/testing.py\n## smart_search_engine/testing.py\n\nfrom smart_search_engine.search import SearchEngine\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Testing:\n def __init__(self):\n self.search_engine = SearchEngine()\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n self.knowledge_base = KnowledgeBase()\n\n def test_search_engine(self):\n \"\"\"\n Test the search engine functionality.\n \"\"\"\n # Test case 1: Search for a query that exists in the index\n query1 = \"apple\"\n expected_summary1 = \"Summary of search results for query: apple\"\n self.knowledge_base.update({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n self.index.create_index({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n summary1 = self.search_engine.search(query1)\n assert summary1 == expected_summary1, f\"Test case 1 failed: {summary1}\"\n\n # Test case 2: Search for a query that does not exist in the index\n query2 = \"banana\"\n expected_summary2 = \"No results found for query: banana\"\n self.knowledge_base.update({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n self.index.create_index({\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}})\n summary2 = self.search_engine.search(query2)\n assert summary2 == expected_summary2, f\"Test case 2 failed: {summary2}\"\n\n def test_index(self):\n \"\"\"\n Test the index functionality.\n \"\"\"\n # Test case 1: Create index with valid data\n data1 = {\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}}\n expected_index1 = {\"apple\": {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}}\n self.knowledge_base.update(data1)\n self.index.create_index(data1)\n index1 = self.index.index\n assert index1 == expected_index1, f\"Test case 1 failed: {index1}\"\n\n # Test case 2: Query index with valid query\n query2 = \"apple\"\n expected_results2 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}]\n results2 = self.index.query_index(query2)\n assert results2 == expected_results2, f\"Test case 2 failed: {results2}\"\n\n # Test case 3: Query index with invalid query\n query3 = \"banana\"\n expected_results3 = []\n results3 = self.index.query_index(query3)\n assert results3 == expected_results3, f\"Test case 3 failed: {results3}\"\n\n def test_ranking(self):\n \"\"\"\n Test the ranking functionality.\n \"\"\"\n # Test case 1: Rank results in descending order of relevance\n results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n expected_ranked_results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n ranked_results1 = self.ranking.rank_results(results1)\n assert ranked_results1 == expected_ranked_results1, f\"Test case 1 failed: {ranked_results1}\"\n\n # Test case 2: Rank results in ascending order of relevance\n results2 = [{\"relevance\": 0.6, \"summary\": \"This is a banana.\"}, {\"relevance\": 0.8, \"summary\": \"This is an apple.\"}]\n expected_ranked_results2 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n ranked_results2 = self.ranking.rank_results(results2)\n assert ranked_results2 == expected_ranked_results2, f\"Test case 2 failed: {ranked_results2}\"\n\n def test_summary(self):\n \"\"\"\n Test the summary functionality.\n \"\"\"\n # Test case 1: Summarize search results into a single string\n results1 = [{\"relevance\": 0.8, \"summary\": \"This is an apple.\"}, {\"relevance\": 0.6, \"summary\": \"This is a banana.\"}]\n expected_summary1 = \"This is an apple.\\nThis is a banana.\"\n summary1 = self.summary.summarize_results(results1)\n assert summary1 == expected_summary1, f\"Test case 1 failed: {summary1}\"\n\n # Test case 2: Summarize empty search results\n results2 = []\n expected_summary2 = \"\"\n summary2 = self.summary.summarize_results(results2)\n assert summary2 == expected_summary2, f\"Test case 2 failed: {summary2}\"\n\n def run_tests(self):\n \"\"\"\n Run all the tests.\n \"\"\"\n self.test_search_engine()\n self.test_index()\n self.test_ranking()\n self.test_summary()\n\nif __name__ == \"__main__\":\n testing = Testing()\n testing.run_tests()\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/monitoring.py\n```python\n## smart_search_engine/monitoring.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/monitoring.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/monitoring.py\n\nimport time\n\nclass Monitoring:\n def __init__(self):\n self.start_time = None\n self.end_time = None\n\n def start(self):\n \"\"\"\n Start the monitoring timer.\n \"\"\"\n self.start_time = time.time()\n\n def stop(self):\n \"\"\"\n Stop the monitoring timer.\n \"\"\"\n self.end_time = time.time()\n\n def get_elapsed_time(self) -> float:\n \"\"\"\n Get the elapsed time in seconds.\n\n Returns:\n float: The elapsed time in seconds.\n \"\"\"\n if self.start_time is None or self.end_time is None:\n raise Exception(\"Monitoring timer has not been started or stopped.\")\n return self.end_time - self.start_time\n```\n\n## Explanation:\nI have implemented the `Monitoring` class in the `smart_search_engine/monitoring.py` file. This class is responsible for starting and stopping a timer to monitor the elapsed time of a process. \n\nThe `Monitoring` class has the following methods:\n- `start()`: Starts the monitoring timer by setting the `start_time` attribute to the current time.\n- `stop()`: Stops the monitoring timer by setting the `end_time` attribute to the current time.\n- `get_elapsed_time()`: Calculates and returns the elapsed time in seconds by subtracting the `start_time` from the `end_time`. If the monitoring timer has not been started or stopped, an exception is raised.\n\nThis implementation follows the design specified in the `Data structures and interfaces` section.", + "\n## context\n\n### Project Name\n\n\n### Original Requirements\n['开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "{\n \"Language\": \"en_us\",\n \"Programming Language\": \"\",\n \"Original Requirements\": \"开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结\",\n \"Project Name\": \"search_engine\",\n \"Product Goals\": [\n \"Develop a search engine based on a large language model\",\n \"Incorporate a private knowledge base for enhanced search capabilities\",\n \"Implement search summarization using the large language model\"\n ],\n \"User Stories\": [\n \"As a user, I want to be able to search for information using natural language queries\",\n \"As a user, I want the search engine to provide relevant and accurate results\",\n \"As a user, I want the search engine to summarize the search results for easier understanding\"\n ],\n \"Competitive Analysis\": [],\n \"Competitive Quadrant Chart\": \"\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [],\n \"UI Design draft\": \"\",\n \"Anything UNCLEAR\": \"\"\n}", + "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.txt\n## Development Code\n```python\nprint('Hello, World')\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\n\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\ninstruction: There are no errors in the provided code.\n\nFile To Rewrite: N/A\n\nStatus: PASS\n\nSend To: NoOne\n---", + "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.sh\n## Development Code\n```python\necho 'Hello World'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\necho Hello World\n## Running Output\nstandard output: \n```text\nHello World\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "The running result indicates no error. The code works fine.\n\n## File To Rewrite:\nNo file needs to be rewritten.\n\n## Status:\nPASS\n\n## Send To:\nNoOne\n\n---", + "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.py\n## Development Code\n```python\npython -c \"print(1/0)\"\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\npython -c print(1/0)\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\nTraceback (most recent call last):\n File \"\", line 1, in \nZeroDivisionError: division by zero\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\nThe error is caused by the development code in file a.py, which attempts to divide by zero. To fix the error, the development code in a.py should be modified to handle the ZeroDivisionError, for example by using a try-except block.\n\nFile To Rewrite:\na.py\n\nStatus:\nFAIL\n\nSend To:\nEngineer\n---", + "\nNOTICE\n1. Role: You are a Development Engineer or QA engineer;\n2. Task: You received this message from another Development Engineer or QA engineer who ran or tested your code. \nBased on the message, first, figure out your own role, i.e. Engineer or QaEngineer,\nthen rewrite the development code or the test code based on your role, the error, and the summary, such that all bugs are fixed and the code performs well.\nAttention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script and triple quotes.\nThe message is as follows:\n# Legacy Code\n```python\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n if self.score > 21 and any(card.rank == 'A' for card in self.hand):\n self.score -= 10\n return self.score\n\n```\n---\n# Unit Test Code\n```python\n\nimport unittest\nfrom blackjack_game.player import Player\nfrom blackjack_game.deck import Deck\nfrom blackjack_game.card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n\n\n```\n---\n# Console logs\n```text\n\n```\n---\nNow you should start rewriting the code:\n## file name of the code to rewrite: Write code with triple quote. Do your best to implement THIS IN ONLY ONE FILE.\n": "```python\n## Development Engineer\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n ace_count = sum(1 for card in self.hand if card.rank == 'A')\n while self.score > 21 and ace_count > 0:\n self.score -= 10\n ace_count -= 1\n return self.score\n```\n\n```python\n## QaEngineer\n\nimport unittest\nfrom player import Player\nfrom deck import Deck\nfrom card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n```" +} \ No newline at end of file From 4b09ffd2f5339e9dfa9791903c261f1df2f20fba Mon Sep 17 00:00:00 2001 From: yzlin Date: Sat, 13 Jan 2024 10:37:15 +0800 Subject: [PATCH 312/637] pre-commit format --- metagpt/actions/debug_code.py | 1 - tests/metagpt/actions/test_debug_code.py | 15 +++++++++------ 2 files changed, 9 insertions(+), 7 deletions(-) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 74a188e9f..e5e0ac5d4 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -104,7 +104,6 @@ class DebugCode(BaseWriteAnalysisCode): logger.info(f"reflection is {resp}") return resp - async def run( self, context: List[Message] = None, diff --git a/tests/metagpt/actions/test_debug_code.py b/tests/metagpt/actions/test_debug_code.py index 675c07f78..262f2e60d 100644 --- a/tests/metagpt/actions/test_debug_code.py +++ b/tests/metagpt/actions/test_debug_code.py @@ -8,13 +8,13 @@ import pytest from metagpt.actions.debug_code import DebugCode, messages_to_str from metagpt.schema import Message -ErrorStr = '''Tested passed: +ErrorStr = """Tested passed: Tests failed: assert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5] -''' +""" -CODE = ''' +CODE = """ def sort_array(arr): # Helper function to count the number of ones in the binary representation def count_ones(n): @@ -27,7 +27,7 @@ def sort_array(arr): return sorted_arr ``` -''' +""" DebugContext = '''Solve the problem in Python: def sort_array(arr): @@ -42,13 +42,16 @@ def sort_array(arr): >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4] """ ''' + + @pytest.mark.asyncio async def test_debug_code(): debug_context = Message(content=DebugContext) new_code = await DebugCode().run(context=debug_context, code=CODE, runtime_result=ErrorStr) assert "def sort_array(arr)" in new_code - + + def test_messages_to_str(): debug_context = Message(content=DebugContext) msg_str = messages_to_str([debug_context]) - assert "user: Solve the problem in Python" in msg_str \ No newline at end of file + assert "user: Solve the problem in Python" in msg_str From f460771978c9997f8e654b14416d42b15cb6d315 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Sat, 13 Jan 2024 10:58:49 +0800 Subject: [PATCH 313/637] Update test cases --- tests/metagpt/test_incremental_dev.py | 81 +++++++++------------------ 1 file changed, 27 insertions(+), 54 deletions(-) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 223b0fb10..44478adc9 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -100,6 +100,33 @@ def test_refined_word_cloud(): raise e +def test_refined_gomoku(): + project_path = f"{DATA_PATH}/Gomoku" + check_or_create_base_tag(project_path) + + args = [ + "Add an AI opponent with fixed difficulty levels. Currently, the game only allows players to compete against themselves. Implement an AI algorithm that can playing with player. This will provide a more engaging and challenging experience for players.", + "--inc", + "--project-path", + project_path, + ] + result = runner.invoke(app, args) + logger.info(result) + logger.info(result.output) + if "Aborting" in result.output: + assert False + else: + tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if tag == "base": + assert False + else: + assert True + try: + subprocess.run(["git", "tag", "refine"], check=True) + except subprocess.CalledProcessError as e: + raise e + + def test_refined_dice_simulator_1(): project_path = f"{DATA_PATH}/dice_simulator_new" check_or_create_base_tag(project_path) @@ -154,33 +181,6 @@ def test_refined_dice_simulator_2(): raise e -def test_refined_dice_simulator_3(): - project_path = f"{DATA_PATH}/dice_simulator_new" - check_or_create_base_tag(project_path) - - args = [ - "Add functionality to set the number of sides on a die; Add functionality to view the history of scores; Add functionality to perform statistical analysis on all scores. The original dice rolling game could roll the dice multiple times and only display the current game result. But the new requirement add function that players to customize the number of sides of the dice and to view the history of scores and display the statistical analysis", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine_3"], check=True) - except subprocess.CalledProcessError as e: - raise e - - def test_refined_pygame_2048_1(): project_path = f"{DATA_PATH}/pygame_2048" check_or_create_base_tag(project_path) @@ -316,33 +316,6 @@ def test_refined_snake_game_2(): raise e -def test_refined_gomoku(): - project_path = f"{DATA_PATH}/Gomoku" - check_or_create_base_tag(project_path) - - args = [ - "Add an AI opponent with fixed difficulty levels. Currently, the game only allows players to compete against themselves. Implement an AI algorithm that can playing with player. This will provide a more engaging and challenging experience for players.", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine"], check=True) - except subprocess.CalledProcessError as e: - raise e - - def check_or_create_base_tag(project_path): # Change the current working directory to the specified project path os.chdir(project_path) From b612393151d933bd66c15362b59c1df2c5629139 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Sat, 13 Jan 2024 13:48:47 +0800 Subject: [PATCH 314/637] fixbug: pydantic validate --- metagpt/utils/file_repository.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/utils/file_repository.py b/metagpt/utils/file_repository.py index 85e7dc8a4..846e811cc 100644 --- a/metagpt/utils/file_repository.py +++ b/metagpt/utils/file_repository.py @@ -63,7 +63,7 @@ class FileRepository: await dependency_file.update(pathname, set(dependencies)) logger.info(f"update dependency: {str(pathname)}:{dependencies}") - return Document(root_path=str(self._relative_path), filename=filename, content=content) + return Document(root_path=str(self._relative_path), filename=str(filename), content=content) async def get_dependency(self, filename: Path | str) -> Set[str]: """Get the dependencies of a file. From 8d1bc25defbb953f6a238b022ba49f883ae0364e Mon Sep 17 00:00:00 2001 From: Arnaud Gelas Date: Sat, 13 Jan 2024 14:49:50 +0100 Subject: [PATCH 315/637] Constrain the language for the qa_engineer The qa_engineer was generating chinese texts and comments while the rest of the project was in English. --- metagpt/roles/qa_engineer.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/metagpt/roles/qa_engineer.py b/metagpt/roles/qa_engineer.py index b1d06d122..81082ef59 100644 --- a/metagpt/roles/qa_engineer.py +++ b/metagpt/roles/qa_engineer.py @@ -36,7 +36,8 @@ class QaEngineer(Role): profile: str = "QaEngineer" goal: str = "Write comprehensive and robust tests to ensure codes will work as expected without bugs" constraints: str = ( - "The test code you write should conform to code standard like PEP8, be modular, " "easy to read and maintain" + "The test code you write should conform to code standard like PEP8, be modular, easy to read and maintain." + "Use same language as user requirement" ) test_round_allowed: int = 5 test_round: int = 0 From 1238c484511e4a54691fe2816b43833eac41399d Mon Sep 17 00:00:00 2001 From: Arnaud Gelas Date: Sat, 13 Jan 2024 15:14:38 +0100 Subject: [PATCH 316/637] Fix: requirements.txt was not written to the disk While the Python packages requirements are correctly detected and saved into the json task file, requirements.txt was always empty. Since it was trying to get packages with the wrong key, packages were always empty when writing in requirements.txt --- metagpt/actions/project_management.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/project_management.py b/metagpt/actions/project_management.py index e40c2034b..ea5e4016e 100644 --- a/metagpt/actions/project_management.py +++ b/metagpt/actions/project_management.py @@ -100,7 +100,7 @@ class WriteTasks(Action): @staticmethod async def _update_requirements(doc): m = json.loads(doc.content) - packages = set(m.get("Required Python third-party packages", set())) + packages = set(m.get("Required Python packages", set())) file_repo = CONFIG.git_repo.new_file_repository() requirement_doc = await file_repo.get(filename=PACKAGE_REQUIREMENTS_FILENAME) if not requirement_doc: From d34073801397546d7669840705c2acca31383384 Mon Sep 17 00:00:00 2001 From: Arnaud Gelas Date: Sat, 13 Jan 2024 15:53:01 +0100 Subject: [PATCH 317/637] Even if you set an investment, the default investment shows in the log --- metagpt/team.py | 1 + 1 file changed, 1 insertion(+) diff --git a/metagpt/team.py b/metagpt/team.py index b98fc2efb..8fd6760f5 100644 --- a/metagpt/team.py +++ b/metagpt/team.py @@ -83,6 +83,7 @@ class Team(BaseModel): """Invest company. raise NoMoneyException when exceed max_budget.""" self.investment = investment CONFIG.max_budget = investment + CONFIG.cost_manager.max_budget = investment logger.info(f"Investment: ${investment}.") @staticmethod From 34b3de1f863e58391009b6180d266872c3c6ac66 Mon Sep 17 00:00:00 2001 From: Arnaud Gelas Date: Sat, 13 Jan 2024 16:10:46 +0100 Subject: [PATCH 318/637] When setting the max budget, I don't want to overcome this limit --- metagpt/team.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/team.py b/metagpt/team.py index 8fd6760f5..aad24efa0 100644 --- a/metagpt/team.py +++ b/metagpt/team.py @@ -88,7 +88,7 @@ class Team(BaseModel): @staticmethod def _check_balance(): - if CONFIG.cost_manager.total_cost > CONFIG.cost_manager.max_budget: + if CONFIG.cost_manager.total_cost >= CONFIG.cost_manager.max_budget: raise NoMoneyException( CONFIG.cost_manager.total_cost, f"Insufficient funds: {CONFIG.cost_manager.max_budget}" ) From 932a26cbb3be1e12a52b6d4e1656b7c6edb0e35e Mon Sep 17 00:00:00 2001 From: lidanyang Date: Mon, 15 Jan 2024 10:50:08 +0800 Subject: [PATCH 319/637] update unittest --- tests/metagpt/actions/test_execute_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/metagpt/actions/test_execute_code.py b/tests/metagpt/actions/test_execute_code.py index 8340272e4..904cc3c58 100644 --- a/tests/metagpt/actions/test_execute_code.py +++ b/tests/metagpt/actions/test_execute_code.py @@ -96,4 +96,4 @@ async def test_run_with_timeout(): code = "import time; time.sleep(2)" message, success = await pi.run(code) assert not success - assert message == "TimeoutError" + assert message.startswith("Cell execution timed out") From f45a368be2cf9860c2046656767b6c4f1bc0f53a Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 15 Jan 2024 11:13:35 +0800 Subject: [PATCH 320/637] 1. add vision config in config.yaml 2. add imitate_webpage.py in example 3. update vision.py --- config/config.yaml | 14 +++++++ examples/imitate_webpage.py | 25 +++++++++++++ metagpt/tools/functions/libs/vision.py | 51 +++++++++++++------------- 3 files changed, 65 insertions(+), 25 deletions(-) create mode 100644 examples/imitate_webpage.py diff --git a/config/config.yaml b/config/config.yaml index 79ebae863..5eab964bd 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -86,6 +86,20 @@ TIMEOUT: 60 # Timeout for llm invocation #AZURE_TTS_SUBSCRIPTION_KEY: "YOUR_API_KEY" #AZURE_TTS_REGION: "eastus" +#### for OPENAI VISION + +OPENAI_VISION_URL: "https://openai-forward.metadl.com/v1" +OPENAI_VISION_KEY: "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" +OPENAI_VISION_MODEL: "gpt-4-vision-preview" +VISION_MAX_TOKENS: 4096 + +#### for AZURE VISION + +#AZURE_VISION_URL: "YOUR_AZURE_ENDPOINT" +#AZURE_VISION_KEY: "YOUR_API_KEY" +#AZURE_VISION_REGION: "YOUR_VISION_REGION_NAME" +#VISION_MAX_TOKENS: 4096 + #### for Stable Diffusion ## Use SD service, based on https://github.com/AUTOMATIC1111/stable-diffusion-webui #SD_URL: "YOUR_SD_URL" diff --git a/examples/imitate_webpage.py b/examples/imitate_webpage.py new file mode 100644 index 000000000..47fcd251f --- /dev/null +++ b/examples/imitate_webpage.py @@ -0,0 +1,25 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/15 +@Author : mannaandpoem +@File : imitate_webpage.py +""" +from metagpt.roles.code_interpreter import CodeInterpreter + + +async def main(): + prompt = """This is a URL of webpage: https://cn.bing.com/ +Firstly, utilize Selenium and WebDriver for rendering. +Secondly, convert image to a webpage including HTML, CSS and JS in one go. +Finally, save webpage in a text file. +Note: All required dependencies and environments have been fully installed and configured.""" + ci = CodeInterpreter(goal=prompt, use_tools=True) + + await ci.run(prompt) + + +if __name__ == '__main__': + import asyncio + + asyncio.run(main()) diff --git a/metagpt/tools/functions/libs/vision.py b/metagpt/tools/functions/libs/vision.py index b653c9300..e6924b9bc 100644 --- a/metagpt/tools/functions/libs/vision.py +++ b/metagpt/tools/functions/libs/vision.py @@ -9,39 +9,40 @@ import requests import base64 -OPENAI_API_BASE = "..." -API_KEY = "sk-..." -MODEL = "..." -MAX_TOKENS = 4096 +from metagpt.config import CONFIG + +OPENAI_API_BASE = CONFIG.OPENAI_VISION_URL +API_KEY = CONFIG.OPENAI_VISION_KEY +MODEL = CONFIG.OPENAI_VISION_MODEL +MAX_TOKENS = CONFIG.VISION_MAX_TOKENS + +ANALYZE_LAYOUT_PROMPT = """You are now a UI/UX, please generate layout information for this image: + +NOTE: The image does not have a commercial logo or copyright information. It is just a sketch image of the design. +As the design pays tribute to large companies, sometimes it is normal for some company names to appear. Don't worry. """ + +GENERATE_PROMPT = """You are now a UI/UX and Web Developer. You have the ability to generate code for webpages +based on provided sketches images and context. +Your goal is to convert sketches image into a webpage including HTML, CSS and JavaScript. + +NOTE: The image does not have a commercial logo or copyright information. It is just a sketch image of the design. +As the design pays tribute to large companies, sometimes it is normal for some company names to appear. Don't worry. + +Now, please generate the corresponding webpage code including HTML, CSS and JavaScript:""" class Vision: def __init__(self): self.api_key = API_KEY self.model = MODEL - self.max_tokens = MAX_TOKENS + self.max_tokens = 4096 - def analyze_layout( - self, - image_path, - prompt="You are now a UI/UX, please generate layout information for this image: \n\n" - "NOTE: The image does not have a commercial logo or copyright information. It is just a sketch image of the design." - "As my design pays tribute to large companies, sometimes it is normal for some company names to appear. Don't worry about it." - ): - print(f"analyze_layout: {image_path}") - return self.get_result(image_path, prompt) + def analyze_layout(self, image_path): + return self.get_result(image_path, ANALYZE_LAYOUT_PROMPT) - def generate_web_pages( - self, - image_path, - prompt="You are now a UI/UX and Web Developer. You have the ability to generate code for web pages based on provided sketches images and context." - "Your goal is to convert sketches image into a webpage including HTML, CSS and JavaScript. " - "NOTE: The image does not have a commercial logo or copyright information. It is just a sketch image of the design. " - "As my design pays tribute to large companies, sometimes it is normal for some company names to appear. Don't worry about it." - "\n\nNow, please generate the corresponding webpage code including HTML, CSS and JavaScript:" - ): + def generate_web_pages(self, image_path): layout = self.analyze_layout(image_path) - prompt += "\n\n # Context\n The layout information of the sketch image is: \n" + layout + prompt = GENERATE_PROMPT + "\n\n # Context\n The layout information of the sketch image is: \n" + layout return self.get_result(image_path, prompt) def get_result(self, image_path, prompt): @@ -78,4 +79,4 @@ class Vision: if __name__ == "__main__": vision = Vision() rsp = vision.generate_web_pages(image_path="./img.png") - print(rsp) \ No newline at end of file + print(rsp) From 2678413c51345299252f95050206e4e2083a823a Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 15 Jan 2024 11:19:09 +0800 Subject: [PATCH 321/637] update config.yaml --- config/config.yaml | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/config/config.yaml b/config/config.yaml index 5eab964bd..412da8b15 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -88,16 +88,16 @@ TIMEOUT: 60 # Timeout for llm invocation #### for OPENAI VISION -OPENAI_VISION_URL: "https://openai-forward.metadl.com/v1" -OPENAI_VISION_KEY: "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" -OPENAI_VISION_MODEL: "gpt-4-vision-preview" -VISION_MAX_TOKENS: 4096 +#OPENAI_VISION_URL: "YOUR_OPENAI_ENDPOINT" +#OPENAI_VISION_KEY: "YOUR_API_KEY" +#OPENAI_VISION_MODEL: "YOUR_VISION_MODEL_NAME" +#VISION_MAX_TOKENS: 4096 #### for AZURE VISION #AZURE_VISION_URL: "YOUR_AZURE_ENDPOINT" #AZURE_VISION_KEY: "YOUR_API_KEY" -#AZURE_VISION_REGION: "YOUR_VISION_REGION_NAME" +#AZURE_VISION_REGION: "YOUR_VISION_MODEL_NAME" #VISION_MAX_TOKENS: 4096 #### for Stable Diffusion From 38929dc1248140bfd6246238f5ab946af7aa483d Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 15 Jan 2024 11:47:36 +0800 Subject: [PATCH 322/637] update imitate_webpage.py --- examples/imitate_webpage.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/examples/imitate_webpage.py b/examples/imitate_webpage.py index b4610d5e0..da46af0a6 100644 --- a/examples/imitate_webpage.py +++ b/examples/imitate_webpage.py @@ -9,7 +9,8 @@ from metagpt.roles.code_interpreter import CodeInterpreter async def main(): - prompt = """This is a URL of webpage: 'https://www.baidu.com/' . + web_url = 'https://www.baidu.com/' + prompt = f"""This is a URL of webpage: '{web_url}' . Firstly, utilize Selenium and WebDriver for rendering. Secondly, convert image to a webpage including HTML, CSS and JS in one go. Finally, save webpage in a text file. From 9eee30bf65d1bccc5226a7e5abae033a0e9acd51 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 15 Jan 2024 12:57:36 +0800 Subject: [PATCH 323/637] update config.yaml and vision.py for configuration of vision --- config/config.yaml | 9 --------- metagpt/tools/functions/libs/vision.py | 7 ++++--- 2 files changed, 4 insertions(+), 12 deletions(-) diff --git a/config/config.yaml b/config/config.yaml index 412da8b15..d8fab693e 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -88,18 +88,9 @@ TIMEOUT: 60 # Timeout for llm invocation #### for OPENAI VISION -#OPENAI_VISION_URL: "YOUR_OPENAI_ENDPOINT" -#OPENAI_VISION_KEY: "YOUR_API_KEY" #OPENAI_VISION_MODEL: "YOUR_VISION_MODEL_NAME" #VISION_MAX_TOKENS: 4096 -#### for AZURE VISION - -#AZURE_VISION_URL: "YOUR_AZURE_ENDPOINT" -#AZURE_VISION_KEY: "YOUR_API_KEY" -#AZURE_VISION_REGION: "YOUR_VISION_MODEL_NAME" -#VISION_MAX_TOKENS: 4096 - #### for Stable Diffusion ## Use SD service, based on https://github.com/AUTOMATIC1111/stable-diffusion-webui #SD_URL: "YOUR_SD_URL" diff --git a/metagpt/tools/functions/libs/vision.py b/metagpt/tools/functions/libs/vision.py index e6924b9bc..8c29b0567 100644 --- a/metagpt/tools/functions/libs/vision.py +++ b/metagpt/tools/functions/libs/vision.py @@ -11,8 +11,8 @@ import base64 from metagpt.config import CONFIG -OPENAI_API_BASE = CONFIG.OPENAI_VISION_URL -API_KEY = CONFIG.OPENAI_VISION_KEY +OPENAI_API_BASE = CONFIG.OPENAI_BASE_URL +API_KEY = CONFIG.OPENAI_API_KEY MODEL = CONFIG.OPENAI_VISION_MODEL MAX_TOKENS = CONFIG.VISION_MAX_TOKENS @@ -77,6 +77,7 @@ class Vision: if __name__ == "__main__": + image_path = "image.png" vision = Vision() - rsp = vision.generate_web_pages(image_path="./img.png") + rsp = vision.generate_web_pages(image_path=image_path) print(rsp) From 841f69d5edc063ab2d9bf340654dd63ba12465db Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 15 Jan 2024 12:57:36 +0800 Subject: [PATCH 324/637] update config.yaml and vision.py for configuration of vision --- config/config.yaml | 9 --------- examples/imitate_webpage.py | 2 +- metagpt/tools/functions/libs/vision.py | 7 ++++--- 3 files changed, 5 insertions(+), 13 deletions(-) diff --git a/config/config.yaml b/config/config.yaml index 412da8b15..d8fab693e 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -88,18 +88,9 @@ TIMEOUT: 60 # Timeout for llm invocation #### for OPENAI VISION -#OPENAI_VISION_URL: "YOUR_OPENAI_ENDPOINT" -#OPENAI_VISION_KEY: "YOUR_API_KEY" #OPENAI_VISION_MODEL: "YOUR_VISION_MODEL_NAME" #VISION_MAX_TOKENS: 4096 -#### for AZURE VISION - -#AZURE_VISION_URL: "YOUR_AZURE_ENDPOINT" -#AZURE_VISION_KEY: "YOUR_API_KEY" -#AZURE_VISION_REGION: "YOUR_VISION_MODEL_NAME" -#VISION_MAX_TOKENS: 4096 - #### for Stable Diffusion ## Use SD service, based on https://github.com/AUTOMATIC1111/stable-diffusion-webui #SD_URL: "YOUR_SD_URL" diff --git a/examples/imitate_webpage.py b/examples/imitate_webpage.py index da46af0a6..6c12c7eda 100644 --- a/examples/imitate_webpage.py +++ b/examples/imitate_webpage.py @@ -9,7 +9,7 @@ from metagpt.roles.code_interpreter import CodeInterpreter async def main(): - web_url = 'https://www.baidu.com/' + web_url = 'https://pytorch.org/' prompt = f"""This is a URL of webpage: '{web_url}' . Firstly, utilize Selenium and WebDriver for rendering. Secondly, convert image to a webpage including HTML, CSS and JS in one go. diff --git a/metagpt/tools/functions/libs/vision.py b/metagpt/tools/functions/libs/vision.py index e6924b9bc..8c29b0567 100644 --- a/metagpt/tools/functions/libs/vision.py +++ b/metagpt/tools/functions/libs/vision.py @@ -11,8 +11,8 @@ import base64 from metagpt.config import CONFIG -OPENAI_API_BASE = CONFIG.OPENAI_VISION_URL -API_KEY = CONFIG.OPENAI_VISION_KEY +OPENAI_API_BASE = CONFIG.OPENAI_BASE_URL +API_KEY = CONFIG.OPENAI_API_KEY MODEL = CONFIG.OPENAI_VISION_MODEL MAX_TOKENS = CONFIG.VISION_MAX_TOKENS @@ -77,6 +77,7 @@ class Vision: if __name__ == "__main__": + image_path = "image.png" vision = Vision() - rsp = vision.generate_web_pages(image_path="./img.png") + rsp = vision.generate_web_pages(image_path=image_path) print(rsp) From cc92d8fb4afdbed517cf6616373b4a3100cf0ed5 Mon Sep 17 00:00:00 2001 From: zhanglei Date: Mon, 15 Jan 2024 14:11:41 +0800 Subject: [PATCH 325/637] add:openai text to speech --- metagpt/provider/openai_api.py | 4 ++++ tests/metagpt/provider/test_openai.py | 12 ++++++++++++ 2 files changed, 16 insertions(+) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 3f3a4e1a7..3a9aca870 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -235,3 +235,7 @@ class OpenAILLM(BaseLLM): async def amoderation(self, content: Union[str, list[str]]): """Moderate content.""" return await self.aclient.moderations.create(input=content) + + async def atext_to_speech(self, **kwargs): + """text to speech""" + return await self.aclient.audio.speech.create(**kwargs) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index ca9e918da..cb0d0d636 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -42,6 +42,18 @@ async def test_aask_code_message(): assert len(rsp["code"]) > 0 +@pytest.mark.asyncio +async def test_text_to_speech(): + llm = LLM() + resp = await llm.atext_to_speech( + model="tts-1", + voice="alloy", + input="人生说起来长,但知道一个岁月回头看,许多事件仅是仓促的。一段一段拼凑一起,合成了人生。苦难当头时,当下不免觉得是折磨;回头看,也不够是一段短短的人生旅程。", + ) + assert 200 == resp.response.status_code + + + class TestOpenAI: def test_make_client_kwargs_without_proxy(self): instance = OpenAILLM(mock_llm_config) From ca63880753f7d6ea560c36ee24e17d721a9a81ba Mon Sep 17 00:00:00 2001 From: zhanglei Date: Mon, 15 Jan 2024 14:18:35 +0800 Subject: [PATCH 326/637] add: openai's text to speech --- tests/metagpt/provider/test_openai.py | 1 - 1 file changed, 1 deletion(-) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index cb0d0d636..bf19a77b8 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -53,7 +53,6 @@ async def test_text_to_speech(): assert 200 == resp.response.status_code - class TestOpenAI: def test_make_client_kwargs_without_proxy(self): instance = OpenAILLM(mock_llm_config) From 4ceff0ec29051033e00a55a2c984d1616c0314f5 Mon Sep 17 00:00:00 2001 From: better629 Date: Mon, 15 Jan 2024 14:48:31 +0800 Subject: [PATCH 327/637] add prompt_schema --- metagpt/actions/action_node.py | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/action_node.py b/metagpt/actions/action_node.py index b511f2662..4f61af4ed 100644 --- a/metagpt/actions/action_node.py +++ b/metagpt/actions/action_node.py @@ -86,7 +86,7 @@ Compare the key's value of nodes_output and the corresponding requirements one b {constraint} ## action -Follow format example's json format, generate output and make sure it follows the format example. +Follow format example's {prompt_schema} format, generate output and make sure it follows the format example. """ REVISE_TEMPLATE = """ @@ -108,7 +108,7 @@ change the nodes_output key's value to meet its comment and no need to add extra {constraint} ## action -Follow format example's json format, generate output and make sure it follows the format example. +Follow format example's {prompt_schema} format, generate output and make sure it follows the format example. """ @@ -469,7 +469,8 @@ class ActionNode: return dict() prompt = template.format( - nodes_output=json.dumps(nodes_output, ensure_ascii=False), tag=TAG, constraint=FORMAT_CONSTRAINT + nodes_output=json.dumps(nodes_output, ensure_ascii=False), tag=TAG, constraint=FORMAT_CONSTRAINT, + prompt_schema="json" ) content = await self.llm.aask(prompt) @@ -567,6 +568,7 @@ class ActionNode: example=example, instruction=instruction, constraint=FORMAT_CONSTRAINT, + prompt_schema="json" ) # step2, use `_aask_v1` to get revise structure result From c92799119405babd15b63624086f7783c462ee5c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 15:41:18 +0800 Subject: [PATCH 328/637] update get_choice_function_arguments. --- metagpt/provider/base_llm.py | 48 +++++++++++++++++++++++++++++++++++- 1 file changed, 47 insertions(+), 1 deletion(-) diff --git a/metagpt/provider/base_llm.py b/metagpt/provider/base_llm.py index dbef15fa1..c482aaf35 100644 --- a/metagpt/provider/base_llm.py +++ b/metagpt/provider/base_llm.py @@ -6,10 +6,14 @@ @File : base_llm.py @Desc : mashenquan, 2023/8/22. + try catch """ +import re import json from abc import ABC, abstractmethod from typing import Optional +from metagpt.logs import logger +from metagpt.utils.common import CodeParser + class BaseLLM(ABC): """LLM API abstract class, requiring all inheritors to provide a series of standard capabilities""" @@ -118,6 +122,30 @@ class BaseLLM(ABC): """ return rsp.get("choices")[0]["message"]["tool_calls"][0]["function"] + def _parse_arguments(self, arguments: str) -> dict: + """parse arguments in openai function call""" + if 'langugae' not in arguments and 'code' not in arguments: + logger.warning(f"Not found `code`, `language`, We assume it is pure code:\n {arguments}\n. ") + return {'language': 'python', 'code': arguments} + + # 匹配language + language_pattern = re.compile(r'[\"\']?language[\"\']?\s*:\s*["\']([^"\']+?)["\']', re.DOTALL) + language_match = language_pattern.search(arguments) + language_value = language_match.group(1) if language_match else None + + # 匹配code + code_pattern = r'(["\']{3}|["])([\s\S]*?)\1' + try: + code_value = re.findall(code_pattern, arguments)[-1][-1] + except Exception as e: + logger.error(f"{e}, when re.findall({code_pattern}, {arguments})") + code_value = None + + if code_value is None: + raise ValueError(f"Parse code error for {arguments}") + # arguments只有code的情况 + return {'language': language_value, 'code': code_value} + def get_choice_function_arguments(self, rsp: dict) -> dict: """Required to provide the first function arguments of choice. @@ -125,7 +153,25 @@ class BaseLLM(ABC): :return dict: return the first function arguments of choice, for example, {'language': 'python', 'code': "print('Hello, World!')"} """ - return json.loads(self.get_choice_function(rsp)["arguments"], strict=False) + try: + arguments: str = self.get_choice_function(rsp)["arguments"] + return json.loads(arguments, strict=False) + except json.decoder.JSONDecodeError as e: + logger.debug(f"Got JSONDecodeError for {arguments}, we will use RegExp to parse code, \n {e}") + return self._parse_arguments(arguments) + except KeyError as e: + if 'tool_calls' in e.args: + txt_rsp = self.get_choice_text(rsp) + # find code + code = CodeParser.parse_code(None, txt_rsp, lang='python') + if code != txt_rsp: + return {'language': 'python', 'code': code} + # no code + return {'language': 'markdown', 'code': txt_rsp} + raise e + except Exception as e: + logger.error(f"Got error `{e}` for parsing\n {rsp}\n") + return {} def messages_to_prompt(self, messages: list[dict]): """[{"role": "user", "content": msg}] to user: etc.""" From bb356fbc02a666d970799798a8f7d921252ec703 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 15:49:07 +0800 Subject: [PATCH 329/637] update truncate. --- metagpt/actions/execute_code.py | 34 +++++++++++++++++++++++---------- 1 file changed, 24 insertions(+), 10 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index c75711e75..458dc0898 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -212,26 +212,40 @@ class ExecutePyCode(ExecuteCode, Action): cell_index = len(self.nb.cells) - 1 success, error_message = await self.run_cell(self.nb.cells[-1], cell_index) - if success: - outputs = self.parse_outputs(self.nb.cells[-1].outputs) - return truncate(remove_escape_and_color_codes(outputs)), True - else: - return error_message, False + if not success: + return truncate(remove_escape_and_color_codes(error_message), is_success=success) + + # code success + outputs = self.parse_outputs(self.nb.cells[-1].outputs) + return truncate(remove_escape_and_color_codes(outputs), is_success=success) else: # TODO: markdown raise NotImplementedError(f"Not support this code type : {language}, Only support code!") -def truncate(result: str, keep_len: int = 2000) -> str: - desc = f"Truncated to show only the last {keep_len} characters\n" +def truncate(result: str, keep_len: int = 2000, is_success: bool = True) -> str | bool: + desc = f"Executed code {'successfully' if is_success else 'failed, please reflect the cause of bug and then debug'}" + if is_success: + desc += f"Truncated to show only {keep_len} characters\n" + else: + desc += "Show complete information for you." + if result.startswith(desc): result = result[len(desc) :] if len(result) > keep_len: - result = result[-keep_len:] - return desc + result + result = result[-keep_len:] if not is_success else result + if not result: + result = 'No output about your code. Only when importing packages it is normal case. Recap and go ahead.' + return result, False - return result + if result.strip().startswith(" Date: Mon, 15 Jan 2024 15:51:36 +0800 Subject: [PATCH 330/637] support for markdown. --- metagpt/actions/execute_code.py | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 458dc0898..1a97e49d6 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -15,7 +15,7 @@ import nbformat from nbclient import NotebookClient from nbclient.exceptions import CellTimeoutError, DeadKernelError from nbformat import NotebookNode -from nbformat.v4 import new_code_cell, new_output +from nbformat.v4 import new_code_cell, new_output, new_markdown_cell from rich.console import Console from rich.syntax import Syntax @@ -91,6 +91,9 @@ class ExecutePyCode(ExecuteCode, Action): def add_code_cell(self, code): self.nb.cells.append(new_code_cell(source=code)) + def add_markdown_cell(self, markdown): + self.nb.cells.append(new_markdown_cell(source=markdown)) + def _display(self, code, language: str = "python"): if language == "python": code = Syntax(code, "python", theme="paraiso-dark", line_numbers=True) @@ -219,8 +222,9 @@ class ExecutePyCode(ExecuteCode, Action): outputs = self.parse_outputs(self.nb.cells[-1].outputs) return truncate(remove_escape_and_color_codes(outputs), is_success=success) else: - # TODO: markdown - raise NotImplementedError(f"Not support this code type : {language}, Only support code!") + # markdown + self.add_markdown_cell(code) + return code, True def truncate(result: str, keep_len: int = 2000, is_success: bool = True) -> str | bool: From 8baa6d094f0316f8ac1abb7d0dd3f89a435dbdaf Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 15 Jan 2024 16:37:42 +0800 Subject: [PATCH 331/637] refine writeprd code --- metagpt/actions/action.py | 4 +- metagpt/actions/debug_error.py | 6 +- metagpt/actions/design_api.py | 18 +-- metagpt/actions/prepare_documents.py | 4 +- metagpt/actions/project_management.py | 20 ++-- metagpt/actions/summarize_code.py | 6 +- metagpt/actions/write_code.py | 8 +- metagpt/actions/write_code_review.py | 2 +- metagpt/actions/write_prd.py | 154 +++++++++++++------------- metagpt/context.py | 3 + metagpt/roles/searcher.py | 3 +- metagpt/schema.py | 22 +++- metagpt/utils/project_repo.py | 14 +++ 13 files changed, 150 insertions(+), 114 deletions(-) diff --git a/metagpt/actions/action.py b/metagpt/actions/action.py index a33918a09..a7eb838b0 100644 --- a/metagpt/actions/action.py +++ b/metagpt/actions/action.py @@ -34,8 +34,8 @@ class Action(SerializationMixin, ContextMixin, BaseModel): node: ActionNode = Field(default=None, exclude=True) @property - def project_repo(self): - return ProjectRepo(self.context.git_repo) + def repo(self) -> ProjectRepo: + return self.context.repo @property def prompt_schema(self): diff --git a/metagpt/actions/debug_error.py b/metagpt/actions/debug_error.py index f491fdd55..5ed31bed8 100644 --- a/metagpt/actions/debug_error.py +++ b/metagpt/actions/debug_error.py @@ -49,7 +49,7 @@ class DebugError(Action): i_context: RunCodeContext = Field(default_factory=RunCodeContext) async def run(self, *args, **kwargs) -> str: - output_doc = await self.project_repo.test_outputs.get(filename=self.i_context.output_filename) + output_doc = await self.repo.test_outputs.get(filename=self.i_context.output_filename) if not output_doc: return "" output_detail = RunCodeResult.loads(output_doc.content) @@ -59,12 +59,12 @@ class DebugError(Action): return "" logger.info(f"Debug and rewrite {self.i_context.test_filename}") - code_doc = await self.project_repo.with_src_path(self.context.src_workspace).srcs.get( + code_doc = await self.repo.with_src_path(self.context.src_workspace).srcs.get( filename=self.i_context.code_filename ) if not code_doc: return "" - test_doc = await self.project_repo.tests.get(filename=self.i_context.test_filename) + test_doc = await self.repo.tests.get(filename=self.i_context.test_filename) if not test_doc: return "" prompt = PROMPT_TEMPLATE.format(code=code_doc.content, test_code=test_doc.content, logs=output_detail.stderr) diff --git a/metagpt/actions/design_api.py b/metagpt/actions/design_api.py index 04c580226..c6f608b7e 100644 --- a/metagpt/actions/design_api.py +++ b/metagpt/actions/design_api.py @@ -40,10 +40,10 @@ class WriteDesign(Action): async def run(self, with_messages: Message, schema: str = None): # Use `git status` to identify which PRD documents have been modified in the `docs/prds` directory. - changed_prds = self.project_repo.docs.prd.changed_files + changed_prds = self.repo.docs.prd.changed_files # Use `git status` to identify which design documents in the `docs/system_designs` directory have undergone # changes. - changed_system_designs = self.project_repo.docs.system_design.changed_files + changed_system_designs = self.repo.docs.system_design.changed_files # For those PRDs and design documents that have undergone changes, regenerate the design content. changed_files = Documents() @@ -73,21 +73,21 @@ class WriteDesign(Action): return system_design_doc async def _update_system_design(self, filename) -> Document: - prd = await self.project_repo.docs.prd.get(filename) - old_system_design_doc = await self.project_repo.docs.system_design.get(filename) + prd = await self.repo.docs.prd.get(filename) + old_system_design_doc = await self.repo.docs.system_design.get(filename) if not old_system_design_doc: system_design = await self._new_system_design(context=prd.content) - doc = await self.project_repo.docs.system_design.save( + doc = await self.repo.docs.system_design.save( filename=filename, content=system_design.instruct_content.model_dump_json(), dependencies={prd.root_relative_path}, ) else: doc = await self._merge(prd_doc=prd, system_design_doc=old_system_design_doc) - await self.project_repo.docs.system_design.save_doc(doc=doc, dependencies={prd.root_relative_path}) + await self.repo.docs.system_design.save_doc(doc=doc, dependencies={prd.root_relative_path}) await self._save_data_api_design(doc) await self._save_seq_flow(doc) - await self.project_repo.resources.system_design.save_pdf(doc=doc) + await self.repo.resources.system_design.save_pdf(doc=doc) return doc async def _save_data_api_design(self, design_doc): @@ -95,7 +95,7 @@ class WriteDesign(Action): data_api_design = m.get("Data structures and interfaces") if not data_api_design: return - pathname = self.project_repo.workdir / DATA_API_DESIGN_FILE_REPO / Path(design_doc.filename).with_suffix("") + pathname = self.repo.workdir / DATA_API_DESIGN_FILE_REPO / Path(design_doc.filename).with_suffix("") await self._save_mermaid_file(data_api_design, pathname) logger.info(f"Save class view to {str(pathname)}") @@ -104,7 +104,7 @@ class WriteDesign(Action): seq_flow = m.get("Program call flow") if not seq_flow: return - pathname = self.project_repo.workdir / Path(SEQ_FLOW_FILE_REPO) / Path(design_doc.filename).with_suffix("") + pathname = self.repo.workdir / Path(SEQ_FLOW_FILE_REPO) / Path(design_doc.filename).with_suffix("") await self._save_mermaid_file(seq_flow, pathname) logger.info(f"Saving sequence flow to {str(pathname)}") diff --git a/metagpt/actions/prepare_documents.py b/metagpt/actions/prepare_documents.py index 56c587cb3..84a4fc1d7 100644 --- a/metagpt/actions/prepare_documents.py +++ b/metagpt/actions/prepare_documents.py @@ -15,6 +15,7 @@ from metagpt.actions import Action, ActionOutput from metagpt.const import REQUIREMENT_FILENAME from metagpt.utils.file_repository import FileRepository from metagpt.utils.git_repository import GitRepository +from metagpt.utils.project_repo import ProjectRepo class PrepareDocuments(Action): @@ -38,13 +39,14 @@ class PrepareDocuments(Action): shutil.rmtree(path) self.config.project_path = path self.context.git_repo = GitRepository(local_path=path, auto_init=True) + self.context.repo = ProjectRepo(self.context.git_repo) async def run(self, with_messages, **kwargs): """Create and initialize the workspace folder, initialize the Git environment.""" self._init_repo() # Write the newly added requirements from the main parameter idea to `docs/requirement.txt`. - doc = await self.project_repo.docs.save(filename=REQUIREMENT_FILENAME, content=with_messages[0].content) + doc = await self.repo.docs.save(filename=REQUIREMENT_FILENAME, content=with_messages[0].content) # Send a Message notification to the WritePRD action, instructing it to process requirements using # `docs/requirement.txt` and `docs/prds/`. return ActionOutput(content=doc.content, instruct_content=doc) diff --git a/metagpt/actions/project_management.py b/metagpt/actions/project_management.py index 9ada629be..fb086d5c2 100644 --- a/metagpt/actions/project_management.py +++ b/metagpt/actions/project_management.py @@ -13,8 +13,8 @@ import json from typing import Optional -from metagpt.actions import ActionOutput from metagpt.actions.action import Action +from metagpt.actions.action_output import ActionOutput from metagpt.actions.project_management_an import PM_NODE from metagpt.const import PACKAGE_REQUIREMENTS_FILENAME from metagpt.logs import logger @@ -34,8 +34,8 @@ class WriteTasks(Action): i_context: Optional[str] = None async def run(self, with_messages): - changed_system_designs = self.project_repo.docs.system_design.changed_files - changed_tasks = self.project_repo.docs.task.changed_files + changed_system_designs = self.repo.docs.system_design.changed_files + changed_tasks = self.repo.docs.task.changed_files change_files = Documents() # Rewrite the system designs that have undergone changes based on the git head diff under # `docs/system_designs/`. @@ -57,16 +57,14 @@ class WriteTasks(Action): return ActionOutput(content=change_files.model_dump_json(), instruct_content=change_files) async def _update_tasks(self, filename): - system_design_doc = await self.project_repo.docs.system_design.get(filename) - task_doc = await self.project_repo.docs.task.get(filename) + system_design_doc = await self.repo.docs.system_design.get(filename) + task_doc = await self.repo.docs.task.get(filename) if task_doc: task_doc = await self._merge(system_design_doc=system_design_doc, task_doc=task_doc) - await self.project_repo.docs.task.save_doc( - doc=task_doc, dependencies={system_design_doc.root_relative_path} - ) + await self.repo.docs.task.save_doc(doc=task_doc, dependencies={system_design_doc.root_relative_path}) else: rsp = await self._run_new_tasks(context=system_design_doc.content) - task_doc = await self.project_repo.docs.task.save( + task_doc = await self.repo.docs.task.save( filename=filename, content=rsp.instruct_content.model_dump_json(), dependencies={system_design_doc.root_relative_path}, @@ -87,7 +85,7 @@ class WriteTasks(Action): async def _update_requirements(self, doc): m = json.loads(doc.content) packages = set(m.get("Required Python third-party packages", set())) - requirement_doc = await self.project_repo.get(filename=PACKAGE_REQUIREMENTS_FILENAME) + requirement_doc = await self.repo.get(filename=PACKAGE_REQUIREMENTS_FILENAME) if not requirement_doc: requirement_doc = Document(filename=PACKAGE_REQUIREMENTS_FILENAME, root_path=".", content="") lines = requirement_doc.content.splitlines() @@ -95,4 +93,4 @@ class WriteTasks(Action): if pkg == "": continue packages.add(pkg) - await self.project_repo.save(filename=PACKAGE_REQUIREMENTS_FILENAME, content="\n".join(packages)) + await self.repo.save(filename=PACKAGE_REQUIREMENTS_FILENAME, content="\n".join(packages)) diff --git a/metagpt/actions/summarize_code.py b/metagpt/actions/summarize_code.py index 182561d59..2b5546546 100644 --- a/metagpt/actions/summarize_code.py +++ b/metagpt/actions/summarize_code.py @@ -98,10 +98,10 @@ class SummarizeCode(Action): async def run(self): design_pathname = Path(self.i_context.design_filename) - design_doc = await self.project_repo.docs.system_design.get(filename=design_pathname.name) + design_doc = await self.repo.docs.system_design.get(filename=design_pathname.name) task_pathname = Path(self.i_context.task_filename) - task_doc = await self.project_repo.docs.task.get(filename=task_pathname.name) - src_file_repo = self.project_repo.with_src_path(self.context.src_workspace).srcs + task_doc = await self.repo.docs.task.get(filename=task_pathname.name) + src_file_repo = self.repo.with_src_path(self.context.src_workspace).srcs code_blocks = [] for filename in self.i_context.codes_filenames: code_doc = await src_file_repo.get(filename) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index c0f1b1a93..aaaa9648a 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -88,12 +88,12 @@ class WriteCode(Action): return code async def run(self, *args, **kwargs) -> CodingContext: - bug_feedback = await self.project_repo.docs.get(filename=BUGFIX_FILENAME) + bug_feedback = await self.repo.docs.get(filename=BUGFIX_FILENAME) coding_context = CodingContext.loads(self.i_context.content) - test_doc = await self.project_repo.test_outputs.get(filename="test_" + coding_context.filename + ".json") + test_doc = await self.repo.test_outputs.get(filename="test_" + coding_context.filename + ".json") summary_doc = None if coding_context.design_doc and coding_context.design_doc.filename: - summary_doc = await self.project_repo.docs.code_summary.get(filename=coding_context.design_doc.filename) + summary_doc = await self.repo.docs.code_summary.get(filename=coding_context.design_doc.filename) logs = "" if test_doc: test_detail = RunCodeResult.loads(test_doc.content) @@ -105,7 +105,7 @@ class WriteCode(Action): code_context = await self.get_codes( coding_context.task_doc, exclude=self.i_context.filename, - project_repo=self.project_repo.with_src_path(self.context.src_workspace), + project_repo=self.repo.with_src_path(self.context.src_workspace), ) prompt = PROMPT_TEMPLATE.format( diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index 21281dde1..8b85608ee 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -143,7 +143,7 @@ class WriteCodeReview(Action): code_context = await WriteCode.get_codes( self.i_context.task_doc, exclude=self.i_context.filename, - project_repo=self.project_repo.with_src_path(self.context.src_workspace), + project_repo=self.repo.with_src_path(self.context.src_workspace), ) context = "\n".join( [ diff --git a/metagpt/actions/write_prd.py b/metagpt/actions/write_prd.py index 38ac62536..d401cc588 100644 --- a/metagpt/actions/write_prd.py +++ b/metagpt/actions/write_prd.py @@ -15,7 +15,6 @@ from __future__ import annotations import json from pathlib import Path -from typing import Optional from metagpt.actions import Action, ActionOutput from metagpt.actions.action_node import ActionNode @@ -58,96 +57,106 @@ NEW_REQ_TEMPLATE = """ class WritePRD(Action): - name: str = "WritePRD" - content: Optional[str] = None + """WritePRD deal with the following situations: + 1. Bugfix: If the requirement is a bugfix, the bugfix document will be generated. + 2. New requirement: If the requirement is a new requirement, the PRD document will be generated. + 3. Requirement update: If the requirement is an update, the PRD document will be updated. + """ async def run(self, with_messages, *args, **kwargs) -> ActionOutput | Message: - # Determine which requirement documents need to be rewritten: Use LLM to assess whether new requirements are - # related to the PRD. If they are related, rewrite the PRD. - requirement_doc = await self.project_repo.docs.get(filename=REQUIREMENT_FILENAME) - if requirement_doc and await self._is_bugfix(requirement_doc.content): - await self.project_repo.docs.save(filename=BUGFIX_FILENAME, content=requirement_doc.content) - await self.project_repo.docs.save(filename=REQUIREMENT_FILENAME, content="") - bug_fix = BugFixContext(filename=BUGFIX_FILENAME) - return Message( - content=bug_fix.model_dump_json(), - instruct_content=bug_fix, - role="", - cause_by=FixBug, - sent_from=self, - send_to="Alex", # the name of Engineer - ) + """Run the action.""" + req: Document = await self.repo.requirement + docs: list[Document] = await self.repo.docs.prd.get_all() + if not req: + raise FileNotFoundError("No requirement document found.") + + if await self._is_bugfix(req.content): + logger.info(f"Bugfix detected: {req.content}") + return await self._handle_bugfix(req) + # remove bugfix file from last round in case of conflict + await self.repo.docs.delete(filename=BUGFIX_FILENAME) + + # if requirement is related to other documents, update them, otherwise create a new one + if related_docs := await self.get_related_docs(req, docs): + logger.info(f"Requirement update detected: {req.content}") + return await self._handle_requirement_update(req, related_docs) else: - await self.project_repo.docs.delete(filename=BUGFIX_FILENAME) + logger.info(f"New requirement detected: {req.content}") + return await self._handle_new_requirement(req) - prd_docs = await self.project_repo.docs.prd.get_all() - change_files = Documents() - for prd_doc in prd_docs: - prd_doc = await self._update_prd(requirement_doc=requirement_doc, prd_doc=prd_doc, *args, **kwargs) - if not prd_doc: - continue - change_files.docs[prd_doc.filename] = prd_doc - logger.info(f"rewrite prd: {prd_doc.filename}") - # If there is no existing PRD, generate one using 'docs/requirement.txt'. - if not change_files.docs: - prd_doc = await self._update_prd(requirement_doc=requirement_doc, *args, **kwargs) - if prd_doc: - change_files.docs[prd_doc.filename] = prd_doc - logger.debug(f"new prd: {prd_doc.filename}") - # Once all files under 'docs/prds/' have been compared with the newly added requirements, trigger the - # 'publish' message to transition the workflow to the next stage. This design allows room for global - # optimization in subsequent steps. - return ActionOutput(content=change_files.model_dump_json(), instruct_content=change_files) + async def _handle_bugfix(self, req: Document) -> Message: + # ... bugfix logic ... + await self.repo.docs.save(filename=BUGFIX_FILENAME, content=req.content) + await self.repo.docs.save(filename=REQUIREMENT_FILENAME, content="") + bug_fix = BugFixContext(filename=BUGFIX_FILENAME) + return Message( + content=bug_fix.model_dump_json(), + instruct_content=bug_fix, + role="", + cause_by=FixBug, + sent_from=self, + send_to="Alex", # the name of Engineer + ) - async def _run_new_requirement(self, requirements) -> ActionOutput: + async def _handle_new_requirement(self, req: Document) -> ActionOutput: + """handle new requirement""" project_name = self.project_name - context = CONTEXT_TEMPLATE.format(requirements=requirements, project_name=project_name) + context = CONTEXT_TEMPLATE.format(requirements=req, project_name=project_name) exclude = [PROJECT_NAME.key] if project_name else [] node = await WRITE_PRD_NODE.fill(context=context, llm=self.llm, exclude=exclude) # schema=schema await self._rename_workspace(node) - return node + new_prd_doc = await self.repo.docs.prd.save( + filename=FileRepository.new_filename() + ".json", content=node.instruct_content.model_dump_json() + ) + await self._save_competitive_analysis(new_prd_doc) + await self.repo.resources.prd.save_pdf(doc=new_prd_doc) + return Documents.from_iterable(documents=[new_prd_doc]).to_action_output() - async def _is_relative(self, new_requirement_doc, old_prd_doc) -> bool: - context = NEW_REQ_TEMPLATE.format(old_prd=old_prd_doc.content, requirements=new_requirement_doc.content) + async def _handle_requirement_update(self, req: Document, related_docs: list[Document]) -> ActionOutput: + # ... requirement update logic ... + for doc in related_docs: + await self._update_prd(req, doc) + return Documents.from_iterable(documents=related_docs).to_action_output() + + async def _is_bugfix(self, context: str) -> bool: + if not self.repo.code_files_exists(): + return False + node = await WP_ISSUE_TYPE_NODE.fill(context, self.llm) + return node.get("issue_type") == "BUG" + + async def get_related_docs(self, req: Document, docs: list[Document]) -> list[Document]: + """get the related documents""" + # refine: use gather to speed up + return [i for i in docs if await self._is_related(req, i)] + + async def _is_related(self, req: Document, old_prd: Document) -> bool: + context = NEW_REQ_TEMPLATE.format(old_prd=old_prd.content, requirements=req.content) node = await WP_IS_RELATIVE_NODE.fill(context, self.llm) return node.get("is_relative") == "YES" - async def _merge(self, new_requirement_doc, prd_doc) -> Document: + async def _merge(self, req: Document, related_doc: Document) -> Document: if not self.project_name: self.project_name = Path(self.project_path).name - prompt = NEW_REQ_TEMPLATE.format(requirements=new_requirement_doc.content, old_prd=prd_doc.content) + prompt = NEW_REQ_TEMPLATE.format(requirements=req.content, old_prd=related_doc.content) node = await WRITE_PRD_NODE.fill(context=prompt, llm=self.llm, schema=self.prompt_schema) - prd_doc.content = node.instruct_content.model_dump_json() + related_doc.content = node.instruct_content.model_dump_json() await self._rename_workspace(node) - return prd_doc + return related_doc - async def _update_prd(self, requirement_doc, prd_doc=None, *args, **kwargs) -> Document | None: - if not prd_doc: - prd = await self._run_new_requirement( - requirements=[requirement_doc.content if requirement_doc else ""], *args, **kwargs - ) - new_prd_doc = await self.project_repo.docs.prd.save( - filename=FileRepository.new_filename() + ".json", content=prd.instruct_content.model_dump_json() - ) - elif await self._is_relative(requirement_doc, prd_doc): - new_prd_doc = await self._merge(requirement_doc, prd_doc) - self.project_repo.docs.prd.save_doc(doc=new_prd_doc) - else: - return None + async def _update_prd(self, req: Document, prd_doc: Document) -> Document: + new_prd_doc: Document = await self._merge(req, prd_doc) + self.repo.docs.prd.save_doc(doc=new_prd_doc) await self._save_competitive_analysis(new_prd_doc) - await self.project_repo.resources.prd.save_pdf(doc=new_prd_doc) + await self.repo.resources.prd.save_pdf(doc=new_prd_doc) return new_prd_doc - async def _save_competitive_analysis(self, prd_doc): + async def _save_competitive_analysis(self, prd_doc: Document): m = json.loads(prd_doc.content) quadrant_chart = m.get("Competitive Quadrant Chart") if not quadrant_chart: return - pathname = ( - self.project_repo.workdir / Path(COMPETITIVE_ANALYSIS_FILE_REPO) / Path(prd_doc.filename).with_suffix("") - ) - if not pathname.parent.exists(): - pathname.parent.mkdir(parents=True, exist_ok=True) + pathname = self.repo.workdir / COMPETITIVE_ANALYSIS_FILE_REPO / Path(prd_doc.filename).stem + pathname.parent.mkdir(parents=True, exist_ok=True) await mermaid_to_file(self.config.mermaid_engine, quadrant_chart, pathname) async def _rename_workspace(self, prd): @@ -158,15 +167,4 @@ class WritePRD(Action): ws_name = CodeParser.parse_str(block="Project Name", text=prd) if ws_name: self.project_name = ws_name - self.project_repo.git_repo.rename_root(self.project_name) - - async def _is_bugfix(self, context) -> bool: - git_workdir = self.project_repo.git_repo.workdir - src_workdir = git_workdir / git_workdir.name - if not src_workdir.exists(): - return False - code_files = self.project_repo.with_src_path(path=git_workdir / git_workdir.name).srcs.all_files - if not code_files: - return False - node = await WP_ISSUE_TYPE_NODE.fill(context, self.llm) - return node.get("issue_type") == "BUG" + self.repo.git_repo.rename_root(self.project_name) diff --git a/metagpt/context.py b/metagpt/context.py index 1e0d91237..2f0264f2d 100644 --- a/metagpt/context.py +++ b/metagpt/context.py @@ -17,6 +17,7 @@ from metagpt.provider.base_llm import BaseLLM from metagpt.provider.llm_provider_registry import create_llm_instance from metagpt.utils.cost_manager import CostManager from metagpt.utils.git_repository import GitRepository +from metagpt.utils.project_repo import ProjectRepo class AttrDict(BaseModel): @@ -58,6 +59,8 @@ class Context(BaseModel): kwargs: AttrDict = AttrDict() config: Config = Config.default() + + repo: Optional[ProjectRepo] = None git_repo: Optional[GitRepository] = None src_workspace: Optional[Path] = None cost_manager: CostManager = CostManager() diff --git a/metagpt/roles/searcher.py b/metagpt/roles/searcher.py index e0d2dbb65..19a73a40e 100644 --- a/metagpt/roles/searcher.py +++ b/metagpt/roles/searcher.py @@ -10,8 +10,9 @@ from pydantic import Field -from metagpt.actions import ActionOutput, SearchAndSummarize +from metagpt.actions import SearchAndSummarize from metagpt.actions.action_node import ActionNode +from metagpt.actions.action_output import ActionOutput from metagpt.logs import logger from metagpt.roles import Role from metagpt.schema import Message diff --git a/metagpt/schema.py b/metagpt/schema.py index 853a9c6bb..e9434b9c0 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -23,7 +23,7 @@ from abc import ABC from asyncio import Queue, QueueEmpty, wait_for from json import JSONDecodeError from pathlib import Path -from typing import Any, Dict, List, Optional, Type, TypeVar, Union +from typing import Any, Dict, Iterable, List, Optional, Type, TypeVar, Union from pydantic import ( BaseModel, @@ -36,6 +36,7 @@ from pydantic import ( model_validator, ) +from metagpt.actions.action_output import ActionOutput from metagpt.const import ( MESSAGE_ROUTE_CAUSE_BY, MESSAGE_ROUTE_FROM, @@ -162,6 +163,25 @@ class Documents(BaseModel): docs: Dict[str, Document] = Field(default_factory=dict) + @classmethod + def from_iterable(cls, documents: Iterable[Document]) -> Documents: + """Create a Documents instance from a list of Document instances. + + :param documents: A list of Document instances. + :return: A Documents instance. + """ + + docs = {doc.filename: doc for doc in documents} + return Documents(docs=docs) + + def to_action_output(self) -> ActionOutput: + """Convert to action output string. + + :return: A string representing action output. + """ + + return ActionOutput(content=self.model_dump_json(), instruct_content=self) + class Message(BaseModel): """list[: ]""" diff --git a/metagpt/utils/project_repo.py b/metagpt/utils/project_repo.py index dd54cb56b..77ac4f897 100644 --- a/metagpt/utils/project_repo.py +++ b/metagpt/utils/project_repo.py @@ -21,6 +21,7 @@ from metagpt.const import ( GRAPH_REPO_FILE_REPO, PRD_PDF_FILE_REPO, PRDS_FILE_REPO, + REQUIREMENT_FILENAME, RESOURCES_FILE_REPO, SD_OUTPUT_FILE_REPO, SEQ_FLOW_FILE_REPO, @@ -93,6 +94,10 @@ class ProjectRepo(FileRepository): self.test_outputs = self._git_repo.new_file_repository(relative_path=TEST_OUTPUTS_FILE_REPO) self._srcs_path = None + @property + async def requirement(self): + return await self.docs.get(filename=REQUIREMENT_FILENAME) + @property def git_repo(self) -> GitRepository: return self._git_repo @@ -107,6 +112,15 @@ class ProjectRepo(FileRepository): raise ValueError("Call with_srcs first.") return self._git_repo.new_file_repository(self._srcs_path) + def code_files_exists(self) -> bool: + git_workdir = self.git_repo.workdir + src_workdir = git_workdir / git_workdir.name + if not src_workdir.exists(): + return False + code_files = self.with_src_path(path=git_workdir / git_workdir.name).srcs.all_files + if not code_files: + return False + def with_src_path(self, path: str | Path) -> ProjectRepo: try: self._srcs_path = Path(path).relative_to(self.workdir) From 4feea49b22b61b91fa9244fbd4df6d8732aa09cc Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 15 Jan 2024 16:41:51 +0800 Subject: [PATCH 332/637] refine writeprd code --- metagpt/schema.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/schema.py b/metagpt/schema.py index e9434b9c0..0a7a07b03 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -36,7 +36,6 @@ from pydantic import ( model_validator, ) -from metagpt.actions.action_output import ActionOutput from metagpt.const import ( MESSAGE_ROUTE_CAUSE_BY, MESSAGE_ROUTE_FROM, @@ -174,11 +173,12 @@ class Documents(BaseModel): docs = {doc.filename: doc for doc in documents} return Documents(docs=docs) - def to_action_output(self) -> ActionOutput: + def to_action_output(self) -> "ActionOutput": """Convert to action output string. :return: A string representing action output. """ + from metagpt.actions.action_output import ActionOutput return ActionOutput(content=self.model_dump_json(), instruct_content=self) From b69f2be165a2c76343f5f7b9b03c18321aa5d588 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 16:51:25 +0800 Subject: [PATCH 333/637] delete type. --- metagpt/actions/execute_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 1a97e49d6..fb0ecd893 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -227,7 +227,7 @@ class ExecutePyCode(ExecuteCode, Action): return code, True -def truncate(result: str, keep_len: int = 2000, is_success: bool = True) -> str | bool: +def truncate(result: str, keep_len: int = 2000, is_success: bool = True): desc = f"Executed code {'successfully' if is_success else 'failed, please reflect the cause of bug and then debug'}" if is_success: desc += f"Truncated to show only {keep_len} characters\n" From ab55303fa10139363adb5b37158e4795ea1dde89 Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 15 Jan 2024 16:54:03 +0800 Subject: [PATCH 334/637] fix bug --- examples/example.pkl | Bin 624 -> 624 bytes metagpt/actions/action.py | 2 ++ tests/metagpt/test_context.py | 1 - tests/metagpt/test_context_mixin.py | 2 +- 4 files changed, 3 insertions(+), 2 deletions(-) diff --git a/examples/example.pkl b/examples/example.pkl index 0469a2e4670ab73437d853671ac4f5f4c22606b7..7c6ab901b210830f4436202b85bee6087e92b82c 100644 GIT binary patch delta 103 zcmeys@_}VSw2E;`qJe3ufw8WMk(s5giMesAZla}`fv%--Vser}nqi_@V&cU5SIqnw zf|GL?9aK!yj8c*e%~Es?lao_)O)N}IbQ6t}jCE6u%nZ$q%uRsu29qx^mVz~SFlhh) D8de ProjectRepo: + if not self.context.repo: + self.context.repo = ProjectRepo(self.context.git_repo) return self.context.repo @property diff --git a/tests/metagpt/test_context.py b/tests/metagpt/test_context.py index d662a906a..d90d0b686 100644 --- a/tests/metagpt/test_context.py +++ b/tests/metagpt/test_context.py @@ -48,7 +48,6 @@ def test_context_1(): assert ctx.git_repo is None assert ctx.src_workspace is None assert ctx.cost_manager is not None - assert ctx.options is not None def test_context_2(): diff --git a/tests/metagpt/test_context_mixin.py b/tests/metagpt/test_context_mixin.py index a098ff0dc..a8a096d69 100644 --- a/tests/metagpt/test_context_mixin.py +++ b/tests/metagpt/test_context_mixin.py @@ -95,7 +95,7 @@ def test_config_mixin_4_multi_inheritance_override_config(): print(obj.__dict__.keys()) assert "private_config" in obj.__dict__.keys() - assert obj.llm.model == "mock_zhipu_model" + assert obj.config.llm.model == "mock_zhipu_model" @pytest.mark.asyncio From 4f93c5fad3f03fd0302e3a93760216fc9ca58ffa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 16:59:49 +0800 Subject: [PATCH 335/637] add only_code arg for WriteCodeByGenerate. --- metagpt/actions/write_analysis_code.py | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 04cad34a5..76d47ba28 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -88,8 +88,14 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): ) -> str: # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) prompt = self.process_msg(context, system_msg) + is_only_code = kwargs.pop("only_code", True) + code_content = await self.llm.aask_code(prompt, **kwargs) - return code_content["code"] + if is_only_code: + return code_content["code"] + else: + return code_content + class WriteCodeWithTools(BaseWriteAnalysisCode): From 7f1584db9e5bd153f5f78f2f79d3d16970f20f0c Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 15 Jan 2024 17:26:35 +0800 Subject: [PATCH 336/637] 1. add test_vision.py 2. add save_webpages function in vision.py and vision.yml --- metagpt/tools/functions/libs/vision.py | 64 +++++++++++++++++--- metagpt/tools/functions/schemas/vision.yml | 20 +++++- tests/metagpt/tools/functions/test_vision.py | 40 ++++++++++++ 3 files changed, 113 insertions(+), 11 deletions(-) create mode 100644 tests/metagpt/tools/functions/test_vision.py diff --git a/metagpt/tools/functions/libs/vision.py b/metagpt/tools/functions/libs/vision.py index 8c29b0567..b10ad7608 100644 --- a/metagpt/tools/functions/libs/vision.py +++ b/metagpt/tools/functions/libs/vision.py @@ -5,6 +5,8 @@ @Author : mannaandpoem @File : vision.py """ +from pathlib import Path + import requests import base64 @@ -34,8 +36,9 @@ Now, please generate the corresponding webpage code including HTML, CSS and Java class Vision: def __init__(self): self.api_key = API_KEY + self.api_base = OPENAI_API_BASE self.model = MODEL - self.max_tokens = 4096 + self.max_tokens = MAX_TOKENS def analyze_layout(self, image_path): return self.get_result(image_path, ANALYZE_LAYOUT_PROMPT) @@ -43,7 +46,8 @@ class Vision: def generate_web_pages(self, image_path): layout = self.analyze_layout(image_path) prompt = GENERATE_PROMPT + "\n\n # Context\n The layout information of the sketch image is: \n" + layout - return self.get_result(image_path, prompt) + result = self.get_result(image_path, prompt) + return result def get_result(self, image_path, prompt): base64_image = self.encode_image(image_path) @@ -67,17 +71,59 @@ class Vision: ], "max_tokens": self.max_tokens, } - response = requests.post(f"{OPENAI_API_BASE}/chat/completions", headers=headers, json=payload) - return response.json()["choices"][0]["message"]["content"] + response = requests.post(f"{self.api_base}/chat/completions", headers=headers, json=payload) + + if response.status_code != 200: + raise ValueError(f"Request failed with status {response.status_code}, {response.text}") + else: + return response.json()["choices"][0]["message"]["content"] @staticmethod def encode_image(image_path): with open(image_path, "rb") as image_file: return base64.b64encode(image_file.read()).decode('utf-8') + @staticmethod + def save_webpages(image_path, webpages) -> Path: + # 在当前目录下创建一个名为webpages的文件夹,用于存储html、css和js文件 + webpages_path = Path(image_path).parent / "webpages" + webpages_path.mkdir(exist_ok=True) -if __name__ == "__main__": - image_path = "image.png" - vision = Vision() - rsp = vision.generate_web_pages(image_path=image_path) - print(rsp) + try: + index_path = webpages_path / "index.html" + index = webpages.split("```html")[1].split("```")[0] + except IndexError: + raise ValueError("No html code found in the result, please check your image and try again.") + + try: + if "styles.css" in index: + style_path = webpages_path / "styles.css" + elif "style.css" in index: + style_path = webpages_path / "style.css" + else: + style_path = None + style = webpages.split("```css")[1].split("```")[0] if style_path else "" + + if "scripts.js" in index: + js_path = webpages_path / "scripts.js" + elif "script.js" in index: + js_path = webpages_path / "script.js" + else: + js_path = None + js = webpages.split("```javascript")[1].split("```")[0] if js_path else "" + except IndexError: + raise ValueError("No css or js code found in the result, please check your image and try again.") + + try: + with open(index_path, "w") as f: + f.write(index) + if style_path: + with open(style_path, "w") as f: + f.write(style) + if js_path: + with open(js_path, "w") as f: + f.write(js) + except FileNotFoundError as e: + raise FileNotFoundError(f"Cannot save the webpages to {str(webpages_path)}") from e + + return webpages_path diff --git a/metagpt/tools/functions/schemas/vision.yml b/metagpt/tools/functions/schemas/vision.yml index 795854e75..4cb247419 100644 --- a/metagpt/tools/functions/schemas/vision.yml +++ b/metagpt/tools/functions/schemas/vision.yml @@ -12,9 +12,25 @@ Vision: image_path: type: str description: "The path of the image file" - required: - image_path returns: type: str - description: "Generated web page content." \ No newline at end of file + description: "Generated webpages content." + + save_webpages: + description: "Save webpages including all code(HTML, CSS and JavaScript) at once" + parameters: + properties: + image_path: + type: str + description: "The path of the image file" + webpages: + type: str + description: "The generated webpages content" + required: + - image_path + - webpages + returns: + type: Path + description: "The path of the saved webpages" \ No newline at end of file diff --git a/tests/metagpt/tools/functions/test_vision.py b/tests/metagpt/tools/functions/test_vision.py new file mode 100644 index 000000000..0359f14f1 --- /dev/null +++ b/tests/metagpt/tools/functions/test_vision.py @@ -0,0 +1,40 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/15 +@Author : mannaandpoem +@File : test_vision.py +""" +import base64 +from unittest.mock import AsyncMock + +from pytest_mock import mocker + +from metagpt import logs +from metagpt.tools.functions.libs.vision import Vision + + +def test_vision_generate_web_pages(): + image_path = "./image.png" + vision = Vision() + rsp = vision.generate_web_pages(image_path=image_path) + logs.logger.info(rsp) + assert "html" in rsp + assert "css" in rsp + assert "javascript" in rsp + + +def test_save_webpages(): + image_path = "./image.png" + vision = Vision() + webpages = """```html: \n + \n``` + "```css: .class { ... } ```\n ```javascript: function() { ... }```""" + webpages_dir = vision.save_webpages(image_path=image_path, webpages=webpages) + logs.logger.info(webpages_dir) + assert webpages_dir.exists() + assert (webpages_dir / "index.html").exists() + assert (webpages_dir / "style.css").exists() or (webpages_dir / "styles.css").exists() + assert (webpages_dir / "script.js").exists() or (webpages_dir / "scripts.js").exists() + + From c715b9c10269856e7f6cf1c49cea4615a8a7733a Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 15 Jan 2024 16:58:01 +0800 Subject: [PATCH 337/637] fix bug --- metagpt/context.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/context.py b/metagpt/context.py index 2f0264f2d..8e9749d66 100644 --- a/metagpt/context.py +++ b/metagpt/context.py @@ -70,8 +70,8 @@ class Context(BaseModel): def new_environ(self): """Return a new os.environ object""" env = os.environ.copy() - i = self.options - env.update({k: v for k, v in i.items() if isinstance(v, str)}) + # i = self.options + # env.update({k: v for k, v in i.items() if isinstance(v, str)}) return env # def use_llm(self, name: Optional[str] = None, provider: LLMType = LLMType.OPENAI) -> BaseLLM: From 92d31fb7509a50eabfe88edd358ea6ed5e879b23 Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 15 Jan 2024 17:53:24 +0800 Subject: [PATCH 338/637] fix bugs --- tests/data/rsp_cache.json | 20 ++++++++++++++++++- tests/metagpt/learn/test_text_to_embedding.py | 3 ++- tests/metagpt/provider/test_openai.py | 3 ++- .../tools/test_openai_text_to_embedding.py | 3 ++- 4 files changed, 25 insertions(+), 4 deletions(-) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index df5300feb..b83222120 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -197,5 +197,23 @@ "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\nimport asyncio\nimport shutil\nfrom pathlib import Path\n\nimport typer\n\nfrom metagpt.config2 import config\nfrom metagpt.const import CONFIG_ROOT, METAGPT_ROOT\n\napp = typer.Typer(add_completion=False, pretty_exceptions_show_locals=False)\n\n\ndef generate_repo(\n idea,\n investment,\n n_round,\n code_review,\n run_tests,\n implement,\n project_name,\n inc,\n project_path,\n reqa_file,\n max_auto_summarize_code,\n recover_path,\n):\n \"\"\"Run the startup logic. Can be called from CLI or other Python scripts.\"\"\"\n from metagpt.roles import (\n Architect,\n Engineer,\n ProductManager,\n ProjectManager,\n QaEngineer,\n )\n from metagpt.team import Team\n\n config.update_via_cli(project_path, project_name, inc, reqa_file, max_auto_summarize_code)\n\n if not recover_path:\n company = Team()\n company.hire(\n [\n ProductManager(),\n Architect(),\n ProjectManager(),\n ]\n )\n\n if implement or code_review:\n company.hire([Engineer(n_borg=5, use_code_review=code_review)])\n\n if run_tests:\n company.hire([QaEngineer()])\n else:\n stg_path = Path(recover_path)\n if not stg_path.exists() or not str(stg_path).endswith(\"team\"):\n raise FileNotFoundError(f\"{recover_path} not exists or not endswith `team`\")\n\n company = Team.deserialize(stg_path=stg_path)\n idea = company.idea\n\n company.invest(investment)\n company.run_project(idea)\n asyncio.run(company.run(n_round=n_round))\n\n\n@app.command(\"\", help=\"Start a new project.\")\ndef startup(\n idea: str = typer.Argument(None, help=\"Your innovative idea, such as 'Create a 2048 game.'\"),\n investment: float = typer.Option(default=3.0, help=\"Dollar amount to invest in the AI company.\"),\n n_round: int = typer.Option(default=5, help=\"Number of rounds for the simulation.\"),\n code_review: bool = typer.Option(default=True, help=\"Whether to use code review.\"),\n run_tests: bool = typer.Option(default=False, help=\"Whether to enable QA for adding & running tests.\"),\n implement: bool = typer.Option(default=True, help=\"Enable or disable code implementation.\"),\n project_name: str = typer.Option(default=\"\", help=\"Unique project name, such as 'game_2048'.\"),\n inc: bool = typer.Option(default=False, help=\"Incremental mode. Use it to coop with existing repo.\"),\n project_path: str = typer.Option(\n default=\"\",\n help=\"Specify the directory path of the old version project to fulfill the incremental requirements.\",\n ),\n reqa_file: str = typer.Option(\n default=\"\", help=\"Specify the source file name for rewriting the quality assurance code.\"\n ),\n max_auto_summarize_code: int = typer.Option(\n default=0,\n help=\"The maximum number of times the 'SummarizeCode' action is automatically invoked, with -1 indicating \"\n \"unlimited. This parameter is used for debugging the workflow.\",\n ),\n recover_path: str = typer.Option(default=None, help=\"recover the project from existing serialized storage\"),\n init_config: bool = typer.Option(default=False, help=\"Initialize the configuration file for MetaGPT.\"),\n):\n \"\"\"Run a startup. Be a boss.\"\"\"\n if init_config:\n copy_config_to()\n return\n\n if idea is None:\n typer.echo(\"Missing argument 'IDEA'. Run 'metagpt --help' for more information.\")\n raise typer.Exit()\n\n return generate_repo(\n idea,\n investment,\n n_round,\n code_review,\n run_tests,\n implement,\n project_name,\n inc,\n project_path,\n reqa_file,\n max_auto_summarize_code,\n recover_path,\n )\n\n\ndef copy_config_to(config_path=METAGPT_ROOT / \"config\" / \"config2.yaml\"):\n \"\"\"Initialize the configuration file for MetaGPT.\"\"\"\n target_path = CONFIG_ROOT / \"config2.yaml\"\n\n # 创建目标目录(如果不存在)\n target_path.parent.mkdir(parents=True, exist_ok=True)\n\n # 如果目标文件已经存在,则重命名为 .bak\n if target_path.exists():\n backup_path = target_path.with_suffix(\".bak\")\n target_path.rename(backup_path)\n print(f\"Existing configuration file backed up at {backup_path}\")\n\n # 复制文件\n shutil.copy(str(config_path), target_path)\n print(f\"Configuration file initialized at {target_path}\")\n\n\nif __name__ == \"__main__\":\n app()\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant app\n participant generate_repo\n participant copy_config_to\n participant Team\n participant ProductManager\n participant Architect\n participant ProjectManager\n participant Engineer\n participant QaEngineer\n\n app -> generate_repo: startup()\n generate_repo -> config: update_via_cli()\n generate_repo -> Team: hire()\n Team -> ProductManager: hire()\n Team -> Architect: hire()\n Team -> ProjectManager: hire()\n generate_repo -> Engineer: hire()\n generate_repo -> QaEngineer: hire()\n generate_repo -> Team: invest()\n generate_repo -> Team: run_project()\n generate_repo -> Team: run()\n\n app -> copy_config_to: copy_config_to()\n copy_config_to -> config: update_via_cli()\n```", "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Time : 2023/12/14 11:40\n@Author : alexanderwu\n@File : write_prd_an.py\n\"\"\"\nfrom typing import List\n\nfrom metagpt.actions.action_node import ActionNode\n\nLANGUAGE = ActionNode(\n key=\"Language\",\n expected_type=str,\n instruction=\"Provide the language used in the project, typically matching the user's requirement language.\",\n example=\"en_us\",\n)\n\nPROGRAMMING_LANGUAGE = ActionNode(\n key=\"Programming Language\",\n expected_type=str,\n instruction=\"Python/JavaScript or other mainstream programming language.\",\n example=\"Python\",\n)\n\nORIGINAL_REQUIREMENTS = ActionNode(\n key=\"Original Requirements\",\n expected_type=str,\n instruction=\"Place the original user's requirements here.\",\n example=\"Create a 2048 game\",\n)\n\nPROJECT_NAME = ActionNode(\n key=\"Project Name\",\n expected_type=str,\n instruction='According to the content of \"Original Requirements,\" name the project using snake case style , '\n \"like 'game_2048' or 'simple_crm.\",\n example=\"game_2048\",\n)\n\nPRODUCT_GOALS = ActionNode(\n key=\"Product Goals\",\n expected_type=List[str],\n instruction=\"Provide up to three clear, orthogonal product goals.\",\n example=[\"Create an engaging user experience\", \"Improve accessibility, be responsive\", \"More beautiful UI\"],\n)\n\nUSER_STORIES = ActionNode(\n key=\"User Stories\",\n expected_type=List[str],\n instruction=\"Provide up to 3 to 5 scenario-based user stories.\",\n example=[\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\",\n ],\n)\n\nCOMPETITIVE_ANALYSIS = ActionNode(\n key=\"Competitive Analysis\",\n expected_type=List[str],\n instruction=\"Provide 5 to 7 competitive products.\",\n example=[\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\",\n ],\n)\n\nCOMPETITIVE_QUADRANT_CHART = ActionNode(\n key=\"Competitive Quadrant Chart\",\n expected_type=str,\n instruction=\"Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\",\n example=\"\"\"quadrantChart\n title \"Reach and engagement of campaigns\"\n x-axis \"Low Reach\" --> \"High Reach\"\n y-axis \"Low Engagement\" --> \"High Engagement\"\n quadrant-1 \"We should expand\"\n quadrant-2 \"Need to promote\"\n quadrant-3 \"Re-evaluate\"\n quadrant-4 \"May be improved\"\n \"Campaign A\": [0.3, 0.6]\n \"Campaign B\": [0.45, 0.23]\n \"Campaign C\": [0.57, 0.69]\n \"Campaign D\": [0.78, 0.34]\n \"Campaign E\": [0.40, 0.34]\n \"Campaign F\": [0.35, 0.78]\n \"Our Target Product\": [0.5, 0.6]\"\"\",\n)\n\nREQUIREMENT_ANALYSIS = ActionNode(\n key=\"Requirement Analysis\",\n expected_type=str,\n instruction=\"Provide a detailed analysis of the requirements.\",\n example=\"\",\n)\n\nREQUIREMENT_POOL = ActionNode(\n key=\"Requirement Pool\",\n expected_type=List[List[str]],\n instruction=\"List down the top-5 requirements with their priority (P0, P1, P2).\",\n example=[[\"P0\", \"The main code ...\"], [\"P0\", \"The game algorithm ...\"]],\n)\n\nUI_DESIGN_DRAFT = ActionNode(\n key=\"UI Design draft\",\n expected_type=str,\n instruction=\"Provide a simple description of UI elements, functions, style, and layout.\",\n example=\"Basic function description with a simple style and layout.\",\n)\n\nANYTHING_UNCLEAR = ActionNode(\n key=\"Anything UNCLEAR\",\n expected_type=str,\n instruction=\"Mention any aspects of the project that are unclear and try to clarify them.\",\n example=\"\",\n)\n\nISSUE_TYPE = ActionNode(\n key=\"issue_type\",\n expected_type=str,\n instruction=\"Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement\",\n example=\"BUG\",\n)\n\nIS_RELATIVE = ActionNode(\n key=\"is_relative\",\n expected_type=str,\n instruction=\"Answer YES/NO. If the requirement is related to the old PRD, answer YES, otherwise NO\",\n example=\"YES\",\n)\n\nREASON = ActionNode(\n key=\"reason\", expected_type=str, instruction=\"Explain the reasoning process from question to answer\", example=\"...\"\n)\n\n\nNODES = [\n LANGUAGE,\n PROGRAMMING_LANGUAGE,\n ORIGINAL_REQUIREMENTS,\n PROJECT_NAME,\n PRODUCT_GOALS,\n USER_STORIES,\n COMPETITIVE_ANALYSIS,\n COMPETITIVE_QUADRANT_CHART,\n REQUIREMENT_ANALYSIS,\n REQUIREMENT_POOL,\n UI_DESIGN_DRAFT,\n ANYTHING_UNCLEAR,\n]\n\nWRITE_PRD_NODE = ActionNode.from_children(\"WritePRD\", NODES)\nWP_ISSUE_TYPE_NODE = ActionNode.from_children(\"WP_ISSUE_TYPE\", [ISSUE_TYPE, REASON])\nWP_IS_RELATIVE_NODE = ActionNode.from_children(\"WP_IS_RELATIVE\", [IS_RELATIVE, REASON])\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nclassDef actionNode fill:#f9f,stroke:#333,stroke-width:2px;\nclassDef actionNodeTitle fill:#f9f,stroke:#333,stroke-width:2px,font-weight:bold;\nclassDef actionNodeExample fill:#f9f,stroke:#333,stroke-width:2px,font-style:italic;\n\nclass ActionNodeTitle actionNodeTitle\nclass ActionNodeExample actionNodeExample\n\nActionNodeTitle:::Language --> \"Language\"\nActionNodeExample:::Language --> \"Provide the language used in the project, typically matching the user's requirement language.\\nExample: en_us\"\n\nActionNodeTitle:::ProgrammingLanguage --> \"Programming Language\"\nActionNodeExample:::ProgrammingLanguage --> \"Python/JavaScript or other mainstream programming language.\\nExample: Python\"\n\nActionNodeTitle:::OriginalRequirements --> \"Original Requirements\"\nActionNodeExample:::OriginalRequirements --> \"Place the original user's requirements here.\\nExample: Create a 2048 game\"\n\nActionNodeTitle:::ProjectName --> \"Project Name\"\nActionNodeExample:::ProjectName --> 'According to the content of \"Original Requirements,\" name the project using snake case style , like \\'game_2048\\' or \\'simple_crm.\\nExample: game_2048'\n\nActionNodeTitle:::ProductGoals --> \"Product Goals\"\nActionNodeExample:::ProductGoals --> \"Provide up to three clear, orthogonal product goals.\\nExample:\\n- Create an engaging user experience\\n- Improve accessibility, be responsive\\n- More beautiful UI\"\n\nActionNodeTitle:::UserStories --> \"User Stories\"\nActionNodeExample:::UserStories --> \"Provide up to 3 to 5 scenario-based user stories.\\nExample:\\n- As a player, I want to be able to choose difficulty levels\\n- As a player, I want to see my score after each game\\n- As a player, I want to get restart button when I lose\\n- As a player, I want to see beautiful UI that make me feel good\\n- As a player, I want to play game via mobile phone\"\n\nActionNodeTitle:::CompetitiveAnalysis --> \"Competitive Analysis\"\nActionNodeExample:::CompetitiveAnalysis --> \"Provide 5 to 7 competitive products.\\nExample:\\n- 2048 Game A: Simple interface, lacks responsive features\\n- play2048.co: Beautiful and responsive UI with my best score shown\\n- 2048game.com: Responsive UI with my best score shown, but many ads\"\n\nActionNodeTitle:::CompetitiveQuadrantChart --> \"Competitive Quadrant Chart\"\nActionNodeExample:::CompetitiveQuadrantChart --> \"Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\\nExample:\\nquadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\"\n\nActionNodeTitle:::RequirementAnalysis --> \"Requirement Analysis\"\nActionNodeExample:::RequirementAnalysis --> \"Provide a detailed analysis of the requirements.\\nExample: \"\n\nActionNodeTitle:::RequirementPool --> \"Requirement Pool\"\nActionNodeExample:::RequirementPool --> \"List down the top-5 requirements with their priority (P0, P1, P2).\\nExample:\\n- P0: The main code ...\\n- P0: The game algorithm ...\"\n\nActionNodeTitle:::UIDesignDraft --> \"UI Design draft\"\nActionNodeExample:::UIDesignDraft --> \"Provide a simple description of UI elements, functions, style, and layout.\\nExample: Basic function description with a simple style and layout.\"\n\nActionNodeTitle:::AnythingUNCLEAR --> \"Anything UNCLEAR\"\nActionNodeExample:::AnythingUNCLEAR --> \"Mention any aspects of the project that are unclear and try to clarify them.\\nExample: \"\n\nActionNodeTitle:::issue_type --> \"issue_type\"\nActionNodeExample:::issue_type --> \"Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement\\nExample: BUG\"\n\nActionNodeTitle:::is_relative --> \"is_relative\"\nActionNodeExample:::is_relative --> \"Answer YES/NO. If the requirement is related to the old PRD, answer YES, otherwise NO\\nExample: YES\"\n\nActionNodeTitle:::reason --> \"reason\"\nActionNodeExample:::reason --> \"Explain the reasoning process from question to answer\\nExample: ...\"\n\nActionNodeTitle:::WritePRD --> \"WritePRD\"\nActionNodeExample:::WritePRD --> \"Language\\nProgramming Language\\nOriginal Requirements\\nProject Name\\nProduct Goals\\nUser Stories\\nCompetitive Analysis\\nCompetitive Quadrant Chart\\nRequirement Analysis\\nRequirement Pool\\nUI Design draft\\nAnything UNCLEAR\"\n\nActionNodeTitle:::WP_ISSUE_TYPE --> \"WP_ISSUE_TYPE\"\nActionNodeExample:::WP_ISSUE_TYPE --> \"issue_type\\nreason\"\n\nActionNodeTitle:::WP_IS_RELATIVE --> \"WP_IS_RELATIVE\"\nActionNodeExample:::WP_IS_RELATIVE --> \"is_relative\\nreason\"\n```", "\n## context\n\n### Project Name\n20240112110833\n\n### Original Requirements\n['开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"zh_cn\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结\",\n \"Product Goals\": [],\n \"User Stories\": [],\n \"Competitive Analysis\": [],\n \"Competitive Quadrant Chart\": \"\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [],\n \"UI Design draft\": \"\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", - "\n## context\n\n### Project Name\n20240112110833\n\n### Original Requirements\n['']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"\",\n \"Product Goals\": [],\n \"User Stories\": [],\n \"Competitive Analysis\": [],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]" + "\n## context\n\n### Project Name\n20240112110833\n\n### Original Requirements\n['']\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"\",\n \"Product Goals\": [],\n \"User Stories\": [],\n \"Competitive Analysis\": [],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\n## context\n\n### Project Name\n\n\n### Original Requirements\n需要一个基于LLM做总结的搜索引擎\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"zh_cn\",\n \"Programming Language\": \"LLM\",\n \"Original Requirements\": \"需要一个基于LLM做总结的搜索引擎\",\n \"Project Name\": \"llm_summary_search_engine\",\n \"Product Goals\": [\n \"提供准确的搜索结果\",\n \"提高搜索引擎的效率\",\n \"优化用户体验\"\n ],\n \"User Stories\": [\n \"作为用户,我希望能够快速找到我需要的信息\",\n \"作为用户,我希望搜索结果准确无误\",\n \"作为用户,我希望搜索引擎能够智能推荐相关内容\"\n ],\n \"Competitive Analysis\": [\n \"搜索引擎A: 提供准确的搜索结果,但界面简陋\",\n \"搜索引擎B: 界面美观,但搜索结果不够准确\",\n \"搜索引擎C: 搜索结果准确,但速度较慢\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"搜索引擎比较\\\"\\n x-axis \\\"低准确性\\\" --> \\\"高准确性\\\"\\n y-axis \\\"低速度\\\" --> \\\"高速度\\\"\\n quadrant-1 \\\"需要改进\\\"\\n quadrant-2 \\\"值得推广\\\"\\n quadrant-3 \\\"重新评估\\\"\\n quadrant-4 \\\"需要扩展\\\"\\n \\\"搜索引擎A\\\": [0.8, 0.3]\\n \\\"搜索引擎B\\\": [0.5, 0.7]\\n \\\"搜索引擎C\\\": [0.9, 0.2]\\n \\\"我们的搜索引擎\\\": [0.7, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"实现LLM技术进行文本摘要\"\n ],\n [\n \"P0\",\n \"构建搜索算法,提高搜索效率\"\n ],\n [\n \"P1\",\n \"设计智能推荐系统\"\n ]\n ],\n \"UI Design draft\": \"简洁的搜索框,清晰的搜索结果页面,智能推荐模块\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\n## context\n\n### Project Name\n\n\n### Original Requirements\nMake a cli snake game\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Make a cli snake game\",\n \"Project Name\": \"cli_snake_game\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility and responsiveness\",\n \"Enhance the game with additional features\"\n ],\n \"User Stories\": [\n \"As a player, I want to control the snake using arrow keys\",\n \"As a player, I want to see my score during the game\",\n \"As a player, I want to have the option to restart the game\",\n \"As a player, I want to see a visually appealing UI\",\n \"As a player, I want to play the game on different platforms\"\n ],\n \"Competitive Analysis\": [\n \"Snake Game A: Simple interface, lacks responsive features\",\n \"SnakeGame.co: Beautiful and responsive UI with high scores displayed\",\n \"SnakeGame.com: Responsive UI with high scores shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of snake games\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Snake Game A\\\": [0.3, 0.6]\\n \\\"SnakeGame.co\\\": [0.45, 0.23]\\n \\\"SnakeGame.com\\\": [0.57, 0.69]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code for controlling the snake and game logic\"\n ],\n [\n \"P1\",\n \"Implementing the scoring system and UI\"\n ],\n [\n \"P2\",\n \"Adding platform compatibility and restart functionality\"\n ]\n ],\n \"UI Design draft\": \"The game will have a simple and intuitive UI with clear controls and a visually appealing design.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\n## context\n{\"Language\":\"en_us\",\"Programming Language\":\"Python\",\"Original Requirements\":\"Make a cli snake game\",\"Project Name\":\"cli_snake_game\",\"Product Goals\":[\"Create an engaging user experience\",\"Improve accessibility and responsiveness\",\"Enhance the game with additional features\"],\"User Stories\":[\"As a player, I want to control the snake using arrow keys\",\"As a player, I want to see my score during the game\",\"As a player, I want to have the option to restart the game\",\"As a player, I want to see a visually appealing UI\",\"As a player, I want to play the game on different platforms\"],\"Competitive Analysis\":[\"Snake Game A: Simple interface, lacks responsive features\",\"SnakeGame.co: Beautiful and responsive UI with high scores displayed\",\"SnakeGame.com: Responsive UI with high scores shown, but many ads\"],\"Competitive Quadrant Chart\":\"quadrantChart\\n title \\\"Reach and engagement of snake games\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Snake Game A\\\": [0.3, 0.6]\\n \\\"SnakeGame.co\\\": [0.45, 0.23]\\n \\\"SnakeGame.com\\\": [0.57, 0.69]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\"Requirement Analysis\":\"\",\"Requirement Pool\":[[\"P0\",\"The main code for controlling the snake and game logic\"],[\"P1\",\"Implementing the scoring system and UI\"],[\"P2\",\"Adding platform compatibility and restart functionality\"]],\"UI Design draft\":\"The game will have a simple and intuitive UI with clear controls and a visually appealing design.\",\"Anything UNCLEAR\":\"\"}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Implementation approach\": \"We will ...\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Implementation approach: # Analyze the difficult points of the requirements, select the appropriate open-source framework\n- File list: typing.List[str] # Only need relative paths. ALWAYS write a main.py or app.py here\n- Data structures and interfaces: # Use mermaid classDiagram code syntax, including classes, method(__init__ etc.) and functions with type annotations, CLEARLY MARK the RELATIONSHIPS between classes, and comply with PEP8 standards. The data structures SHOULD BE VERY DETAILED and the API should be comprehensive with a complete design.\n- Program call flow: # Use sequenceDiagram code syntax, COMPLETE and VERY DETAILED, using CLASSES AND API DEFINED ABOVE accurately, covering the CRUD AND INIT of each object, SYNTAX MUST BE CORRECT.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Implementation approach\": \"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\n \"File list\": [\n \"main.py\",\n \"game.py\"\n ],\n \"Data structures and interfaces\": \"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\n \"Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on the scoring system and how the game over condition will be handled.\"\n}\n[/CONTENT]", + "\n## context\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Required Python packages\": [\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"openapi: 3.0.0 ...\",\n \"Shared Knowledge\": \"'game.py' contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on how to start and initialize third-party libraries.\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Required Python packages: typing.List[str] # Provide required Python packages in requirements.txt format.\n- Required Other language third-party packages: typing.List[str] # List down the required packages for languages other than Python.\n- Logic Analysis: typing.List[typing.List[str]] # Provide a list of files with the classes/methods/functions to be implemented, including dependency analysis and imports.\n- Task list: typing.List[str] # Break down the tasks into a list of filenames, prioritized by dependency order.\n- Full API spec: # Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end and back-end communication is not required, leave it blank.\n- Shared Knowledge: # Detail any shared knowledge, like common utility functions or configuration variables.\n- Anything UNCLEAR: # Mention any unclear aspects in the project management context and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Required Python packages\": [\n \"pygame==2.0.1\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"\",\n \"Shared Knowledge\": \"'game.py' contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on the scoring system and how the game over condition will be handled.\"\n}\n[/CONTENT]", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = []\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = []\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n\n```\n\n\n\n# Format example 1\n## Code Review: game.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: game.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: game.py\n1. No, the code is not implemented as per the requirements. The Game class should have a method to handle game over condition and update the scoring system.\n2. No, the code logic is not completely correct. The Snake class is missing the implementation of the move, change_direction, and grow methods.\n3. Yes, the existing code follows the \"Data structures and interfaces\" provided.\n4. No, the move, change_direction, and grow methods of the Snake class are not implemented. To achieve this, the methods need to be implemented with the appropriate logic for moving the snake, changing its direction, and growing its body.\n5. Yes, all necessary pre-dependencies have been imported.\n6. No, there are no methods from other files being reused in the current code.\n\n## Actions\n1. Implement a method in the Game class to handle the game over condition and update the scoring system.\n2. Implement the move, change_direction, and grow methods in the Snake class.\n3. Reuse methods from other files if necessary.\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = []\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n\n```\n\n## Code Review: game.py\n1. No, the code is not implemented as per the requirements. The Game class should have a method to handle game over condition and update the scoring system.\n2. No, the code logic is not completely correct. The Snake class is missing the implementation of the move, change_direction, and grow methods.\n3. Yes, the existing code follows the \"Data structures and interfaces\" provided.\n4. No, the move, change_direction, and grow methods of the Snake class are not implemented. To achieve this, the methods need to be implemented with the appropriate logic for moving the snake, changing its direction, and growing its body.\n5. Yes, all necessary pre-dependencies have been imported.\n6. No, there are no methods from other files being reused in the current code.\n\n## Actions\n1. Implement a method in the Game class to handle the game over condition and update the scoring system.\n2. Implement the move, change_direction, and grow methods in the Snake class.\n3. Reuse methods from other files if necessary.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite game.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## game.py\n...\n```\n": "```Code\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n def handle_game_over(self):\n # Handle game over condition\n pass\n\n def update_score(self):\n # Update scoring system\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = []\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n def handle_game_over(self):\n # Handle game over condition\n pass\n\n def update_score(self):\n # Update scoring system\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = []\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n\n```\n\n\n\n# Format example 1\n## Code Review: game.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: game.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: game.py\n1. No, the code is not implemented as per the requirements. The `handle_game_over` and `update_score` methods are missing, and the `handle_input` method needs to be implemented to handle user input.\n2. The code logic is not completely correct. The `Snake` class `__init__` method does not initialize the `body` attribute with the `start_pos` and `length` parameters.\n3. The existing code follows the \"Data structures and interfaces\" for the `Game` and `Snake` classes.\n4. The `handle_game_over` and `update_score` methods are not implemented. The `handle_input` method needs to be implemented to handle user input.\n5. The necessary pre-dependencies have been imported.\n6. The methods from other files are not being reused in the existing code.\n\n## Actions\n1. Implement the `handle_game_over` method to handle the game over condition.\n2. Implement the `update_score` method to update the scoring system.\n3. Implement the `handle_input` method to handle user input and update the game state accordingly.\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n def handle_game_over(self):\n # Handle game over condition\n pass\n\n def update_score(self):\n # Update scoring system\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = []\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n\n```\n\n## Code Review: game.py\n1. No, the code is not implemented as per the requirements. The `handle_game_over` and `update_score` methods are missing, and the `handle_input` method needs to be implemented to handle user input.\n2. The code logic is not completely correct. The `Snake` class `__init__` method does not initialize the `body` attribute with the `start_pos` and `length` parameters.\n3. The existing code follows the \"Data structures and interfaces\" for the `Game` and `Snake` classes.\n4. The `handle_game_over` and `update_score` methods are not implemented. The `handle_input` method needs to be implemented to handle user input.\n5. The necessary pre-dependencies have been imported.\n6. The methods from other files are not being reused in the existing code.\n\n## Actions\n1. Implement the `handle_game_over` method to handle the game over condition.\n2. Implement the `update_score` method to update the scoring system.\n3. Implement the `handle_input` method to handle user input and update the game state accordingly.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite game.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## game.py\n...\n```\n": "```Code\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n def handle_game_over(self):\n # Handle game over condition\n pass\n\n def update_score(self):\n # Update scoring system\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos] # Initialize body with start_pos\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Legacy Code\n```Code\n----- game.py\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n def handle_game_over(self):\n # Handle game over condition\n pass\n\n def update_score(self):\n # Update scoring system\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos] # Initialize body with start_pos\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: main.py\n```python\n## main.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: main.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.handle_input()\n game.update()\n game.draw()\n\n if game.handle_game_over():\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n----- game.py\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n def handle_game_over(self):\n # Handle game over condition\n pass\n\n def update_score(self):\n # Update scoring system\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos] # Initialize body with start_pos\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.handle_input()\n game.update()\n game.draw()\n\n if game.handle_game_over():\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n\n\n# Format example 1\n## Code Review: main.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: main.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: main.py\n1. The code is implemented as per the requirements, but the game over condition and scoring system need to be clarified and implemented.\n \n## Actions\n1. Clarify and implement the game over condition and scoring system in the `Game` class.\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n----- game.py\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n def handle_game_over(self):\n # Handle game over condition\n pass\n\n def update_score(self):\n # Update scoring system\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos] # Initialize body with start_pos\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.handle_input()\n game.update()\n game.draw()\n\n if game.handle_game_over():\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n## Code Review: main.py\n1. The code is implemented as per the requirements, but the game over condition and scoring system need to be clarified and implemented.\n \n## Actions\n1. Clarify and implement the game over condition and scoring system in the `Game` class.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite main.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## main.py\n...\n```\n": "```python\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.handle_input()\n game.update()\n game.draw()\n\n if game.is_game_over():\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n----- game.py\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n def handle_game_over(self):\n # Handle game over condition\n pass\n\n def update_score(self):\n # Update scoring system\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos] # Initialize body with start_pos\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.handle_input()\n game.update()\n game.draw()\n\n if game.is_game_over():\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n\n\n# Format example 1\n## Code Review: main.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: main.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: main.py\n1. The code is implemented as per the requirements, but the `is_game_over` method is missing. To achieve this, we need to implement the `is_game_over` method in the Game class to check for the game over condition.\n2. The code logic is correct, but the missing `is_game_over` method needs to be added to handle the game over condition.\n3. The existing code follows the \"Data structures and interfaces\" as per the provided class diagram.\n4. All functions are implemented, but the `is_game_over` method is missing in the Game class. To achieve this, we need to implement the `is_game_over` method to handle the game over condition.\n5. All necessary pre-dependencies have been imported.\n6. Methods from other files are not being reused in the current code.\n\n## Actions\n1. Implement the `is_game_over` method in the Game class to handle the game over condition.\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Tasks\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"'game.py' contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n----- game.py\n## game.py\n\nclass Game:\n def __init__(self):\n self.snake = []\n self.food = (0, 0)\n self.score = 0\n\n def update(self):\n # Update game state\n pass\n\n def draw(self):\n # Draw game\n pass\n\n def handle_input(self):\n # Handle user input\n pass\n\n def restart(self):\n # Restart game\n pass\n\n def handle_game_over(self):\n # Handle game over condition\n pass\n\n def update_score(self):\n # Update scoring system\n pass\n\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos] # Initialize body with start_pos\n self.direction = direction\n\n def move(self):\n # Move snake\n pass\n\n def change_direction(self, new_direction: str):\n # Change snake direction\n pass\n\n def grow(self):\n # Grow snake\n pass\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.handle_input()\n game.update()\n game.draw()\n\n if game.is_game_over():\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n## Code Review: main.py\n1. The code is implemented as per the requirements, but the `is_game_over` method is missing. To achieve this, we need to implement the `is_game_over` method in the Game class to check for the game over condition.\n2. The code logic is correct, but the missing `is_game_over` method needs to be added to handle the game over condition.\n3. The existing code follows the \"Data structures and interfaces\" as per the provided class diagram.\n4. All functions are implemented, but the `is_game_over` method is missing in the Game class. To achieve this, we need to implement the `is_game_over` method to handle the game over condition.\n5. All necessary pre-dependencies have been imported.\n6. Methods from other files are not being reused in the current code.\n\n## Actions\n1. Implement the `is_game_over` method in the Game class to handle the game over condition.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite main.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## main.py\n...\n```\n": "```python\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.handle_input()\n game.update()\n game.draw()\n\n if game.is_game_over(): # Check for game over condition\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n```", + "You are a python code to Mermaid Sequence Diagram translator in function detail#SYSTEM_MSG_END#```python\n#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n\"\"\"\n@Author : alexanderwu\n@File : write_review.py\n\"\"\"\nimport asyncio\nfrom typing import List, Literal\n\nfrom metagpt.actions import Action\nfrom metagpt.actions.action_node import ActionNode\n\nREVIEW = ActionNode(\n key=\"Review\",\n expected_type=List[str],\n instruction=\"Act as an experienced reviewer and critically assess the given output. Provide specific and\"\n \" constructive feedback, highlighting areas for improvement and suggesting changes.\",\n example=[\n \"The logic in the function `calculate_total` seems flawed. Shouldn't it consider the discount rate as well?\",\n \"The TODO function is not implemented yet? Should we implement it before commit?\",\n ],\n)\n\nREVIEW_RESULT = ActionNode(\n key=\"ReviewResult\",\n expected_type=Literal[\"LGTM\", \"LBTM\"],\n instruction=\"LGTM/LBTM. If the code is fully implemented, \" \"give a LGTM, otherwise provide a LBTM.\",\n example=\"LBTM\",\n)\n\nNEXT_STEPS = ActionNode(\n key=\"NextSteps\",\n expected_type=str,\n instruction=\"Based on the code review outcome, suggest actionable steps. This can include code changes, \"\n \"refactoring suggestions, or any follow-up tasks.\",\n example=\"\"\"1. Refactor the `process_data` method to improve readability and efficiency.\n2. Cover edge cases in the `validate_user` function.\n3. Implement a the TODO in the `calculate_total` function.\n4. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n\"\"\",\n)\n\nWRITE_DRAFT = ActionNode(\n key=\"WriteDraft\",\n expected_type=str,\n instruction=\"Could you write draft code for move function in order to implement it?\",\n example=\"Draft: ...\",\n)\n\n\nWRITE_FUNCTION = ActionNode(\n key=\"WriteFunction\",\n expected_type=str,\n instruction=\"write code for the function not implemented.\",\n example=\"\"\"\n```Code\n...\n```\n\"\"\",\n)\n\n\nREWRITE_CODE = ActionNode(\n key=\"RewriteCode\",\n expected_type=str,\n instruction=\"\"\"rewrite code based on the Review and Actions\"\"\",\n example=\"\"\"\n```python\n## example.py\ndef calculate_total(price, quantity):\n total = price * quantity\n```\n\"\"\",\n)\n\n\nCODE_REVIEW_CONTEXT = \"\"\"\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\n\n# Context\n## System Design\n{\"Implementation approach\": \"我们将使用HTML、CSS和JavaScript来实现这个单机的响应式2048游戏。为了确保游戏性能流畅和响应式设计,我们会选择使用Vue.js框架,因为它易于上手且适合构建交互式界面。我们还将使用localStorage来记录玩家的最高分。\", \"File list\": [\"index.html\", \"styles.css\", \"main.js\", \"game.js\", \"storage.js\"], \"Data structures and interfaces\": \"classDiagram\\\n class Game {\\\n -board Array\\\n -score Number\\\n -bestScore Number\\\n +constructor()\\\n +startGame()\\\n +move(direction: String)\\\n +getBoard() Array\\\n +getScore() Number\\\n +getBestScore() Number\\\n +setBestScore(score: Number)\\\n }\\\n class Storage {\\\n +getBestScore() Number\\\n +setBestScore(score: Number)\\\n }\\\n class Main {\\\n +init()\\\n +bindEvents()\\\n }\\\n Game --> Storage : uses\\\n Main --> Game : uses\", \"Program call flow\": \"sequenceDiagram\\\n participant M as Main\\\n participant G as Game\\\n participant S as Storage\\\n M->>G: init()\\\n G->>S: getBestScore()\\\n S-->>G: return bestScore\\\n M->>G: bindEvents()\\\n M->>G: startGame()\\\n loop Game Loop\\\n M->>G: move(direction)\\\n G->>S: setBestScore(score)\\\n S-->>G: return\\\n end\", \"Anything UNCLEAR\": \"目前项目要求明确,没有不清楚的地方。\"}\n\n## Tasks\n{\"Required Python packages\": [\"无需Python包\"], \"Required Other language third-party packages\": [\"vue.js\"], \"Logic Analysis\": [[\"index.html\", \"作为游戏的入口文件和主要的HTML结构\"], [\"styles.css\", \"包含所有的CSS样式,确保游戏界面美观\"], [\"main.js\", \"包含Main类,负责初始化游戏和绑定事件\"], [\"game.js\", \"包含Game类,负责游戏逻辑,如开始游戏、移动方块等\"], [\"storage.js\", \"包含Storage类,用于获取和设置玩家的最高分\"]], \"Task list\": [\"index.html\", \"styles.css\", \"storage.js\", \"game.js\", \"main.js\"], \"Full API spec\": \"\", \"Shared Knowledge\": \"\\'game.js\\' 包含游戏逻辑相关的函数,被 \\'main.js\\' 调用。\", \"Anything UNCLEAR\": \"目前项目要求明确,没有不清楚的地方。\"}\n\n## Code Files\n----- index.html\n\n\n\n \n \n 2048游戏\n \n \n\n\n
\n

2048

\n
\n
\n
分数
\n
{{ score }}
\n
\n
\n
最高分
\n
{{ bestScore }}
\n
\n
\n
\n
\n
\n {{ cell !== 0 ? cell : \\'\\' }}\n
\n
\n
\n \n
\n\n \n \n \n \n\n\n\n----- styles.css\n/* styles.css */\nbody, html {\n margin: 0;\n padding: 0;\n font-family: \\'Arial\\', sans-serif;\n}\n\n#app {\n text-align: center;\n font-size: 18px;\n color: #776e65;\n}\n\nh1 {\n color: #776e65;\n font-size: 72px;\n font-weight: bold;\n margin: 20px 0;\n}\n\n.scores-container {\n display: flex;\n justify-content: center;\n margin-bottom: 20px;\n}\n\n.score-container, .best-container {\n background: #bbada0;\n padding: 10px;\n border-radius: 5px;\n margin: 0 10px;\n min-width: 100px;\n text-align: center;\n}\n\n.score-header, .best-header {\n color: #eee4da;\n font-size: 18px;\n margin-bottom: 5px;\n}\n\n.game-container {\n max-width: 500px;\n margin: 0 auto 20px;\n background: #bbada0;\n padding: 15px;\n border-radius: 10px;\n position: relative;\n}\n\n.grid-row {\n display: flex;\n}\n\n.grid-cell {\n background: #cdc1b4;\n width: 100px;\n height: 100px;\n margin: 5px;\n display: flex;\n justify-content: center;\n align-items: center;\n font-size: 35px;\n font-weight: bold;\n color: #776e65;\n border-radius: 3px;\n}\n\n/* Dynamic classes for different number cells */\n.number-cell-2 {\n background: #eee4da;\n}\n\n.number-cell-4 {\n background: #ede0c8;\n}\n\n.number-cell-8 {\n background: #f2b179;\n color: #f9f6f2;\n}\n\n.number-cell-16 {\n background: #f59563;\n color: #f9f6f2;\n}\n\n.number-cell-32 {\n background: #f67c5f;\n color: #f9f6f2;\n}\n\n.number-cell-64 {\n background: #f65e3b;\n color: #f9f6f2;\n}\n\n.number-cell-128 {\n background: #edcf72;\n color: #f9f6f2;\n}\n\n.number-cell-256 {\n background: #edcc61;\n color: #f9f6f2;\n}\n\n.number-cell-512 {\n background: #edc850;\n color: #f9f6f2;\n}\n\n.number-cell-1024 {\n background: #edc53f;\n color: #f9f6f2;\n}\n\n.number-cell-2048 {\n background: #edc22e;\n color: #f9f6f2;\n}\n\n/* Larger numbers need smaller font sizes */\n.number-cell-1024, .number-cell-2048 {\n font-size: 30px;\n}\n\nbutton {\n background-color: #8f7a66;\n color: #f9f6f2;\n border: none;\n border-radius: 3px;\n padding: 10px 20px;\n font-size: 18px;\n cursor: pointer;\n outline: none;\n}\n\nbutton:hover {\n background-color: #9f8b76;\n}\n\n----- storage.js\n## storage.js\nclass Storage {\n // 获取最高分\n getBestScore() {\n // 尝试从localStorage中获取最高分,如果不存在则默认为0\n const bestScore = localStorage.getItem(\\'bestScore\\');\n return bestScore ? Number(bestScore) : 0;\n }\n\n // 设置最高分\n setBestScore(score) {\n // 将最高分设置到localStorage中\n localStorage.setItem(\\'bestScore\\', score.toString());\n }\n}\n\n\n\n## Code to be Reviewed: game.js\n```Code\n## game.js\nclass Game {\n constructor() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = 0;\n }\n\n createEmptyBoard() {\n const board = [];\n for (let i = 0; i < 4; i++) {\n board[i] = [0, 0, 0, 0];\n }\n return board;\n }\n\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.addRandomTile();\n this.addRandomTile();\n }\n\n addRandomTile() {\n let emptyCells = [];\n for (let r = 0; r < 4; r++) {\n for (let c = 0; c < 4; c++) {\n if (this.board[r][c] === 0) {\n emptyCells.push({ r, c });\n }\n }\n }\n if (emptyCells.length > 0) {\n let randomCell = emptyCells[Math.floor(Math.random() * emptyCells.length)];\n this.board[randomCell.r][randomCell.c] = Math.random() < 0.9 ? 2 : 4;\n }\n }\n\n move(direction) {\n // This function will handle the logic for moving tiles\n // in the specified direction and merging them\n // It will also update the score and add a new random tile if the move is successful\n // The actual implementation of this function is complex and would require\n // a significant amount of code to handle all the cases for moving and merging tiles\n // For the purposes of this example, we will not implement the full logic\n // Instead, we will just call addRandomTile to simulate a move\n this.addRandomTile();\n }\n\n getBoard() {\n return this.board;\n }\n\n getScore() {\n return this.score;\n }\n\n getBestScore() {\n return this.bestScore;\n }\n\n setBestScore(score) {\n this.bestScore = score;\n }\n}\n\n```\n\"\"\"\n\n\nCODE_REVIEW_SMALLEST_CONTEXT = \"\"\"\n## Code to be Reviewed: game.js\n```Code\n// game.js\nclass Game {\n constructor() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = 0;\n }\n\n createEmptyBoard() {\n const board = [];\n for (let i = 0; i < 4; i++) {\n board[i] = [0, 0, 0, 0];\n }\n return board;\n }\n\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.addRandomTile();\n this.addRandomTile();\n }\n\n addRandomTile() {\n let emptyCells = [];\n for (let r = 0; r < 4; r++) {\n for (let c = 0; c < 4; c++) {\n if (this.board[r][c] === 0) {\n emptyCells.push({ r, c });\n }\n }\n }\n if (emptyCells.length > 0) {\n let randomCell = emptyCells[Math.floor(Math.random() * emptyCells.length)];\n this.board[randomCell.r][randomCell.c] = Math.random() < 0.9 ? 2 : 4;\n }\n }\n\n move(direction) {\n // This function will handle the logic for moving tiles\n // in the specified direction and merging them\n // It will also update the score and add a new random tile if the move is successful\n // The actual implementation of this function is complex and would require\n // a significant amount of code to handle all the cases for moving and merging tiles\n // For the purposes of this example, we will not implement the full logic\n // Instead, we will just call addRandomTile to simulate a move\n this.addRandomTile();\n }\n\n getBoard() {\n return this.board;\n }\n\n getScore() {\n return this.score;\n }\n\n getBestScore() {\n return this.bestScore;\n }\n\n setBestScore(score) {\n this.bestScore = score;\n }\n}\n\n```\n\"\"\"\n\n\nCODE_REVIEW_SAMPLE = \"\"\"\n## Code Review: game.js\n1. The code partially implements the requirements. The `Game` class is missing the full implementation of the `move` method, which is crucial for the game\\'s functionality.\n2. The code logic is not completely correct. The `move` method is not implemented, which means the game cannot process player moves.\n3. The existing code follows the \"Data structures and interfaces\" in terms of class structure but lacks full method implementations.\n4. Not all functions are implemented. The `move` method is incomplete and does not handle the logic for moving and merging tiles.\n5. All necessary pre-dependencies seem to be imported since the code does not indicate the need for additional imports.\n6. The methods from other files (such as `Storage`) are not being used in the provided code snippet, but the class structure suggests that they will be used correctly.\n\n## Actions\n1. Implement the `move` method to handle tile movements and merging. This is a complex task that requires careful consideration of the game\\'s rules and logic. Here is a simplified version of how one might begin to implement the `move` method:\n ```javascript\n move(direction) {\n // Simplified logic for moving tiles up\n if (direction === \\'up\\') {\n for (let col = 0; col < 4; col++) {\n let tiles = this.board.map(row => row[col]).filter(val => val !== 0);\n let merged = [];\n for (let i = 0; i < tiles.length; i++) {\n if (tiles[i] === tiles[i + 1]) {\n tiles[i] *= 2;\n this.score += tiles[i];\n tiles[i + 1] = 0;\n merged.push(i);\n }\n }\n tiles = tiles.filter(val => val !== 0);\n while (tiles.length < 4) {\n tiles.push(0);\n }\n for (let row = 0; row < 4; row++) {\n this.board[row][col] = tiles[row];\n }\n }\n }\n // Additional logic needed for \\'down\\', \\'left\\', \\'right\\'\n // ...\n this.addRandomTile();\n }\n ```\n2. Integrate the `Storage` class methods to handle the best score. This means updating the `startGame` and `setBestScore` methods to use `Storage` for retrieving and setting the best score:\n ```javascript\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = new Storage().getBestScore(); // Retrieve the best score from storage\n this.addRandomTile();\n this.addRandomTile();\n }\n\n setBestScore(score) {\n if (score > this.bestScore) {\n this.bestScore = score;\n new Storage().setBestScore(score); // Set the new best score in storage\n }\n }\n ```\n\n## Code Review Result\nLBTM\n\n```\n\"\"\"\n\n\nWRITE_CODE_NODE = ActionNode.from_children(\"WRITE_REVIEW_NODE\", [REVIEW, REVIEW_RESULT, NEXT_STEPS])\nWRITE_MOVE_NODE = ActionNode.from_children(\"WRITE_MOVE_NODE\", [WRITE_DRAFT, WRITE_FUNCTION])\n\n\nCR_FOR_MOVE_FUNCTION_BY_3 = \"\"\"\nThe move function implementation provided appears to be well-structured and follows a clear logic for moving and merging tiles in the specified direction. However, there are a few potential improvements that could be made to enhance the code:\n\n1. Encapsulation: The logic for moving and merging tiles could be encapsulated into smaller, reusable functions to improve readability and maintainability.\n\n2. Magic Numbers: There are some magic numbers (e.g., 4, 3) used in the loops that could be replaced with named constants for improved readability and easier maintenance.\n\n3. Comments: Adding comments to explain the logic and purpose of each section of the code can improve understanding for future developers who may need to work on or maintain the code.\n\n4. Error Handling: It's important to consider error handling for unexpected input or edge cases to ensure the function behaves as expected in all scenarios.\n\nOverall, the code could benefit from refactoring to improve readability, maintainability, and extensibility. If you would like, I can provide a refactored version of the move function that addresses these considerations.\n\"\"\"\n\n\nclass WriteCodeAN(Action):\n \"\"\"Write a code review for the context.\"\"\"\n\n async def run(self, context):\n self.llm.system_prompt = \"You are an outstanding engineer and can implement any code\"\n return await WRITE_MOVE_NODE.fill(context=context, llm=self.llm, schema=\"json\")\n\n\nasync def main():\n await WriteCodeAN().run(CODE_REVIEW_SMALLEST_CONTEXT)\n\n\nif __name__ == \"__main__\":\n asyncio.run(main())\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant User\n participant WriteCodeAN\n participant WRITE_MOVE_NODE\n\n User->>WriteCodeAN: Run\n WriteCodeAN->>WRITE_MOVE_NODE: Fill\n WRITE_MOVE_NODE->>WriteCodeAN: Filled\n WriteCodeAN->>User: Return Filled Context\n```", + "You are a function parser.#MSG_SEP#You can convert spoken words into function parameters.#SYSTEM_MSG_END#text_to_image function parameters description:\nparameter `text`: The text used for image conversion.\nparameter `size_type`: size type\n\n---\nExamples:\nIf want you to do `Draw a girl`, return `text_to_image(text=\"Draw a girl\", size_type=\"512x512\")` brief and clear.\nIf want you to do `Draw an apple`, return `text_to_image(text=\"Draw an apple\", size_type=\"512x512\")` brief and clear.\n\n---\n\nRefer to the `text_to_image` function description, and fill in the function parameters according to the example \"I want you to do xx\" in the Examples section.\nNow I want you to do `Draw an apple`, return function parameters in Examples format above, brief and clear.": "`text_to_image(text=\"Draw an apple\", size_type=\"512x512\")`", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\": \"We will use the Pygame library to create the game interface and handle user input. The game logic will be implemented using Python classes and data structures.\", \"File list\": [\"main.py\", \"game.py\"], \"Data structures and interfaces\": \"classDiagram\\n class Game {\\n -grid: List[List[int]]\\n -score: int\\n -game_over: bool\\n +__init__()\\n +reset_game()\\n +move(direction: str)\\n +is_game_over() bool\\n +get_empty_cells() List[Tuple[int, int]]\\n +add_new_tile()\\n +get_score() int\\n }\\n class UI {\\n -game: Game\\n +__init__(game: Game)\\n +draw_grid()\\n +draw_score()\\n +draw_game_over()\\n +handle_input()\\n }\\n Game --> UI\", \"Program call flow\": \"sequenceDiagram\\n participant M as Main\\n participant G as Game\\n participant U as UI\\n M->>G: reset_game()\\n M->>U: draw_grid()\\n M->>U: draw_score()\\n M->>U: handle_input()\\n U->>G: move(direction)\\n G->>G: add_new_tile()\\n G->>U: draw_grid()\\n G->>U: draw_score()\\n G->>U: draw_game_over()\\n G->>G: is_game_over()\\n G->>G: get_empty_cells()\\n G->>G: get_score()\", \"Anything UNCLEAR\": \"...\"}\n\n## Tasks\n{\"Required Python packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party dependencies required\"], \"Logic Analysis\": [[\"game.py\", \"Contains Game class and related functions for game logic\"], [\"main.py\", \"Contains main function, initializes the game and UI\"]], \"Task list\": [\"game.py\", \"main.py\"], \"Full API spec\": \"\", \"Shared Knowledge\": \"The game logic will be implemented using Python classes and data structures. The Pygame library will be used to create the game interface and handle user input.\", \"Anything UNCLEAR\": \"...\"}\n\n## Legacy Code\n```Code\n----- main.py\nif __name__ == \"__main__\":\nmain()\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## game.py\n\nfrom typing import List, Tuple\nimport random\n\nclass Game:\n def __init__(self):\n self.grid: List[List[int]] = [[0 for _ in range(4)] for _ in range(4)]\n self.score: int = 0\n self.game_over: bool = False\n\n def reset_game(self):\n self.grid = [[0 for _ in range(4)] for _ in range(4)]\n self.score = 0\n self.game_over = False\n self.add_new_tile()\n self.add_new_tile()\n\n def move(self, direction: str):\n # Implement the logic to move the tiles in the specified direction\n pass\n\n def is_game_over(self) -> bool:\n # Implement the logic to check if the game is over\n pass\n\n def get_empty_cells(self) -> List[Tuple[int, int]]:\n # Implement the logic to get the coordinates of empty cells in the grid\n pass\n\n def add_new_tile(self):\n empty_cells = self.get_empty_cells()\n if empty_cells:\n row, col = random.choice(empty_cells)\n self.grid[row][col] = 2 if random.random() < 0.9 else 4\n\n def get_score(self) -> int:\n return self.score\n```", + "\n## context\n\n### Project Name\n\n\n### Original Requirements\n开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"zh_cn\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结\",\n \"Project Name\": \"language_model_search\",\n \"Product Goals\": [\n \"提供基于大语言模型的高效搜索功能\",\n \"整合私有知识库,实现个性化搜索结果\",\n \"实现搜索结果的自动总结功能\"\n ],\n \"User Stories\": [\n \"作为用户,我希望能够快速找到我需要的信息\",\n \"作为用户,我希望搜索结果能够根据我的偏好进行个性化排序\",\n \"作为用户,我希望搜索结果能够自动总结,方便我快速了解内容\"\n ],\n \"Competitive Analysis\": [\n \"搜索引擎A: 提供基于大语言模型的搜索功能,但个性化程度较低\",\n \"知识库B: 整合私有知识库,但搜索速度较慢\",\n \"语言模型搜索C: 提供搜索结果自动总结功能,但搜索准确度有待提高\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"搜索引擎竞争分析\\\"\\n x-axis \\\"低搜索准确度\\\" --> \\\"高搜索准确度\\\"\\n y-axis \\\"低个性化程度\\\" --> \\\"高个性化程度\\\"\\n quadrant-1 \\\"需改进\\\"\\n quadrant-2 \\\"需提升\\\"\\n quadrant-3 \\\"重新评估\\\"\\n quadrant-4 \\\"扩展发展\\\"\\n \\\"搜索引擎A\\\": [0.6, 0.3]\\n \\\"知识库B\\\": [0.4, 0.2]\\n \\\"语言模型搜索C\\\": [0.7, 0.5]\\n \\\"我们的目标产品\\\": [0.8, 0.7]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"基于大语言模型的高效搜索功能\"\n ],\n [\n \"P1\",\n \"整合私有知识库,实现个性化搜索结果\"\n ],\n [\n \"P2\",\n \"实现搜索结果的自动总结功能\"\n ]\n ],\n \"UI Design draft\": \"搜索页面简洁明了,搜索结果清晰展示,提供个性化排序和自动总结功能。\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]" } \ No newline at end of file diff --git a/tests/metagpt/learn/test_text_to_embedding.py b/tests/metagpt/learn/test_text_to_embedding.py index 8891960c1..280951ffa 100644 --- a/tests/metagpt/learn/test_text_to_embedding.py +++ b/tests/metagpt/learn/test_text_to_embedding.py @@ -11,7 +11,7 @@ from pathlib import Path import pytest -from metagpt.config2 import config +from metagpt.config2 import Config from metagpt.learn.text_to_embedding import text_to_embedding from metagpt.utils.common import aread @@ -19,6 +19,7 @@ from metagpt.utils.common import aread @pytest.mark.asyncio async def test_text_to_embedding(mocker): # mock + config = Config.default() mock_post = mocker.patch("aiohttp.ClientSession.post") mock_response = mocker.AsyncMock() mock_response.status = 200 diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index bf19a77b8..2d52ad10e 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -57,7 +57,8 @@ class TestOpenAI: def test_make_client_kwargs_without_proxy(self): instance = OpenAILLM(mock_llm_config) kwargs = instance._make_client_kwargs() - assert kwargs == {"api_key": "mock_api_key", "base_url": "mock_base_url"} + assert kwargs["api_key"] == "mock_api_key" + assert kwargs["base_url"] == "mock_base_url" assert "http_client" not in kwargs def test_make_client_kwargs_with_proxy(self): diff --git a/tests/metagpt/tools/test_openai_text_to_embedding.py b/tests/metagpt/tools/test_openai_text_to_embedding.py index 047206d48..81b3895c3 100644 --- a/tests/metagpt/tools/test_openai_text_to_embedding.py +++ b/tests/metagpt/tools/test_openai_text_to_embedding.py @@ -10,7 +10,7 @@ from pathlib import Path import pytest -from metagpt.config2 import config +from metagpt.config2 import Config from metagpt.tools.openai_text_to_embedding import oas3_openai_text_to_embedding from metagpt.utils.common import aread @@ -18,6 +18,7 @@ from metagpt.utils.common import aread @pytest.mark.asyncio async def test_embedding(mocker): # mock + config = Config.default() mock_post = mocker.patch("aiohttp.ClientSession.post") mock_response = mocker.AsyncMock() mock_response.status = 200 From f1cfeb234e93aabad43c2af4377284f45438c323 Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 15 Jan 2024 18:02:57 +0800 Subject: [PATCH 339/637] fix bugs --- examples/example.pkl | Bin 624 -> 624 bytes tests/metagpt/learn/test_text_to_embedding.py | 2 +- 2 files changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/example.pkl b/examples/example.pkl index 7c6ab901b210830f4436202b85bee6087e92b82c..eecd3ec982ad1828ddc0465d3c7417eb57ed6f1b 100644 GIT binary patch delta 88 zcmV~$Q4NGZ3DJOQEu6Y_*uVk_2C#q=q#yy_^}W+O9cyGang=l?YlRdZ nu&RX0)*Eff)a>37;=E;VNgb(ohb$s6Zj6yNm0aY1I-hWTYC0Jj diff --git a/tests/metagpt/learn/test_text_to_embedding.py b/tests/metagpt/learn/test_text_to_embedding.py index 280951ffa..f50f6a7aa 100644 --- a/tests/metagpt/learn/test_text_to_embedding.py +++ b/tests/metagpt/learn/test_text_to_embedding.py @@ -26,7 +26,7 @@ async def test_text_to_embedding(mocker): data = await aread(Path(__file__).parent / "../../data/openai/embedding.json") mock_response.json.return_value = json.loads(data) mock_post.return_value.__aenter__.return_value = mock_response - type(config.get_openai_llm()).proxy = mocker.PropertyMock(return_value="http://mock.proxy") + config.get_openai_llm().proxy = mocker.PropertyMock(return_value="http://mock.proxy") # Prerequisites assert config.get_openai_llm().api_key From 00f7f93234d0c19286aca3d16233367be2d5fd2a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 18:09:56 +0800 Subject: [PATCH 340/637] add scrape_web. --- metagpt/tools/__init__.py | 6 ++++++ .../tools/functions/schemas/scrape_web.yml | 21 +++++++++++++++++++ 2 files changed, 27 insertions(+) create mode 100644 metagpt/tools/functions/schemas/scrape_web.yml diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index 41c8708b2..c24dc6fce 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -76,6 +76,12 @@ TOOL_TYPE_MAPPINGS = { desc="Related to text2image, image2image using stable diffusion model.", usage_prompt="", ), + "scrape_web": ToolType( + name="scrape_web", + module="metagpt.tools.scrape_web", + desc="Scrape data from web page.", + usage_prompt="", + ), "other": ToolType( name="other", module="", diff --git a/metagpt/tools/functions/schemas/scrape_web.yml b/metagpt/tools/functions/schemas/scrape_web.yml new file mode 100644 index 000000000..ecca3fbed --- /dev/null +++ b/metagpt/tools/functions/schemas/scrape_web.yml @@ -0,0 +1,21 @@ +scrape_web: + type: async funciton + description: "Scrape and save the HTML structure and inner text content of a web page using Playwright." + parameters: + properties: + url: + type: str + description: "web url" + \*url: + type: Non-Keyword Arguments + description: "other web urls, you can assagin sub url link to it." + required: + - url + returns: + inner_text: + type: str + description: The inner text content of the web page. + html: + type: str + description: The html structure of the web page. + From 75628caf4d68b7519c63a84c0203326ea05ace5c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 18:10:57 +0800 Subject: [PATCH 341/637] add scrape_web.py --- .../functions/libs/scrape_web/__init__.py | 1 + .../functions/libs/scrape_web/scrape_web.py | 26 +++++++++++++++++++ 2 files changed, 27 insertions(+) create mode 100644 metagpt/tools/functions/libs/scrape_web/__init__.py create mode 100644 metagpt/tools/functions/libs/scrape_web/scrape_web.py diff --git a/metagpt/tools/functions/libs/scrape_web/__init__.py b/metagpt/tools/functions/libs/scrape_web/__init__.py new file mode 100644 index 000000000..d5cd1524b --- /dev/null +++ b/metagpt/tools/functions/libs/scrape_web/__init__.py @@ -0,0 +1 @@ +from metagpt.tools.functions.libs.scrape_web.scrape_web import scrape_web diff --git a/metagpt/tools/functions/libs/scrape_web/scrape_web.py b/metagpt/tools/functions/libs/scrape_web/scrape_web.py new file mode 100644 index 000000000..5cd984f4d --- /dev/null +++ b/metagpt/tools/functions/libs/scrape_web/scrape_web.py @@ -0,0 +1,26 @@ +import asyncio + +from metagpt.tools.web_browser_engine_playwright import PlaywrightWrapper + + +async def scrape_web(url, *urls): + """ + Scrape and save the HTML structure and inner text content of a web page using Playwright. + + Args: + url (str): The main URL to fetch inner text from. + *urls (str): Additional URLs to fetch inner text from. + + Returns: + (dict): The inner text content and html structure of the web page, key are : 'inner_text', 'html'. + + Raises: + Any exceptions that may occur during the Playwright operation. + """ + # Create a PlaywrightWrapper instance for the Chromium browser + web = await PlaywrightWrapper("chromium").run(url, *urls) + + # Return the inner text content of the web page + return {"inner_text": web.inner_text, "html": web.html} + +# 需要改三个地方: yaml, 对应路径下init, MetaGPT/metagpt/prompts/ml_engineer.py中ML_MODULE_MAP From a7e088845e508464281daed1301ce32c0acc0797 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 18:11:22 +0800 Subject: [PATCH 342/637] update scrape_web docstring. --- metagpt/tools/functions/libs/scrape_web/scrape_web.py | 3 --- 1 file changed, 3 deletions(-) diff --git a/metagpt/tools/functions/libs/scrape_web/scrape_web.py b/metagpt/tools/functions/libs/scrape_web/scrape_web.py index 5cd984f4d..e68ce0e64 100644 --- a/metagpt/tools/functions/libs/scrape_web/scrape_web.py +++ b/metagpt/tools/functions/libs/scrape_web/scrape_web.py @@ -13,9 +13,6 @@ async def scrape_web(url, *urls): Returns: (dict): The inner text content and html structure of the web page, key are : 'inner_text', 'html'. - - Raises: - Any exceptions that may occur during the Playwright operation. """ # Create a PlaywrightWrapper instance for the Chromium browser web = await PlaywrightWrapper("chromium").run(url, *urls) From 66db86ae2a66ebd532bbdc67f03a89c8a638cfee Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 15 Jan 2024 18:19:57 +0800 Subject: [PATCH 343/637] update test_vision.py for mock --- .../tools/functions/libs/test_vision.py | 48 +++++++++++++++++++ tests/metagpt/tools/functions/test_vision.py | 40 ---------------- 2 files changed, 48 insertions(+), 40 deletions(-) create mode 100644 tests/metagpt/tools/functions/libs/test_vision.py delete mode 100644 tests/metagpt/tools/functions/test_vision.py diff --git a/tests/metagpt/tools/functions/libs/test_vision.py b/tests/metagpt/tools/functions/libs/test_vision.py new file mode 100644 index 000000000..f4f97c46a --- /dev/null +++ b/tests/metagpt/tools/functions/libs/test_vision.py @@ -0,0 +1,48 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/15 +@Author : mannaandpoem +@File : test_vision.py +""" +import pytest + +from metagpt import logs +from metagpt.tools.functions.libs.vision import Vision + + +@pytest.fixture +def mock_webpages(): + return """```html\n\n +\n\n```\n +```css\n.class { ... }\n```\n +```javascript\nfunction() { ... }\n```\n""" + + +def test_vision_generate_webpages(mocker, mock_webpages): + mocker.patch( + "metagpt.tools.functions.libs.vision.Vision.generate_web_pages", + return_value=mock_webpages + ) + image_path = "image.png" + vision = Vision() + rsp = vision.generate_web_pages(image_path=image_path) + logs.logger.info(rsp) + assert "html" in rsp + assert "css" in rsp + assert "javascript" in rsp + + +def test_save_webpages(mocker, mock_webpages): + mocker.patch( + "metagpt.tools.functions.libs.vision.Vision.generate_web_pages", + return_value=mock_webpages + ) + image_path = "image.png" + vision = Vision() + webpages = vision.generate_web_pages(image_path) + webpages_dir = vision.save_webpages(image_path=image_path, webpages=webpages) + logs.logger.info(webpages_dir) + assert webpages_dir.exists() + + diff --git a/tests/metagpt/tools/functions/test_vision.py b/tests/metagpt/tools/functions/test_vision.py deleted file mode 100644 index 0359f14f1..000000000 --- a/tests/metagpt/tools/functions/test_vision.py +++ /dev/null @@ -1,40 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -""" -@Time : 2024/01/15 -@Author : mannaandpoem -@File : test_vision.py -""" -import base64 -from unittest.mock import AsyncMock - -from pytest_mock import mocker - -from metagpt import logs -from metagpt.tools.functions.libs.vision import Vision - - -def test_vision_generate_web_pages(): - image_path = "./image.png" - vision = Vision() - rsp = vision.generate_web_pages(image_path=image_path) - logs.logger.info(rsp) - assert "html" in rsp - assert "css" in rsp - assert "javascript" in rsp - - -def test_save_webpages(): - image_path = "./image.png" - vision = Vision() - webpages = """```html: \n - \n``` - "```css: .class { ... } ```\n ```javascript: function() { ... }```""" - webpages_dir = vision.save_webpages(image_path=image_path, webpages=webpages) - logs.logger.info(webpages_dir) - assert webpages_dir.exists() - assert (webpages_dir / "index.html").exists() - assert (webpages_dir / "style.css").exists() or (webpages_dir / "styles.css").exists() - assert (webpages_dir / "script.js").exists() or (webpages_dir / "scripts.js").exists() - - From 559a1604ad1e273d9de50fd466b1d0ac2a045d56 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 18:26:32 +0800 Subject: [PATCH 344/637] restore. --- metagpt/provider/base_llm.py | 48 +----------------------------------- 1 file changed, 1 insertion(+), 47 deletions(-) diff --git a/metagpt/provider/base_llm.py b/metagpt/provider/base_llm.py index c482aaf35..dbef15fa1 100644 --- a/metagpt/provider/base_llm.py +++ b/metagpt/provider/base_llm.py @@ -6,14 +6,10 @@ @File : base_llm.py @Desc : mashenquan, 2023/8/22. + try catch """ -import re import json from abc import ABC, abstractmethod from typing import Optional -from metagpt.logs import logger -from metagpt.utils.common import CodeParser - class BaseLLM(ABC): """LLM API abstract class, requiring all inheritors to provide a series of standard capabilities""" @@ -122,30 +118,6 @@ class BaseLLM(ABC): """ return rsp.get("choices")[0]["message"]["tool_calls"][0]["function"] - def _parse_arguments(self, arguments: str) -> dict: - """parse arguments in openai function call""" - if 'langugae' not in arguments and 'code' not in arguments: - logger.warning(f"Not found `code`, `language`, We assume it is pure code:\n {arguments}\n. ") - return {'language': 'python', 'code': arguments} - - # 匹配language - language_pattern = re.compile(r'[\"\']?language[\"\']?\s*:\s*["\']([^"\']+?)["\']', re.DOTALL) - language_match = language_pattern.search(arguments) - language_value = language_match.group(1) if language_match else None - - # 匹配code - code_pattern = r'(["\']{3}|["])([\s\S]*?)\1' - try: - code_value = re.findall(code_pattern, arguments)[-1][-1] - except Exception as e: - logger.error(f"{e}, when re.findall({code_pattern}, {arguments})") - code_value = None - - if code_value is None: - raise ValueError(f"Parse code error for {arguments}") - # arguments只有code的情况 - return {'language': language_value, 'code': code_value} - def get_choice_function_arguments(self, rsp: dict) -> dict: """Required to provide the first function arguments of choice. @@ -153,25 +125,7 @@ class BaseLLM(ABC): :return dict: return the first function arguments of choice, for example, {'language': 'python', 'code': "print('Hello, World!')"} """ - try: - arguments: str = self.get_choice_function(rsp)["arguments"] - return json.loads(arguments, strict=False) - except json.decoder.JSONDecodeError as e: - logger.debug(f"Got JSONDecodeError for {arguments}, we will use RegExp to parse code, \n {e}") - return self._parse_arguments(arguments) - except KeyError as e: - if 'tool_calls' in e.args: - txt_rsp = self.get_choice_text(rsp) - # find code - code = CodeParser.parse_code(None, txt_rsp, lang='python') - if code != txt_rsp: - return {'language': 'python', 'code': code} - # no code - return {'language': 'markdown', 'code': txt_rsp} - raise e - except Exception as e: - logger.error(f"Got error `{e}` for parsing\n {rsp}\n") - return {} + return json.loads(self.get_choice_function(rsp)["arguments"], strict=False) def messages_to_prompt(self, messages: list[dict]): """[{"role": "user", "content": msg}] to user: etc.""" From 67021f24a08418aa5e6f022c729b6feb3cba7802 Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 15 Jan 2024 18:46:54 +0800 Subject: [PATCH 345/637] fix bugs --- examples/example.pkl | Bin 624 -> 624 bytes tests/data/rsp_cache.json | 12 +++++++++++- 2 files changed, 11 insertions(+), 1 deletion(-) diff --git a/examples/example.pkl b/examples/example.pkl index eecd3ec982ad1828ddc0465d3c7417eb57ed6f1b..f1912a97357ecce5f530b4f18fd7664d9b1a9450 100644 GIT binary patch delta 88 zcmWN{%ME}a3;@uOFbdZurF;srcxth%;R+@$;0kWyD2^cK_dfsma|=E0Lop$6lLA?) nfg~p2juFdht$SP=^U1``dx?XqxT-X^s delta 88 zcmV~$Q4NGZ3 Storage : uses\\\n Main --> Game : uses\", \"Program call flow\": \"sequenceDiagram\\\n participant M as Main\\\n participant G as Game\\\n participant S as Storage\\\n M->>G: init()\\\n G->>S: getBestScore()\\\n S-->>G: return bestScore\\\n M->>G: bindEvents()\\\n M->>G: startGame()\\\n loop Game Loop\\\n M->>G: move(direction)\\\n G->>S: setBestScore(score)\\\n S-->>G: return\\\n end\", \"Anything UNCLEAR\": \"目前项目要求明确,没有不清楚的地方。\"}\n\n## Tasks\n{\"Required Python packages\": [\"无需Python包\"], \"Required Other language third-party packages\": [\"vue.js\"], \"Logic Analysis\": [[\"index.html\", \"作为游戏的入口文件和主要的HTML结构\"], [\"styles.css\", \"包含所有的CSS样式,确保游戏界面美观\"], [\"main.js\", \"包含Main类,负责初始化游戏和绑定事件\"], [\"game.js\", \"包含Game类,负责游戏逻辑,如开始游戏、移动方块等\"], [\"storage.js\", \"包含Storage类,用于获取和设置玩家的最高分\"]], \"Task list\": [\"index.html\", \"styles.css\", \"storage.js\", \"game.js\", \"main.js\"], \"Full API spec\": \"\", \"Shared Knowledge\": \"\\'game.js\\' 包含游戏逻辑相关的函数,被 \\'main.js\\' 调用。\", \"Anything UNCLEAR\": \"目前项目要求明确,没有不清楚的地方。\"}\n\n## Code Files\n----- index.html\n\n\n\n \n \n 2048游戏\n \n \n\n\n
\n

2048

\n
\n
\n
分数
\n
{{ score }}
\n
\n
\n
最高分
\n
{{ bestScore }}
\n
\n
\n
\n
\n
\n {{ cell !== 0 ? cell : \\'\\' }}\n
\n
\n
\n \n
\n\n \n \n \n \n\n\n\n----- styles.css\n/* styles.css */\nbody, html {\n margin: 0;\n padding: 0;\n font-family: \\'Arial\\', sans-serif;\n}\n\n#app {\n text-align: center;\n font-size: 18px;\n color: #776e65;\n}\n\nh1 {\n color: #776e65;\n font-size: 72px;\n font-weight: bold;\n margin: 20px 0;\n}\n\n.scores-container {\n display: flex;\n justify-content: center;\n margin-bottom: 20px;\n}\n\n.score-container, .best-container {\n background: #bbada0;\n padding: 10px;\n border-radius: 5px;\n margin: 0 10px;\n min-width: 100px;\n text-align: center;\n}\n\n.score-header, .best-header {\n color: #eee4da;\n font-size: 18px;\n margin-bottom: 5px;\n}\n\n.game-container {\n max-width: 500px;\n margin: 0 auto 20px;\n background: #bbada0;\n padding: 15px;\n border-radius: 10px;\n position: relative;\n}\n\n.grid-row {\n display: flex;\n}\n\n.grid-cell {\n background: #cdc1b4;\n width: 100px;\n height: 100px;\n margin: 5px;\n display: flex;\n justify-content: center;\n align-items: center;\n font-size: 35px;\n font-weight: bold;\n color: #776e65;\n border-radius: 3px;\n}\n\n/* Dynamic classes for different number cells */\n.number-cell-2 {\n background: #eee4da;\n}\n\n.number-cell-4 {\n background: #ede0c8;\n}\n\n.number-cell-8 {\n background: #f2b179;\n color: #f9f6f2;\n}\n\n.number-cell-16 {\n background: #f59563;\n color: #f9f6f2;\n}\n\n.number-cell-32 {\n background: #f67c5f;\n color: #f9f6f2;\n}\n\n.number-cell-64 {\n background: #f65e3b;\n color: #f9f6f2;\n}\n\n.number-cell-128 {\n background: #edcf72;\n color: #f9f6f2;\n}\n\n.number-cell-256 {\n background: #edcc61;\n color: #f9f6f2;\n}\n\n.number-cell-512 {\n background: #edc850;\n color: #f9f6f2;\n}\n\n.number-cell-1024 {\n background: #edc53f;\n color: #f9f6f2;\n}\n\n.number-cell-2048 {\n background: #edc22e;\n color: #f9f6f2;\n}\n\n/* Larger numbers need smaller font sizes */\n.number-cell-1024, .number-cell-2048 {\n font-size: 30px;\n}\n\nbutton {\n background-color: #8f7a66;\n color: #f9f6f2;\n border: none;\n border-radius: 3px;\n padding: 10px 20px;\n font-size: 18px;\n cursor: pointer;\n outline: none;\n}\n\nbutton:hover {\n background-color: #9f8b76;\n}\n\n----- storage.js\n## storage.js\nclass Storage {\n // 获取最高分\n getBestScore() {\n // 尝试从localStorage中获取最高分,如果不存在则默认为0\n const bestScore = localStorage.getItem(\\'bestScore\\');\n return bestScore ? Number(bestScore) : 0;\n }\n\n // 设置最高分\n setBestScore(score) {\n // 将最高分设置到localStorage中\n localStorage.setItem(\\'bestScore\\', score.toString());\n }\n}\n\n\n\n## Code to be Reviewed: game.js\n```Code\n## game.js\nclass Game {\n constructor() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = 0;\n }\n\n createEmptyBoard() {\n const board = [];\n for (let i = 0; i < 4; i++) {\n board[i] = [0, 0, 0, 0];\n }\n return board;\n }\n\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.addRandomTile();\n this.addRandomTile();\n }\n\n addRandomTile() {\n let emptyCells = [];\n for (let r = 0; r < 4; r++) {\n for (let c = 0; c < 4; c++) {\n if (this.board[r][c] === 0) {\n emptyCells.push({ r, c });\n }\n }\n }\n if (emptyCells.length > 0) {\n let randomCell = emptyCells[Math.floor(Math.random() * emptyCells.length)];\n this.board[randomCell.r][randomCell.c] = Math.random() < 0.9 ? 2 : 4;\n }\n }\n\n move(direction) {\n // This function will handle the logic for moving tiles\n // in the specified direction and merging them\n // It will also update the score and add a new random tile if the move is successful\n // The actual implementation of this function is complex and would require\n // a significant amount of code to handle all the cases for moving and merging tiles\n // For the purposes of this example, we will not implement the full logic\n // Instead, we will just call addRandomTile to simulate a move\n this.addRandomTile();\n }\n\n getBoard() {\n return this.board;\n }\n\n getScore() {\n return this.score;\n }\n\n getBestScore() {\n return this.bestScore;\n }\n\n setBestScore(score) {\n this.bestScore = score;\n }\n}\n\n```\n\"\"\"\n\n\nCODE_REVIEW_SMALLEST_CONTEXT = \"\"\"\n## Code to be Reviewed: game.js\n```Code\n// game.js\nclass Game {\n constructor() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = 0;\n }\n\n createEmptyBoard() {\n const board = [];\n for (let i = 0; i < 4; i++) {\n board[i] = [0, 0, 0, 0];\n }\n return board;\n }\n\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.addRandomTile();\n this.addRandomTile();\n }\n\n addRandomTile() {\n let emptyCells = [];\n for (let r = 0; r < 4; r++) {\n for (let c = 0; c < 4; c++) {\n if (this.board[r][c] === 0) {\n emptyCells.push({ r, c });\n }\n }\n }\n if (emptyCells.length > 0) {\n let randomCell = emptyCells[Math.floor(Math.random() * emptyCells.length)];\n this.board[randomCell.r][randomCell.c] = Math.random() < 0.9 ? 2 : 4;\n }\n }\n\n move(direction) {\n // This function will handle the logic for moving tiles\n // in the specified direction and merging them\n // It will also update the score and add a new random tile if the move is successful\n // The actual implementation of this function is complex and would require\n // a significant amount of code to handle all the cases for moving and merging tiles\n // For the purposes of this example, we will not implement the full logic\n // Instead, we will just call addRandomTile to simulate a move\n this.addRandomTile();\n }\n\n getBoard() {\n return this.board;\n }\n\n getScore() {\n return this.score;\n }\n\n getBestScore() {\n return this.bestScore;\n }\n\n setBestScore(score) {\n this.bestScore = score;\n }\n}\n\n```\n\"\"\"\n\n\nCODE_REVIEW_SAMPLE = \"\"\"\n## Code Review: game.js\n1. The code partially implements the requirements. The `Game` class is missing the full implementation of the `move` method, which is crucial for the game\\'s functionality.\n2. The code logic is not completely correct. The `move` method is not implemented, which means the game cannot process player moves.\n3. The existing code follows the \"Data structures and interfaces\" in terms of class structure but lacks full method implementations.\n4. Not all functions are implemented. The `move` method is incomplete and does not handle the logic for moving and merging tiles.\n5. All necessary pre-dependencies seem to be imported since the code does not indicate the need for additional imports.\n6. The methods from other files (such as `Storage`) are not being used in the provided code snippet, but the class structure suggests that they will be used correctly.\n\n## Actions\n1. Implement the `move` method to handle tile movements and merging. This is a complex task that requires careful consideration of the game\\'s rules and logic. Here is a simplified version of how one might begin to implement the `move` method:\n ```javascript\n move(direction) {\n // Simplified logic for moving tiles up\n if (direction === \\'up\\') {\n for (let col = 0; col < 4; col++) {\n let tiles = this.board.map(row => row[col]).filter(val => val !== 0);\n let merged = [];\n for (let i = 0; i < tiles.length; i++) {\n if (tiles[i] === tiles[i + 1]) {\n tiles[i] *= 2;\n this.score += tiles[i];\n tiles[i + 1] = 0;\n merged.push(i);\n }\n }\n tiles = tiles.filter(val => val !== 0);\n while (tiles.length < 4) {\n tiles.push(0);\n }\n for (let row = 0; row < 4; row++) {\n this.board[row][col] = tiles[row];\n }\n }\n }\n // Additional logic needed for \\'down\\', \\'left\\', \\'right\\'\n // ...\n this.addRandomTile();\n }\n ```\n2. Integrate the `Storage` class methods to handle the best score. This means updating the `startGame` and `setBestScore` methods to use `Storage` for retrieving and setting the best score:\n ```javascript\n startGame() {\n this.board = this.createEmptyBoard();\n this.score = 0;\n this.bestScore = new Storage().getBestScore(); // Retrieve the best score from storage\n this.addRandomTile();\n this.addRandomTile();\n }\n\n setBestScore(score) {\n if (score > this.bestScore) {\n this.bestScore = score;\n new Storage().setBestScore(score); // Set the new best score in storage\n }\n }\n ```\n\n## Code Review Result\nLBTM\n\n```\n\"\"\"\n\n\nWRITE_CODE_NODE = ActionNode.from_children(\"WRITE_REVIEW_NODE\", [REVIEW, REVIEW_RESULT, NEXT_STEPS])\nWRITE_MOVE_NODE = ActionNode.from_children(\"WRITE_MOVE_NODE\", [WRITE_DRAFT, WRITE_FUNCTION])\n\n\nCR_FOR_MOVE_FUNCTION_BY_3 = \"\"\"\nThe move function implementation provided appears to be well-structured and follows a clear logic for moving and merging tiles in the specified direction. However, there are a few potential improvements that could be made to enhance the code:\n\n1. Encapsulation: The logic for moving and merging tiles could be encapsulated into smaller, reusable functions to improve readability and maintainability.\n\n2. Magic Numbers: There are some magic numbers (e.g., 4, 3) used in the loops that could be replaced with named constants for improved readability and easier maintenance.\n\n3. Comments: Adding comments to explain the logic and purpose of each section of the code can improve understanding for future developers who may need to work on or maintain the code.\n\n4. Error Handling: It's important to consider error handling for unexpected input or edge cases to ensure the function behaves as expected in all scenarios.\n\nOverall, the code could benefit from refactoring to improve readability, maintainability, and extensibility. If you would like, I can provide a refactored version of the move function that addresses these considerations.\n\"\"\"\n\n\nclass WriteCodeAN(Action):\n \"\"\"Write a code review for the context.\"\"\"\n\n async def run(self, context):\n self.llm.system_prompt = \"You are an outstanding engineer and can implement any code\"\n return await WRITE_MOVE_NODE.fill(context=context, llm=self.llm, schema=\"json\")\n\n\nasync def main():\n await WriteCodeAN().run(CODE_REVIEW_SMALLEST_CONTEXT)\n\n\nif __name__ == \"__main__\":\n asyncio.run(main())\n\n```\n\n---\nTranslate the code above into Mermaid Sequence Diagram.": "```mermaid\nsequenceDiagram\n participant User\n participant WriteCodeAN\n participant WRITE_MOVE_NODE\n\n User->>WriteCodeAN: Run\n WriteCodeAN->>WRITE_MOVE_NODE: Fill\n WRITE_MOVE_NODE->>WriteCodeAN: Filled\n WriteCodeAN->>User: Return Filled Context\n```", "You are a function parser.#MSG_SEP#You can convert spoken words into function parameters.#SYSTEM_MSG_END#text_to_image function parameters description:\nparameter `text`: The text used for image conversion.\nparameter `size_type`: size type\n\n---\nExamples:\nIf want you to do `Draw a girl`, return `text_to_image(text=\"Draw a girl\", size_type=\"512x512\")` brief and clear.\nIf want you to do `Draw an apple`, return `text_to_image(text=\"Draw an apple\", size_type=\"512x512\")` brief and clear.\n\n---\n\nRefer to the `text_to_image` function description, and fill in the function parameters according to the example \"I want you to do xx\" in the Examples section.\nNow I want you to do `Draw an apple`, return function parameters in Examples format above, brief and clear.": "`text_to_image(text=\"Draw an apple\", size_type=\"512x512\")`", "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\": \"We will use the Pygame library to create the game interface and handle user input. The game logic will be implemented using Python classes and data structures.\", \"File list\": [\"main.py\", \"game.py\"], \"Data structures and interfaces\": \"classDiagram\\n class Game {\\n -grid: List[List[int]]\\n -score: int\\n -game_over: bool\\n +__init__()\\n +reset_game()\\n +move(direction: str)\\n +is_game_over() bool\\n +get_empty_cells() List[Tuple[int, int]]\\n +add_new_tile()\\n +get_score() int\\n }\\n class UI {\\n -game: Game\\n +__init__(game: Game)\\n +draw_grid()\\n +draw_score()\\n +draw_game_over()\\n +handle_input()\\n }\\n Game --> UI\", \"Program call flow\": \"sequenceDiagram\\n participant M as Main\\n participant G as Game\\n participant U as UI\\n M->>G: reset_game()\\n M->>U: draw_grid()\\n M->>U: draw_score()\\n M->>U: handle_input()\\n U->>G: move(direction)\\n G->>G: add_new_tile()\\n G->>U: draw_grid()\\n G->>U: draw_score()\\n G->>U: draw_game_over()\\n G->>G: is_game_over()\\n G->>G: get_empty_cells()\\n G->>G: get_score()\", \"Anything UNCLEAR\": \"...\"}\n\n## Tasks\n{\"Required Python packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party dependencies required\"], \"Logic Analysis\": [[\"game.py\", \"Contains Game class and related functions for game logic\"], [\"main.py\", \"Contains main function, initializes the game and UI\"]], \"Task list\": [\"game.py\", \"main.py\"], \"Full API spec\": \"\", \"Shared Knowledge\": \"The game logic will be implemented using Python classes and data structures. The Pygame library will be used to create the game interface and handle user input.\", \"Anything UNCLEAR\": \"...\"}\n\n## Legacy Code\n```Code\n----- main.py\nif __name__ == \"__main__\":\nmain()\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## game.py\n\nfrom typing import List, Tuple\nimport random\n\nclass Game:\n def __init__(self):\n self.grid: List[List[int]] = [[0 for _ in range(4)] for _ in range(4)]\n self.score: int = 0\n self.game_over: bool = False\n\n def reset_game(self):\n self.grid = [[0 for _ in range(4)] for _ in range(4)]\n self.score = 0\n self.game_over = False\n self.add_new_tile()\n self.add_new_tile()\n\n def move(self, direction: str):\n # Implement the logic to move the tiles in the specified direction\n pass\n\n def is_game_over(self) -> bool:\n # Implement the logic to check if the game is over\n pass\n\n def get_empty_cells(self) -> List[Tuple[int, int]]:\n # Implement the logic to get the coordinates of empty cells in the grid\n pass\n\n def add_new_tile(self):\n empty_cells = self.get_empty_cells()\n if empty_cells:\n row, col = random.choice(empty_cells)\n self.grid[row][col] = 2 if random.random() < 0.9 else 4\n\n def get_score(self) -> int:\n return self.score\n```", - "\n## context\n\n### Project Name\n\n\n### Original Requirements\n开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"zh_cn\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结\",\n \"Project Name\": \"language_model_search\",\n \"Product Goals\": [\n \"提供基于大语言模型的高效搜索功能\",\n \"整合私有知识库,实现个性化搜索结果\",\n \"实现搜索结果的自动总结功能\"\n ],\n \"User Stories\": [\n \"作为用户,我希望能够快速找到我需要的信息\",\n \"作为用户,我希望搜索结果能够根据我的偏好进行个性化排序\",\n \"作为用户,我希望搜索结果能够自动总结,方便我快速了解内容\"\n ],\n \"Competitive Analysis\": [\n \"搜索引擎A: 提供基于大语言模型的搜索功能,但个性化程度较低\",\n \"知识库B: 整合私有知识库,但搜索速度较慢\",\n \"语言模型搜索C: 提供搜索结果自动总结功能,但搜索准确度有待提高\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"搜索引擎竞争分析\\\"\\n x-axis \\\"低搜索准确度\\\" --> \\\"高搜索准确度\\\"\\n y-axis \\\"低个性化程度\\\" --> \\\"高个性化程度\\\"\\n quadrant-1 \\\"需改进\\\"\\n quadrant-2 \\\"需提升\\\"\\n quadrant-3 \\\"重新评估\\\"\\n quadrant-4 \\\"扩展发展\\\"\\n \\\"搜索引擎A\\\": [0.6, 0.3]\\n \\\"知识库B\\\": [0.4, 0.2]\\n \\\"语言模型搜索C\\\": [0.7, 0.5]\\n \\\"我们的目标产品\\\": [0.8, 0.7]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"基于大语言模型的高效搜索功能\"\n ],\n [\n \"P1\",\n \"整合私有知识库,实现个性化搜索结果\"\n ],\n [\n \"P2\",\n \"实现搜索结果的自动总结功能\"\n ]\n ],\n \"UI Design draft\": \"搜索页面简洁明了,搜索结果清晰展示,提供个性化排序和自动总结功能。\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]" + "\n## context\n\n### Project Name\n\n\n### Original Requirements\n开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结\n\n### Search Information\n-\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Language\": \"en_us\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"Create a 2048 game\",\n \"Project Name\": \"game_2048\",\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ],\n \"User Stories\": [\n \"As a player, I want to be able to choose difficulty levels\",\n \"As a player, I want to see my score after each game\",\n \"As a player, I want to get restart button when I lose\",\n \"As a player, I want to see beautiful UI that make me feel good\",\n \"As a player, I want to play game via mobile phone\"\n ],\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"Reach and engagement of campaigns\\\"\\n x-axis \\\"Low Reach\\\" --> \\\"High Reach\\\"\\n y-axis \\\"Low Engagement\\\" --> \\\"High Engagement\\\"\\n quadrant-1 \\\"We should expand\\\"\\n quadrant-2 \\\"Need to promote\\\"\\n quadrant-3 \\\"Re-evaluate\\\"\\n quadrant-4 \\\"May be improved\\\"\\n \\\"Campaign A\\\": [0.3, 0.6]\\n \\\"Campaign B\\\": [0.45, 0.23]\\n \\\"Campaign C\\\": [0.57, 0.69]\\n \\\"Campaign D\\\": [0.78, 0.34]\\n \\\"Campaign E\\\": [0.40, 0.34]\\n \\\"Campaign F\\\": [0.35, 0.78]\\n \\\"Our Target Product\\\": [0.5, 0.6]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ],\n \"UI Design draft\": \"Basic function description with a simple style and layout.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Language: # Provide the language used in the project, typically matching the user's requirement language.\n- Programming Language: # Python/JavaScript or other mainstream programming language.\n- Original Requirements: # Place the original user's requirements here.\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style , like 'game_2048' or 'simple_crm.\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n- User Stories: typing.List[str] # Provide up to 3 to 5 scenario-based user stories.\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n- Competitive Quadrant Chart: # Use mermaid quadrantChart syntax. Distribute scores evenly between 0 and 1\n- Requirement Analysis: # Provide a detailed analysis of the requirements.\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n- UI Design draft: # Provide a simple description of UI elements, functions, style, and layout.\n- Anything UNCLEAR: # Mention any aspects of the project that are unclear and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Language\": \"zh_cn\",\n \"Programming Language\": \"Python\",\n \"Original Requirements\": \"开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结\",\n \"Project Name\": \"language_model_search\",\n \"Product Goals\": [\n \"提供基于大语言模型的高效搜索功能\",\n \"整合私有知识库,实现个性化搜索结果\",\n \"实现搜索结果的自动总结功能\"\n ],\n \"User Stories\": [\n \"作为用户,我希望能够快速找到我需要的信息\",\n \"作为用户,我希望搜索结果能够根据我的偏好进行个性化排序\",\n \"作为用户,我希望搜索结果能够自动总结,方便我快速了解内容\"\n ],\n \"Competitive Analysis\": [\n \"搜索引擎A: 提供基于大语言模型的搜索功能,但个性化程度较低\",\n \"知识库B: 整合私有知识库,但搜索速度较慢\",\n \"语言模型搜索C: 提供搜索结果自动总结功能,但搜索准确度有待提高\"\n ],\n \"Competitive Quadrant Chart\": \"quadrantChart\\n title \\\"搜索引擎竞争分析\\\"\\n x-axis \\\"低搜索准确度\\\" --> \\\"高搜索准确度\\\"\\n y-axis \\\"低个性化程度\\\" --> \\\"高个性化程度\\\"\\n quadrant-1 \\\"需改进\\\"\\n quadrant-2 \\\"需提升\\\"\\n quadrant-3 \\\"重新评估\\\"\\n quadrant-4 \\\"扩展发展\\\"\\n \\\"搜索引擎A\\\": [0.6, 0.3]\\n \\\"知识库B\\\": [0.4, 0.2]\\n \\\"语言模型搜索C\\\": [0.7, 0.5]\\n \\\"我们的目标产品\\\": [0.8, 0.7]\",\n \"Requirement Analysis\": \"\",\n \"Requirement Pool\": [\n [\n \"P0\",\n \"基于大语言模型的高效搜索功能\"\n ],\n [\n \"P1\",\n \"整合私有知识库,实现个性化搜索结果\"\n ],\n [\n \"P2\",\n \"实现搜索结果的自动总结功能\"\n ]\n ],\n \"UI Design draft\": \"搜索页面简洁明了,搜索结果清晰展示,提供个性化排序和自动总结功能。\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\n## context\nNone\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Project Name\": \"game_2048\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style with underline, like 'game_2048' or 'simple_crm.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Project Name\": \"game_2048\"\n}\n[/CONTENT]", + "\n## context\nCompare the key's value of nodes_output and the corresponding requirements one by one. If a key's value that does not match the requirement is found, provide the comment content on how to modify it. No output is required for matching keys.\n\n### nodes_output\n{\"Project Name\": {\"value\": \"game snake\", \"requirement\": \"According to the content of \\\"Original Requirements,\\\" name the project using snake case style with underline, like 'game_2048' or 'simple_crm.\"}}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"key1\": \"comment1\",\n \"key2\": \"comment2\",\n \"keyn\": \"commentn\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- key1: # the first key name of mismatch key\n- key2: # the second key name of mismatch key\n- keyn: # the last key name of mismatch key\n\n## constraint\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow format example's json format, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Project Name\": \"Use snake case style with underline, like 'game_2048' or 'simple_crm.'\"\n}\n[/CONTENT]", + "\n## context\nCompare the key's value of nodes_output and the corresponding requirements one by one. If a key's value that does not match the requirement is found, provide the comment content on how to modify it. No output is required for matching keys.\n\n### nodes_output\n{\"Project Name\": {\"value\": \"game_2048\", \"requirement\": \"According to the content of \\\"Original Requirements,\\\" name the project using snake case style with underline, like 'game_2048' or 'simple_crm.\"}}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"key1\": \"comment1\",\n \"key2\": \"comment2\",\n \"keyn\": \"commentn\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- key1: # the first key name of mismatch key\n- key2: # the second key name of mismatch key\n- keyn: # the last key name of mismatch key\n\n## constraint\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow format example's json format, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Project Name\": \"According to the content of \\\"Original Requirements,\\\" name the project using snake case style with underline, like 'game_2048' or 'simple_crm.\"\n}\n[/CONTENT]", + "\n## context\nchange the nodes_output key's value to meet its comment and no need to add extra comment.\n\n### nodes_output\n{\"Project Name\": {\"value\": \"game snake\", \"comment\": \"Use snake case style with underline, like 'game_2048' or 'simple_crm.'\"}}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Project Name\": \"game_2048\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style with underline, like 'game_2048' or 'simple_crm.\n\n\n## constraint\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow format example's json format, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Project Name\": \"game_snake\"\n}\n[/CONTENT]", + "\n## context\nCompare the key's value of nodes_output and the corresponding requirements one by one. If a key's value that does not match the requirement is found, provide the comment content on how to modify it. No output is required for matching keys.\n\n### nodes_output\n{\"Project Name\": {\"value\": \"game_snake\", \"requirement\": \"According to the content of \\\"Original Requirements,\\\" name the project using snake case style with underline, like 'game_2048' or 'simple_crm.\"}}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"key1\": \"comment1\",\n \"key2\": \"comment2\",\n \"keyn\": \"commentn\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- key1: # the first key name of mismatch key\n- key2: # the second key name of mismatch key\n- keyn: # the last key name of mismatch key\n\n## constraint\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow format example's json format, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Project Name\": \"The project name should be in snake case style with underscores, such as 'game_2048' or 'simple_crm'.\"\n}\n[/CONTENT]", + "\n## context\nchange the nodes_output key's value to meet its comment and no need to add extra comment.\n\n### nodes_output\n{\"Project Name\": {\"value\": \"game_snake\", \"comment\": \"The project name should be in snake case style with underscores, such as 'game_2048' or 'simple_crm'.\"}}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Project Name\": \"game_2048\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Project Name: # According to the content of \"Original Requirements,\" name the project using snake case style with underline, like 'game_2048' or 'simple_crm.\n\n\n## constraint\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow format example's json format, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Project Name\": \"game_snake\"\n}\n[/CONTENT]", + "You are an action classifier#SYSTEM_MSG_END#If the text explicitly want you to Generate a voice file from the input text, text-to-speech, return `[SKILL]: text_to_speech` brief and clear. For instance: [SKILL]: text_to_speech\nIf the text explicitly want you to Create a drawing based on the text., return `[SKILL]: text_to_image` brief and clear. For instance: [SKILL]: text_to_image\nIf the text explicitly want you to Perform Google searches to provide real-time information., return `[SKILL]: web_search` brief and clear. For instance: [SKILL]: web_search\nOtherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: No, I do not have a poison apple. Do you have a poison apple?\n": "[TALK]: No, I do not have a poison apple. Do you have a poison apple?", + "You are an action classifier#SYSTEM_MSG_END#If the text explicitly want you to Generate a voice file from the input text, text-to-speech, return `[SKILL]: text_to_speech` brief and clear. For instance: [SKILL]: text_to_speech\nIf the text explicitly want you to Create a drawing based on the text., return `[SKILL]: text_to_image` brief and clear. For instance: [SKILL]: text_to_image\nIf the text explicitly want you to Perform Google searches to provide real-time information., return `[SKILL]: web_search` brief and clear. For instance: [SKILL]: web_search\nOtherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: Sure, I can draw you an apple. Draw me an apple.\n": "[SKILL]: text_to_image", + "You are a function parser.#MSG_SEP#You can convert spoken words into function parameters.#SYSTEM_MSG_END#text_to_image function parameters description:\nparameter `text`: The text used for image conversion.\nparameter `size_type`: size type\n\n---\nExamples:\nIf want you to do `Draw a girl`, return `text_to_image(text=\"Draw a girl\", size_type=\"512x512\")` brief and clear.\nIf want you to do `Draw an apple`, return `text_to_image(text=\"Draw an apple\", size_type=\"512x512\")` brief and clear.\n\n---\n\nRefer to the `text_to_image` function description, and fill in the function parameters according to the example \"I want you to do xx\" in the Examples section.\nNow I want you to do `Sure, I can draw you an apple. Draw me an apple.`, return function parameters in Examples format above, brief and clear.": "`text_to_image(text=\"Sure, I can draw you an apple. Draw me an apple.\", size_type=\"512x512\")`", + "You are an action classifier#SYSTEM_MSG_END#Otherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: Sure, I can draw you an apple. Draw me an apple.\n": "[DRAW]: draw an apple" } \ No newline at end of file From 30a884506f240d1c9e63b2639d0bff865d6214cb Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 15 Jan 2024 18:56:32 +0800 Subject: [PATCH 346/637] fix bugs --- tests/data/rsp_cache.json | 38 ++++++++++++++++++- ...ve_writing.py => test_creative_writing.py} | 0 .../examples/{game24.py => test_game24.py} | 0 3 files changed, 37 insertions(+), 1 deletion(-) rename tests/metagpt/strategy/examples/{creative_writing.py => test_creative_writing.py} (100%) rename tests/metagpt/strategy/examples/{game24.py => test_game24.py} (100%) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index 2d43e20c5..9ebe50a3c 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -225,5 +225,41 @@ "You are an action classifier#SYSTEM_MSG_END#If the text explicitly want you to Generate a voice file from the input text, text-to-speech, return `[SKILL]: text_to_speech` brief and clear. For instance: [SKILL]: text_to_speech\nIf the text explicitly want you to Create a drawing based on the text., return `[SKILL]: text_to_image` brief and clear. For instance: [SKILL]: text_to_image\nIf the text explicitly want you to Perform Google searches to provide real-time information., return `[SKILL]: web_search` brief and clear. For instance: [SKILL]: web_search\nOtherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: No, I do not have a poison apple. Do you have a poison apple?\n": "[TALK]: No, I do not have a poison apple. Do you have a poison apple?", "You are an action classifier#SYSTEM_MSG_END#If the text explicitly want you to Generate a voice file from the input text, text-to-speech, return `[SKILL]: text_to_speech` brief and clear. For instance: [SKILL]: text_to_speech\nIf the text explicitly want you to Create a drawing based on the text., return `[SKILL]: text_to_image` brief and clear. For instance: [SKILL]: text_to_image\nIf the text explicitly want you to Perform Google searches to provide real-time information., return `[SKILL]: web_search` brief and clear. For instance: [SKILL]: web_search\nOtherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: Sure, I can draw you an apple. Draw me an apple.\n": "[SKILL]: text_to_image", "You are a function parser.#MSG_SEP#You can convert spoken words into function parameters.#SYSTEM_MSG_END#text_to_image function parameters description:\nparameter `text`: The text used for image conversion.\nparameter `size_type`: size type\n\n---\nExamples:\nIf want you to do `Draw a girl`, return `text_to_image(text=\"Draw a girl\", size_type=\"512x512\")` brief and clear.\nIf want you to do `Draw an apple`, return `text_to_image(text=\"Draw an apple\", size_type=\"512x512\")` brief and clear.\n\n---\n\nRefer to the `text_to_image` function description, and fill in the function parameters according to the example \"I want you to do xx\" in the Examples section.\nNow I want you to do `Sure, I can draw you an apple. Draw me an apple.`, return function parameters in Examples format above, brief and clear.": "`text_to_image(text=\"Sure, I can draw you an apple. Draw me an apple.\", size_type=\"512x512\")`", - "You are an action classifier#SYSTEM_MSG_END#Otherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: Sure, I can draw you an apple. Draw me an apple.\n": "[DRAW]: draw an apple" + "You are an action classifier#SYSTEM_MSG_END#Otherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is \"xxxx\" return [TALK]: xxxx\n\nNow what specific action is explicitly mentioned in the text: Sure, I can draw you an apple. Draw me an apple.\n": "[DRAW]: draw an apple", + "Here is an Example for 1 input and 8 possible thoughts:\nInput: 2 8 8 14\nPossible next steps:\n2 + 8 = 10 (left: 8 10 14)\n8 / 2 = 4 (left: 4 8 14)\n14 + 2 = 16 (left: 8 8 16)\n2 * 8 = 16 (left: 8 14 16)\n8 - 2 = 6 (left: 6 8 14)\n14 - 8 = 6 (left: 2 6 8)\n14 / 2 = 7 (left: 7 8 8)\n14 - 2 = 12 (left: 8 8 12)\n\nHere is my task for 1 input and 5 possible thoughts:\nInput: 4 5 6 10\nPossible next steps:\n\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Here is the list of possible next steps for the given input in JSON format:\n\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"4 + 5 = 9 (left: 6 9 10)\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"5 + 6 = 11 (left: 4 11 10)\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"4 * 5 = 20 (left: 6 20 10)\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"6 - 4 = 2 (left: 2 5 10)\"\n },\n {\n \"node_id\": \"5\",\n \"node_state_instruction\": \"10 - 4 = 6 (left: 6 5 6)\"\n }\n]\n```", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n6 9 10\n": "6 * 9 - 10 = 54 - 10 = 44\n(9 - 6) * 10 = 3 * 10 = 30\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n6 20 10\n": "I'm sorry, but it seems that you haven't completed the calculation for the numbers 6, 20, and 10. If you'd like, I can help you with that.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n4 11 10\n": "4 11 10 cannot reach 24.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n2 5 10\n": "2 * 5 * 10 = 100\n(2 + 5) * 10 = 70\n2 5 10 are all too big\nimpossible", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n6 5 6\n": "I'm sorry, but it seems that you have not provided a complete expression for the numbers 6, 5, and 6 to reach 24. If you would like to try again, please provide a complete expression using the numbers 6, 5, and 6 to reach the target number of 24.", + "Here is an Example for 1 input and 8 possible thoughts:\nInput: 2 8 8 14\nPossible next steps:\n2 + 8 = 10 (left: 8 10 14)\n8 / 2 = 4 (left: 4 8 14)\n14 + 2 = 16 (left: 8 8 16)\n2 * 8 = 16 (left: 8 14 16)\n8 - 2 = 6 (left: 6 8 14)\n14 - 8 = 6 (left: 2 6 8)\n14 / 2 = 7 (left: 7 8 8)\n14 - 2 = 12 (left: 8 8 12)\n\nHere is my task for 1 input and 5 possible thoughts:\nInput: 4 11 10\nPossible next steps:\n\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Here are the possible next steps for the given input in the requested JSON format:\n\nInput: 4 11 10\nPossible next steps:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"4 + 11 = 15 (left: 10 15)\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"11 - 4 = 7 (left: 7 10)\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"4 * 11 = 44 (left: 10 44)\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"11 / 4 = 2.75 (left: 2.75 10)\"\n },\n {\n \"node_id\": \"5\",\n \"node_state_instruction\": \"11 + 4 = 15 (left: 10 15)\"\n }\n]\n```", + "Here is an Example for 1 input and 8 possible thoughts:\nInput: 2 8 8 14\nPossible next steps:\n2 + 8 = 10 (left: 8 10 14)\n8 / 2 = 4 (left: 4 8 14)\n14 + 2 = 16 (left: 8 8 16)\n2 * 8 = 16 (left: 8 14 16)\n8 - 2 = 6 (left: 6 8 14)\n14 - 8 = 6 (left: 2 6 8)\n14 / 2 = 7 (left: 7 8 8)\n14 - 2 = 12 (left: 8 8 12)\n\nHere is my task for 1 input and 5 possible thoughts:\nInput: 6 9 10\nPossible next steps:\n\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Certainly! Here are the possible next steps for the given input in JSON format:\n\nInput: 6 9 10\nPossible next steps:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"6 + 9 = 15 (left: 10 15)\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"6 * 9 = 54 (left: 10 54)\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"9 - 6 = 3 (left: 3 9 10)\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"10 - 6 = 4 (left: 4 6 9)\"\n },\n {\n \"node_id\": \"5\",\n \"node_state_instruction\": \"10 / 6 = 1 (left: 1 6 9)\"\n }\n]\n```", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n10 15\n": "I'm sorry, I cannot evaluate the numbers 10 and 15 without an operation. If you provide an operation, I can help you determine if they can reach 24.", + "Here is an Example for 1 input and 8 possible thoughts:\nInput: 2 8 8 14\nPossible next steps:\n2 + 8 = 10 (left: 8 10 14)\n8 / 2 = 4 (left: 4 8 14)\n14 + 2 = 16 (left: 8 8 16)\n2 * 8 = 16 (left: 8 14 16)\n8 - 2 = 6 (left: 6 8 14)\n14 - 8 = 6 (left: 2 6 8)\n14 / 2 = 7 (left: 7 8 8)\n14 - 2 = 12 (left: 8 8 12)\n\nHere is my task for 1 input and 5 possible thoughts:\nInput: 6 20 10\nPossible next steps:\n\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Certainly! Here is the list of nodes for the given input and possible next steps:\n\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"6 + 20 = 26 (left: 10 26)\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"20 - 6 = 14 (left: 14 10)\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"6 * 20 = 120 (left: 10 120)\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"20 / 6 = 3.33 (left: 3.33 10)\"\n },\n {\n \"node_id\": \"5\",\n \"node_state_instruction\": \"20 + 6 = 26 (left: 10 26)\"\n }\n]\n```", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n1 6 9\n": "1 6 9 cannot reach 24 using addition, subtraction, multiplication, or division. Therefore, it is impossible for these numbers to reach 24.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n10 54\n": "It seems like you didn't provide the complete set of numbers for the last calculation. Could you please provide the third number so that I can help you evaluate if they can reach 24?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n7 10\n": "I'm sorry, but it seems like you didn't provide a complete set of numbers for the last question. Could you please provide the third number so that I can evaluate if they can reach 24?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n10 44\n": "It seems like you didn't provide the second number for the last set. Could you please provide the second number so that I can evaluate if they can reach 24?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n10 26\n": "It seems that you have not provided the complete set of numbers for the last question. Could you please provide the third number for the set?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n10 120\n": "It seems that you have not provided the complete set of numbers for the last question. Could you please provide the third number for the evaluation?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n3 9 10\n": "3 * 9 * 10 = 270\n(10 - 3) * 9 = 63\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n2.75 10\n": "It seems like you have a mix of numbers and operations. Could you please provide the operations for the last set of numbers (2.75 and 10) so that I can help you evaluate if they can reach 24?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n4 6 9\n": "4 + 6 + 9 = 19\n(9 - 4) * 6 = 5 * 6 = 30\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n14 10\n": "I'm sorry, it seems like you didn't provide the complete set of numbers for the last expression. Could you please provide the third number so that I can evaluate if they can reach 24?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n3.33 10\n": "It seems like you have provided a list of numbers and evaluated if they can reach 24 through addition, subtraction, multiplication, and division. Here are the evaluations:\n\n10 14 - Sure\n11 12 - Impossible\n4 4 10 - Sure\n4 9 11 - Sure\n5 7 8 - Likely\n5 6 6 - Likely\n10 10 11 - Impossible\n1 3 3 - Impossible\n\nIt looks like you have evaluated the combinations accurately. As for the last set of numbers \"3.33 10,\" it seems incomplete. If you provide the intended operation or additional numbers, I can help you evaluate it.", + "Here is an Example for 1 input and 8 possible thoughts:\nInput: 2 8 8 14\nPossible next steps:\n2 + 8 = 10 (left: 8 10 14)\n8 / 2 = 4 (left: 4 8 14)\n14 + 2 = 16 (left: 8 8 16)\n2 * 8 = 16 (left: 8 14 16)\n8 - 2 = 6 (left: 6 8 14)\n14 - 8 = 6 (left: 2 6 8)\n14 / 2 = 7 (left: 7 8 8)\n14 - 2 = 12 (left: 8 8 12)\n\nHere is my task for 1 input and 5 possible thoughts:\nInput: 10 15\nPossible next steps:\n\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Here's the JSON format for the possible next steps for the given input:\n\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"10 + 15 = 25 (left: 25)\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"15 - 10 = 5 (left: 5)\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"10 * 15 = 150 (left: 150)\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"15 / 10 = 1.5 (left: 1.5)\"\n },\n {\n \"node_id\": \"5\",\n \"node_state_instruction\": \"15 + 10 = 25 (left: 25)\"\n }\n]\n```", + "Here is an Example for 1 input and 8 possible thoughts:\nInput: 2 8 8 14\nPossible next steps:\n2 + 8 = 10 (left: 8 10 14)\n8 / 2 = 4 (left: 4 8 14)\n14 + 2 = 16 (left: 8 8 16)\n2 * 8 = 16 (left: 8 14 16)\n8 - 2 = 6 (left: 6 8 14)\n14 - 8 = 6 (left: 2 6 8)\n14 / 2 = 7 (left: 7 8 8)\n14 - 2 = 12 (left: 8 8 12)\n\nHere is my task for 1 input and 5 possible thoughts:\nInput: 3 9 10\nPossible next steps:\n\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Here are the possible next steps for the given input in JSON format:\n\nInput: 3 9 10\nPossible next steps:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"3 + 9 = 12 (left: 10 12)\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"9 - 3 = 6 (left: 6 9 10)\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"10 - 3 = 7 (left: 7 9 10)\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"3 * 9 = 27 (left: 10 27)\"\n },\n {\n \"node_id\": \"5\",\n \"node_state_instruction\": \"9 / 3 = 3 (left: 3 9 10)\"\n }\n]\n```", + "Here is an Example for 1 input and 8 possible thoughts:\nInput: 2 8 8 14\nPossible next steps:\n2 + 8 = 10 (left: 8 10 14)\n8 / 2 = 4 (left: 4 8 14)\n14 + 2 = 16 (left: 8 8 16)\n2 * 8 = 16 (left: 8 14 16)\n8 - 2 = 6 (left: 6 8 14)\n14 - 8 = 6 (left: 2 6 8)\n14 / 2 = 7 (left: 7 8 8)\n14 - 2 = 12 (left: 8 8 12)\n\nHere is my task for 1 input and 5 possible thoughts:\nInput: 4 6 9\nPossible next steps:\n\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Certainly! Here are the possible next steps for the given input in the requested JSON format:\n\nInput: 4 6 9\nPossible next steps:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"4 + 6 = 10 (left: 10 9)\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"4 * 6 = 24 (left: 24 9)\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"4 - 6 = -2 (left: -2 9)\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"6 + 9 = 15 (left: 4 15)\"\n },\n {\n \"node_id\": \"5\",\n \"node_state_instruction\": \"6 * 9 = 54 (left: 4 54)\"\n }\n]\n```\nI hope this helps! If you need further assistance, feel free to ask.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n10 27\n": "I'm sorry, but it seems like you didn't provide the third number for the last set. Could you please provide the third number so I can evaluate if the given numbers can reach 24?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n1.5\n": "The evaluation of the given numbers is as follows:\n\n10 14: Sure\n11 12: Impossible\n4 4 10: Sure\n4 9 11: Sure\n5 7 8: Likely\n5 6 6: Likely\n10 10 11: Impossible\n1 3 3: Impossible\n1.5: I'm sorry, I cannot evaluate a single number for reaching 24.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n4 15\n": "It seems like you didn't provide the complete set of numbers for the last question. Could you please provide the third number so that I can evaluate if they can reach 24?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n24 9\n": "24 + 9 = 33\n24 - 9 = 15\n24 * 9 = 216\n24 / 9 = 2.67\nIt is impossible to reach 24 with the given numbers.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n10 9\n": "It seems like you might have missed providing the operation for the last set of numbers \"10 9\". If you could provide the operation, I can help evaluate if the given numbers can reach 24.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n25\n": "It seems that you have provided a list of numbers and evaluated whether they can reach 24 through addition, subtraction, multiplication, and division. Based on your evaluations, the combinations of numbers that can reach 24 are \"10 14\" and \"4 4 10,\" which are sure to reach 24. The combinations \"11 12\" and \"4 9 11\" are also sure to reach 24. The combination \"5 7 8\" is likely to reach 24, and the combinations \"5 6 6\" and \"10 10 11\" are unlikely to reach 24. The combination \"1 3 3\" is impossible to reach 24. If you have any more numbers to evaluate, feel free to share them with me.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n4 54\n": "I'm sorry, but I cannot evaluate the expression \"4 54\" without an operator. Could you please provide the operator for the expression?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n-2 9\n": "I'm sorry, but it seems like you didn't provide the third number for the last expression. Could you please provide the third number so that I can evaluate if the given numbers can reach 24?", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n10 12\n": "It seems like you have evaluated the combinations correctly. Based on the evaluations you provided, the combinations that can reach 24 are:\n\n- 10 14 (sure)\n- 4 4 10 (sure)\n- 4 9 11 (sure)\n- 5 7 8 (likely)\n- 5 6 6 (likely)\n\nThe combinations that cannot reach 24 are:\n\n- 11 12 (impossible)\n- 10 10 11 (impossible)\n- 1 3 3 (impossible)\n\nAs for the combination \"10 12,\" it seems like it was cut off. If you provide the complete expression, I can help evaluate it for you.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n150\n": "It seems like you have evaluated the given numbers correctly. Here are the results:\n\n10 14: Sure\n11 12: Impossible\n4 4 10: Sure\n4 9 11: Sure\n5 7 8: Likely\n5 6 6: Likely\n10 10 11: Impossible\n1 3 3: Impossible\n\nAs for the number 150, it is not clear what operation or combination of numbers it is associated with. If you have a specific question or context for the number 150, please provide more details so that I can assist you further.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n5\n": "The numbers 10 and 14 can reach 24, so that is sure. The numbers 11 and 12 cannot reach 24, so that is impossible. The numbers 4, 4, and 10 can reach 24, so that is sure. The numbers 4, 9, and 11 can reach 24, so that is sure. The numbers 5, 7, and 8 cannot reach 24, but they are within a reasonable range, so that is likely. The numbers 5, 6, and 6 cannot reach 24, but they are within a reasonable range, so that is likely. The numbers 10, 10, and 11 cannot reach 24, so that is impossible. The numbers 1, 3, and 3 cannot reach 24, so that is impossible. The number 5 by itself cannot reach 24.", + "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n7 9 10\n": "7 * 9 - 10 = 63 - 10 = 53\n(9 - 7) * 10 = 2 * 10 = 20\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely" } \ No newline at end of file diff --git a/tests/metagpt/strategy/examples/creative_writing.py b/tests/metagpt/strategy/examples/test_creative_writing.py similarity index 100% rename from tests/metagpt/strategy/examples/creative_writing.py rename to tests/metagpt/strategy/examples/test_creative_writing.py diff --git a/tests/metagpt/strategy/examples/game24.py b/tests/metagpt/strategy/examples/test_game24.py similarity index 100% rename from tests/metagpt/strategy/examples/game24.py rename to tests/metagpt/strategy/examples/test_game24.py From b430e2c88fe6db7104faf9dc44cad639f95965c9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 19:02:37 +0800 Subject: [PATCH 347/637] update scrape_web module. --- metagpt/tools/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index c24dc6fce..2f8941fdb 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -78,7 +78,7 @@ TOOL_TYPE_MAPPINGS = { ), "scrape_web": ToolType( name="scrape_web", - module="metagpt.tools.scrape_web", + module=str(TOOL_LIBS_PATH / "scrape_web"), desc="Scrape data from web page.", usage_prompt="", ), From 6b52ee3e7c2a6d78c9552d3bb3b1e67fbbb1486f Mon Sep 17 00:00:00 2001 From: better629 Date: Mon, 15 Jan 2024 20:10:39 +0800 Subject: [PATCH 348/637] fix UserWarning: Pydantic serializer warning Expected str but got dict --- metagpt/schema.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/schema.py b/metagpt/schema.py index 853a9c6bb..c0f867831 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -212,7 +212,7 @@ class Message(BaseModel): return any_to_str_set(send_to if send_to else {MESSAGE_ROUTE_TO_ALL}) @field_serializer("instruct_content", mode="plain") - def ser_instruct_content(self, ic: BaseModel) -> Union[str, None]: + def ser_instruct_content(self, ic: BaseModel) -> Union[dict, None]: ic_dict = None if ic: # compatible with custom-defined ActionOutput From 465279dfcc3fb5f75c7b4cda9349af558f1ad590 Mon Sep 17 00:00:00 2001 From: better629 Date: Mon, 15 Jan 2024 20:22:04 +0800 Subject: [PATCH 349/637] update unittest due to implement update --- tests/metagpt/serialize_deserialize/test_write_prd.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tests/metagpt/serialize_deserialize/test_write_prd.py b/tests/metagpt/serialize_deserialize/test_write_prd.py index 820ee237c..945ec5efd 100644 --- a/tests/metagpt/serialize_deserialize/test_write_prd.py +++ b/tests/metagpt/serialize_deserialize/test_write_prd.py @@ -18,5 +18,5 @@ async def test_action_serdeser(new_filename): new_action = WritePRD(**ser_action_dict) assert new_action.name == "WritePRD" - action_output = await new_action.run(with_messages=Message(content="write a cli snake game")) - assert len(action_output.content) > 0 + with pytest.raises(FileNotFoundError): + action_output = await new_action.run(with_messages=Message(content="write a cli snake game")) From b800e57def0c40986c0d3993a672c5a57fa9dd10 Mon Sep 17 00:00:00 2001 From: better629 Date: Mon, 15 Jan 2024 20:23:46 +0800 Subject: [PATCH 350/637] fix format --- metagpt/actions/action_node.py | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/action_node.py b/metagpt/actions/action_node.py index 4f61af4ed..ca41c76a5 100644 --- a/metagpt/actions/action_node.py +++ b/metagpt/actions/action_node.py @@ -469,8 +469,10 @@ class ActionNode: return dict() prompt = template.format( - nodes_output=json.dumps(nodes_output, ensure_ascii=False), tag=TAG, constraint=FORMAT_CONSTRAINT, - prompt_schema="json" + nodes_output=json.dumps(nodes_output, ensure_ascii=False), + tag=TAG, + constraint=FORMAT_CONSTRAINT, + prompt_schema="json", ) content = await self.llm.aask(prompt) @@ -568,7 +570,7 @@ class ActionNode: example=example, instruction=instruction, constraint=FORMAT_CONSTRAINT, - prompt_schema="json" + prompt_schema="json", ) # step2, use `_aask_v1` to get revise structure result From 7ffb2208e21248a73c04cbdff11282d447f1c016 Mon Sep 17 00:00:00 2001 From: better629 Date: Mon, 15 Jan 2024 20:24:43 +0800 Subject: [PATCH 351/637] fix format --- tests/metagpt/serialize_deserialize/test_write_prd.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/metagpt/serialize_deserialize/test_write_prd.py b/tests/metagpt/serialize_deserialize/test_write_prd.py index 945ec5efd..afc483e9a 100644 --- a/tests/metagpt/serialize_deserialize/test_write_prd.py +++ b/tests/metagpt/serialize_deserialize/test_write_prd.py @@ -19,4 +19,4 @@ async def test_action_serdeser(new_filename): new_action = WritePRD(**ser_action_dict) assert new_action.name == "WritePRD" with pytest.raises(FileNotFoundError): - action_output = await new_action.run(with_messages=Message(content="write a cli snake game")) + await new_action.run(with_messages=Message(content="write a cli snake game")) From d1666c3307289edd0fcb53c8ba881574ee0dca19 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 21:17:01 +0800 Subject: [PATCH 352/637] update get_choice_function_arguments. --- metagpt/provider/openai_api.py | 71 +++++++++++++++++++++++----------- 1 file changed, 48 insertions(+), 23 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 747e36480..66d215eda 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -9,6 +9,7 @@ @Modified By: mashenquan, 2023/12/1. Fix bug: Unclosed connection caused by openai 0.x. """ +import re import json from typing import AsyncIterator, Union @@ -37,6 +38,7 @@ from metagpt.utils.token_counter import ( count_string_tokens, get_max_completion_tokens, ) +from metagpt.utils.common import CodeParser def log_and_reraise(retry_state): @@ -147,10 +149,7 @@ class OpenAILLM(BaseLLM): def _func_configs(self, messages: list[dict], timeout=3, **kwargs) -> dict: """Note: Keep kwargs consistent with https://platform.openai.com/docs/api-reference/chat/create""" if "tools" not in kwargs: - configs = { - "tools": [{"type": "function", "function": GENERAL_FUNCTION_SCHEMA}], - "tool_choice": GENERAL_TOOL_CHOICE, - } + configs = {"tools": [{"type": "function", "function": GENERAL_FUNCTION_SCHEMA}]} kwargs.update(configs) return self._cons_kwargs(messages=messages, timeout=timeout, **kwargs) @@ -161,23 +160,7 @@ class OpenAILLM(BaseLLM): self._update_costs(rsp.usage) return rsp - def _process_message(self, messages: Union[str, Message, list[dict], list[Message], list[str]]) -> list[dict]: - """convert messages to list[dict].""" - if isinstance(messages, list): - messages = [Message(content=msg) if isinstance(msg, str) else msg for msg in messages] - return [msg if isinstance(msg, dict) else msg.to_dict() for msg in messages] - - if isinstance(messages, Message): - messages = [messages.to_dict()] - elif isinstance(messages, str): - messages = [{"role": "user", "content": messages}] - else: - raise ValueError( - f"Only support messages type are: str, Message, list[dict], but got {type(messages).__name__}!" - ) - return messages - - async def aask_code(self, messages: Union[str, Message, list[dict]], **kwargs) -> dict: + async def aask_code(self, messages: list[dict], **kwargs) -> dict: """Use function of tools to ask a code. Note: Keep kwargs consistent with https://platform.openai.com/docs/api-reference/chat/create @@ -187,18 +170,60 @@ class OpenAILLM(BaseLLM): >>> rsp = await llm.aask_code(msg) # -> {'language': 'python', 'code': "print('Hello, World!')"} """ - messages = self._process_message(messages) rsp = await self._achat_completion_function(messages, **kwargs) return self.get_choice_function_arguments(rsp) + def _parse_arguments(self, arguments: str) -> dict: + """parse arguments in openai function call""" + if 'langugae' not in arguments and 'code' not in arguments: + logger.warning(f"Not found `code`, `language`, We assume it is pure code:\n {arguments}\n. ") + return {'language': 'python', 'code': arguments} + + # 匹配language + language_pattern = re.compile(r'[\"\']?language[\"\']?\s*:\s*["\']([^"\']+?)["\']', re.DOTALL) + language_match = language_pattern.search(arguments) + language_value = language_match.group(1) if language_match else None + + # 匹配code + code_pattern = r'(["\'`]{3}|["\'`])([\s\S]*?)\1' + try: + code_value = re.findall(code_pattern, arguments)[-1][-1] + except Exception as e: + logger.error(f"{e}, when re.findall({code_pattern}, {arguments})") + code_value = None + + if code_value is None: + raise ValueError(f"Parse code error for {arguments}") + # arguments只有code的情况 + return {'language': language_value, 'code': code_value} + @handle_exception def get_choice_function_arguments(self, rsp: ChatCompletion) -> dict: """Required to provide the first function arguments of choice. + :param dict rsp: same as in self.get_choice_function(rsp) :return dict: return the first function arguments of choice, for example, {'language': 'python', 'code': "print('Hello, World!')"} """ - return json.loads(rsp.choices[0].message.tool_calls[0].function.arguments) + message = rsp.choices[0].message + if ( + message.tool_calls is not None and + message.tool_calls[0].function is not None and + message.tool_calls[0].function.arguments is not None + ): + # reponse is code + try: + return json.loads(message.tool_calls[0].function.arguments, strict=False) + except json.decoder.JSONDecodeError as e: + logger.debug(f"Got JSONDecodeError for {message.tool_calls[0].function.arguments},\ + we will use RegExp to parse code, \n {e}") + return {'language': 'python', 'code': self._parse_arguments(message.tool_calls[0].function.arguments)} + elif message.tool_calls is None and message.content is not None: + # reponse is message + return {'language': 'markdown', 'code': self.get_choice_text(rsp)} + else: + logger.error(f"Failed to parse \n {rsp}\n") + raise Exception(f"Failed to parse \n {rsp}\n") def get_choice_text(self, rsp: ChatCompletion) -> str: """Required to provide the first text of choice""" From f9b1cce654e36ac764acd7db0b7d5c74404dc877 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 15 Jan 2024 22:21:56 +0800 Subject: [PATCH 353/637] update code-intepreter by auto aask. --- metagpt/actions/write_analysis_code.py | 2 +- metagpt/provider/openai_api.py | 23 +++++++++++++++++++++++ metagpt/roles/code_interpreter.py | 9 ++++++--- 3 files changed, 30 insertions(+), 4 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 76d47ba28..bceb100b1 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -88,7 +88,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): ) -> str: # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) prompt = self.process_msg(context, system_msg) - is_only_code = kwargs.pop("only_code", True) + is_only_code = kwargs.pop("only_code", False) code_content = await self.llm.aask_code(prompt, **kwargs) if is_only_code: diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 66d215eda..7bdb4bfbe 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -154,7 +154,30 @@ class OpenAILLM(BaseLLM): return self._cons_kwargs(messages=messages, timeout=timeout, **kwargs) + def _process_message(self, messages: Union[str, Message, list[dict], list[Message], list[str]]) -> list[dict]: + """convert messages to list[dict].""" + # 全部转成list + if not isinstance(messages, list): + messages = [messages] + + # 转成list[dict] + processed_messages = [] + for msg in messages: + if isinstance(msg, str): + processed_messages.append({"role": "user", "content": msg}) + elif isinstance(msg, dict): + assert set(msg.keys()) == set(['role', 'content']) + processed_messages.append(msg) + elif isinstance(msg, Message): + processed_messages.append(msg.to_dict()) + else: + raise ValueError( + f"Only support message type are: str, Message, dict, but got {type(messages).__name__}!" + ) + return processed_messages + async def _achat_completion_function(self, messages: list[dict], timeout=3, **chat_configs) -> ChatCompletion: + messages = self._process_message(messages) kwargs = self._func_configs(messages=messages, timeout=timeout, **chat_configs) rsp: ChatCompletion = await self.aclient.chat.completions.create(**kwargs) self._update_costs(rsp.usage) diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 164c7cb12..afd51a575 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -52,7 +52,7 @@ class CodeInterpreter(Role): async def _act_on_task(self, current_task: Task) -> TaskResult: code, result, is_success = await self._write_and_exec_code() - task_result = TaskResult(code=code, result=result, is_success=is_success) + task_result = TaskResult(code=code['code'], result=result, is_success=is_success) return task_result async def _write_and_exec_code(self, max_retry: int = 3): @@ -63,10 +63,10 @@ class CodeInterpreter(Role): ### write code ### code, cause_by = await self._write_code() - self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) + self.working_memory.add(Message(content=code['code'], role="assistant", cause_by=cause_by)) ### execute code ### - result, success = await self.execute_code.run(code) + result, success = await self.execute_code.run(**code) print(result) self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) @@ -91,6 +91,9 @@ class CodeInterpreter(Role): context = self.planner.get_useful_memories() code = await todo.run(context=context, plan=self.planner.plan, temperature=0.0) + # 暂时在这里转换 WriteCodeWithTools 的输出 + if isinstance(code, str): + code = {'code': code, 'language': 'python'} return code, todo From 7005a1e86f5f43144bf6f31010a145849dca1f14 Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 15 Jan 2024 23:12:09 +0800 Subject: [PATCH 354/637] fix pylint --- metagpt/roles/role.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index ad3c44ac1..47a4f45a7 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -481,6 +481,8 @@ class Role(SerializationMixin, ContextMixin, BaseModel): rsp = await self._act_by_order() elif self.rc.react_mode == RoleReactMode.PLAN_AND_ACT: rsp = await self._plan_and_act() + else: + raise ValueError(f"Unsupported react mode: {self.rc.react_mode}") self._set_state(state=-1) # current reaction is complete, reset state to -1 and todo back to None return rsp From cd9798643f6b550674c20ef029cad4044e79a4f9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Tue, 16 Jan 2024 10:08:34 +0800 Subject: [PATCH 355/637] fixbug: external dependency --- tests/metagpt/actions/test_summarize_code.py | 49 +++++++++++++------- 1 file changed, 33 insertions(+), 16 deletions(-) diff --git a/tests/metagpt/actions/test_summarize_code.py b/tests/metagpt/actions/test_summarize_code.py index 88d432b5e..e73192406 100644 --- a/tests/metagpt/actions/test_summarize_code.py +++ b/tests/metagpt/actions/test_summarize_code.py @@ -6,12 +6,17 @@ @File : test_summarize_code.py @Modifiled By: mashenquan, 2023-12-6. Unit test for summarize_code.py """ +import shutil +import uuid +from pathlib import Path + import pytest from metagpt.actions.summarize_code import SummarizeCode -from metagpt.context import CONTEXT +from metagpt.context import Context from metagpt.logs import logger from metagpt.schema import CodeSummarizeContext +from metagpt.utils.git_repository import GitRepository from metagpt.utils.project_repo import ProjectRepo DESIGN_CONTENT = """ @@ -177,22 +182,34 @@ class Snake: @pytest.mark.asyncio async def test_summarize_code(): - CONTEXT.src_workspace = CONTEXT.git_repo.workdir / "src" - project_repo = ProjectRepo(CONTEXT.git_repo) - await project_repo.docs.system_design.save(filename="1.json", content=DESIGN_CONTENT) - await project_repo.docs.task.save(filename="1.json", content=TASK_CONTENT) - await project_repo.with_src_path(CONTEXT.src_workspace).srcs.save(filename="food.py", content=FOOD_PY) - assert project_repo.srcs.workdir == CONTEXT.src_workspace - await project_repo.srcs.save(filename="game.py", content=GAME_PY) - await project_repo.srcs.save(filename="main.py", content=MAIN_PY) - await project_repo.srcs.save(filename="snake.py", content=SNAKE_PY) + git_dir = Path(__file__).parent / f"unittest/{uuid.uuid4().hex}" + git_dir.mkdir(parents=True, exist_ok=True) - all_files = project_repo.srcs.all_files - ctx = CodeSummarizeContext(design_filename="1.json", task_filename="1.json", codes_filenames=all_files) - action = SummarizeCode(i_context=ctx) - rsp = await action.run() - assert rsp - logger.info(rsp) + try: + context = Context() + context.git_repo = GitRepository(local_path=git_dir) + context.src_workspace = context.git_repo.workdir / "src" + project_repo = ProjectRepo(context.git_repo) + await project_repo.docs.system_design.save(filename="1.json", content=DESIGN_CONTENT) + await project_repo.docs.task.save(filename="1.json", content=TASK_CONTENT) + await project_repo.with_src_path(context.src_workspace).srcs.save(filename="food.py", content=FOOD_PY) + assert project_repo.srcs.workdir == context.src_workspace + await project_repo.srcs.save(filename="game.py", content=GAME_PY) + await project_repo.srcs.save(filename="main.py", content=MAIN_PY) + await project_repo.srcs.save(filename="snake.py", content=SNAKE_PY) + + all_files = project_repo.srcs.all_files + summarization_context = CodeSummarizeContext( + design_filename="1.json", task_filename="1.json", codes_filenames=all_files + ) + action = SummarizeCode(context=context, i_context=summarization_context) + rsp = await action.run() + assert rsp + logger.info(rsp) + except Exception as e: + assert not e + finally: + shutil.rmtree(git_dir) if __name__ == "__main__": From 29d8326c06899535bb2d8953246aa30466b8f72f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Wed, 10 Jan 2024 21:13:36 +0800 Subject: [PATCH 356/637] feat: +ver feat: Moderation + llm arg feat: +log --- metagpt/document_store/faiss_store.py | 6 +++++- metagpt/tools/moderation.py | 4 ++-- requirements.txt | 2 +- setup.py | 2 +- 4 files changed, 9 insertions(+), 5 deletions(-) diff --git a/metagpt/document_store/faiss_store.py b/metagpt/document_store/faiss_store.py index 1271f1c23..6f97141c2 100644 --- a/metagpt/document_store/faiss_store.py +++ b/metagpt/document_store/faiss_store.py @@ -40,7 +40,11 @@ class FaissStore(LocalStore): return FAISS.load_local(self.raw_data_path.parent, self.embedding, self.fname) def _write(self, docs, metadatas): - store = FAISS.from_texts(docs, self.embedding, metadatas=metadatas) + try: + store = FAISS.from_texts(docs, self.embedding, metadatas=metadatas) + except Exception as e: + logger.error(f"Failed to write. error: {e}") + raise e return store def persist(self): diff --git a/metagpt/tools/moderation.py b/metagpt/tools/moderation.py index cda164ec5..8effc0e8b 100644 --- a/metagpt/tools/moderation.py +++ b/metagpt/tools/moderation.py @@ -11,8 +11,8 @@ from metagpt.llm import LLM class Moderation: - def __init__(self): - self.llm = LLM() + def __init__(self, llm=None): + self.llm = llm or LLM() def handle_moderation_results(self, results): resp = [] diff --git a/requirements.txt b/requirements.txt index 0a54236f0..f8e4d2585 100644 --- a/requirements.txt +++ b/requirements.txt @@ -14,7 +14,7 @@ lancedb==0.4.0 langchain==0.0.352 loguru==0.6.0 meilisearch==0.21.0 -numpy==1.24.3 +numpy>=1.24.3 openai==1.6.0 openpyxl beautifulsoup4==4.12.2 diff --git a/setup.py b/setup.py index d997b5f62..ea84fe299 100644 --- a/setup.py +++ b/setup.py @@ -57,7 +57,7 @@ extras_require["dev"] = (["pylint~=3.0.3", "black~=23.3.0", "isort~=5.12.0", "pr setup( name="metagpt", - version="0.6.3", + version="0.6.4", description="The Multi-Agent Framework", long_description=long_description, long_description_content_type="text/markdown", From 29fd7117ef5cf187506e53727437881068118113 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 16 Jan 2024 11:57:08 +0800 Subject: [PATCH 357/637] update module. --- metagpt/tools/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index 2f8941fdb..73de03156 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -78,7 +78,7 @@ TOOL_TYPE_MAPPINGS = { ), "scrape_web": ToolType( name="scrape_web", - module=str(TOOL_LIBS_PATH / "scrape_web"), + module="metagpt.tools.functions.libs.scrape_web.scrape_web", desc="Scrape data from web page.", usage_prompt="", ), From eef77d1628bf48657ecf307ba69ab21f21e2d71e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 16 Jan 2024 12:29:52 +0800 Subject: [PATCH 358/637] display markdown. --- metagpt/actions/execute_code.py | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index fb0ecd893..9fadd0acd 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -18,6 +18,8 @@ from nbformat import NotebookNode from nbformat.v4 import new_code_cell, new_output, new_markdown_cell from rich.console import Console from rich.syntax import Syntax +from rich.markdown import Markdown + from metagpt.actions import Action from metagpt.logs import logger @@ -97,8 +99,12 @@ class ExecutePyCode(ExecuteCode, Action): def _display(self, code, language: str = "python"): if language == "python": code = Syntax(code, "python", theme="paraiso-dark", line_numbers=True) - self.console.print("\n") self.console.print(code) + elif language == "markdown": + code = Markdown(code, inline_code_theme="paraiso-dark") + self.console.print(code) + else: + raise ValueError(f"Only support for python, markdown, but got {language}") def add_output_to_cell(self, cell, output): if "outputs" not in cell: @@ -221,10 +227,12 @@ class ExecutePyCode(ExecuteCode, Action): # code success outputs = self.parse_outputs(self.nb.cells[-1].outputs) return truncate(remove_escape_and_color_codes(outputs), is_success=success) - else: + elif language == 'markdown': # markdown self.add_markdown_cell(code) return code, True + else: + raise ValueError(f"Only support for language: python, markdown, but got {language}, ") def truncate(result: str, keep_len: int = 2000, is_success: bool = True): From 95ce190f32a88d59455bb8bb982b64dd3a5018c0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 16 Jan 2024 14:30:07 +0800 Subject: [PATCH 359/637] feature: display markdown content. --- metagpt/actions/execute_code.py | 36 ++++++++++++++++++++++++++++++--- 1 file changed, 33 insertions(+), 3 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 9fadd0acd..6d9135ec3 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -19,7 +19,10 @@ from nbformat.v4 import new_code_cell, new_output, new_markdown_cell from rich.console import Console from rich.syntax import Syntax from rich.markdown import Markdown - +from rich.panel import Panel +from rich.box import MINIMAL +from rich.live import Live +from rich.console import Group from metagpt.actions import Action from metagpt.logs import logger @@ -101,8 +104,7 @@ class ExecutePyCode(ExecuteCode, Action): code = Syntax(code, "python", theme="paraiso-dark", line_numbers=True) self.console.print(code) elif language == "markdown": - code = Markdown(code, inline_code_theme="paraiso-dark") - self.console.print(code) + _display_markdown(code) else: raise ValueError(f"Only support for python, markdown, but got {language}") @@ -265,3 +267,31 @@ def remove_escape_and_color_codes(input_str): pattern = re.compile(r"\x1b\[[0-9;]*[mK]") result = pattern.sub("", input_str) return result + + +def _display_markdown(content: str): + # 使用正则表达式逐个匹配代码块 + matches = re.finditer(r'```(.+?)```', content, re.DOTALL) + start_index = 0 + content_panels = [] + # 逐个打印匹配到的文本和代码 + for match in matches: + text_content = content[start_index:match.start()].strip() + code_content = match.group(0).strip()[3:-3] # Remove triple backticks + + if text_content: + content_panels.append(Panel(Markdown(text_content), box=MINIMAL)) + + if code_content: + content_panels.append(Panel(Markdown(f"```{code_content}"), box=MINIMAL)) + start_index = match.end() + + # 打印剩余文本(如果有) + remaining_text = content[start_index:].strip() + if remaining_text: + content_panels.append(Panel(Markdown(remaining_text), box=MINIMAL)) + + # 在Live模式中显示所有Panel + with Live(auto_refresh=False, console=Console(), vertical_overflow="visible") as live: + live.update(Group(*content_panels)) + live.refresh() From 43558c208ebd1dbf6bf980f0a271abaed4557a7d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 16 Jan 2024 15:03:12 +0800 Subject: [PATCH 360/637] doc: add ipywidgets. --- requirements.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/requirements.txt b/requirements.txt index 7ef6d884e..016c2f5d5 100644 --- a/requirements.txt +++ b/requirements.txt @@ -65,3 +65,4 @@ networkx~=3.2.1 google-generativeai==0.3.2 # playwright==1.40.0 # playwright extras require anytree +ipywidgets==8.1.1 From f4e39a462ceb99dee6dff5bd523ba0c5f72695f5 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 16 Jan 2024 15:42:48 +0800 Subject: [PATCH 361/637] 1. remove INC prompt in some ActionNode 2. replace Code archive from data to tests/data/incremental_dev_project 3. update test case for ActionNode --- metagpt/actions/design_api_an.py | 47 --- metagpt/actions/project_management_an.py | 60 +--- metagpt/actions/write_code.py | 20 ++ metagpt/actions/write_code_guideline_an.py | 3 +- metagpt/actions/write_prd_an.py | 27 -- metagpt/roles/engineer.py | 14 +- .../data/incremental_dev_project}/Gomoku.zip | Bin .../dice_simulator_new.zip | Bin .../number_guessing_game.zip | Bin .../incremental_dev_project}/pygame_2048.zip | Bin tests/data/incremental_dev_project/readme.md | 3 + .../simple_add_calculator.zip | Bin .../incremental_dev_project}/snake_game.zip | Bin .../incremental_dev_project}/word_cloud.zip | Bin tests/metagpt/actions/test_design_api_an.py | 3 + .../actions/test_project_management_an.py | 4 + .../actions/test_write_code_guideline_an.py | 91 ++++- tests/metagpt/actions/test_write_prd_an.py | 4 + tests/metagpt/test_incremental_dev.py | 338 ++++-------------- 19 files changed, 194 insertions(+), 420 deletions(-) rename {data => tests/data/incremental_dev_project}/Gomoku.zip (100%) rename {data => tests/data/incremental_dev_project}/dice_simulator_new.zip (100%) rename {data => tests/data/incremental_dev_project}/number_guessing_game.zip (100%) rename {data => tests/data/incremental_dev_project}/pygame_2048.zip (100%) create mode 100644 tests/data/incremental_dev_project/readme.md rename {data => tests/data/incremental_dev_project}/simple_add_calculator.zip (100%) rename {data => tests/data/incremental_dev_project}/snake_game.zip (100%) rename {data => tests/data/incremental_dev_project}/word_cloud.zip (100%) diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index ee1941350..3890e656a 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -18,14 +18,6 @@ IMPLEMENTATION_APPROACH = ActionNode( example="We will ...", ) -INCREMENTAL_IMPLEMENTATION_APPROACH = ActionNode( - key="Incremental Implementation approach", - expected_type=str, - instruction="Analyze the challenging aspects of the requirements and select a suitable open-source framework. " - "Outline the incremental steps involved in the implementation process with the detailed strategies.", - example="we will ...", -) - REFINED_IMPLEMENTATION_APPROACH = ActionNode( key="Refined Implementation Approach", expected_type=str, @@ -63,16 +55,6 @@ DATA_STRUCTURES_AND_INTERFACES = ActionNode( example=MMC1, ) -INCREMENTAL_DATA_STRUCTURES_AND_INTERFACES = ActionNode( - key="Incremental Data structures and interfaces", - expected_type=str, - instruction="Extend the existing mermaid classDiagram code syntax to incorporate new classes, " - "methods (including __init__), and functions with precise type annotations. Clearly delineate additional " - "relationships between classes, maintaining adherence to PEP8 standards. Enhance the level of detail in data " - "structures, ensuring a comprehensive API design that seamlessly integrates with the existing structure.", - example=MMC1_REFINE, -) - REFINED_DATA_STRUCTURES_AND_INTERFACES = ActionNode( key="Refined Data Structures and Interfaces", expected_type=str, @@ -108,32 +90,6 @@ ANYTHING_UNCLEAR = ActionNode( example="Clarification needed on third-party API integration, ...", ) -INC_DESIGN_CONTEXT = """ -## Legacy Content -{old_design} - -## New Requirements -{requirements} - -## PRD Increment Content -{prd_increment} -""" - -MERGE_DESIGN_CONTEXT = """ -Role: You are a professional Architect tasked with overseeing incremental development. -Based on new requirements, review and refine the system design. Integrate existing architecture with incremental design changes, ensuring the refined design encompasses all architectural elements, enhancements, and adjustments. Retain content unrelated to incremental development needs for coherence and clarity. - -# Context -## New Requirements -{requirements} - -## Legacy Content -{old_design} - -## Design Increment Content -{design_increment} -""" - NODES = [ IMPLEMENTATION_APPROACH, # PROJECT_NAME, @@ -143,8 +99,6 @@ NODES = [ ANYTHING_UNCLEAR, ] -INC_NODES = [INCREMENTAL_IMPLEMENTATION_APPROACH, INCREMENTAL_DATA_STRUCTURES_AND_INTERFACES, REFINED_PROGRAM_CALL_FLOW] - REFINE_NODES = [ REFINED_IMPLEMENTATION_APPROACH, REFINED_FILE_LIST, @@ -154,7 +108,6 @@ REFINE_NODES = [ ] DESIGN_API_NODE = ActionNode.from_children("DesignAPI", NODES) -INCREMENTAL_DESIGN_NODES = ActionNode.from_children("Incremental_Design_API", INC_NODES) REFINED_DESIGN_NODES = ActionNode.from_children("Refined_Design_API", REFINE_NODES) diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index 559c5ef8e..8b0196707 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -35,24 +35,12 @@ LOGIC_ANALYSIS = ActionNode( ], ) -INCREMENTAL_LOGIC_ANALYSIS = ActionNode( - key="Incremental Logic Analysis", - expected_type=List[List[str]], - instruction="Provide a list of files with the classes/methods/functions to be implemented or modified " - "incrementally. Include thorough dependency analysis, consider potential impacts on existing code, and document" - " necessary imports.", - example=[ - ["new_feature.py", "Introduces NewFeature class and related functions"], - ["utils.py", "Modifies existing utility functions to support incremental changes"], - ], -) - REFINED_LOGIC_ANALYSIS = ActionNode( key="Refined Logic Analysis", expected_type=List[List[str]], instruction="Review and refine the logic analysis by merging the Legacy Content and Incremental Content. " "Provide a comprehensive list of files with classes/methods/functions to be implemented or modified incrementally. " - "Include thorough dependency analysis, consider potential impacts on existing code, and document necessary imports.", + "Include dependency analysis, consider potential impacts on existing code, and document necessary imports.", example=[ ["game.py", "Contains Game class and ... functions"], ["main.py", "Contains main function, from game import Game"], @@ -68,15 +56,6 @@ TASK_LIST = ActionNode( example=["game.py", "main.py"], ) -INCREMENTAL_TASK_LIST = ActionNode( - key="Incremental Task list", - expected_type=List[str], - instruction="Break down the incremental development tasks into a prioritized list of filenames." - "Organize the tasks based on dependency order, ensuring a systematic and efficient implementation." - "Only output filename! Do not include comments in the list ", - example=["new_feature.py", "main.py"], -) - REFINED_TASK_LIST = ActionNode( key="Refined Task list", expected_type=List[str], @@ -101,14 +80,6 @@ SHARED_KNOWLEDGE = ActionNode( example="`game.py` contains functions shared across the project.", ) -INCREMENTAL_SHARED_KNOWLEDGE = ActionNode( - key="Incremental Shared Knowledge", - expected_type=str, - instruction="Document any new shared knowledge generated during incremental development. This includes common " - "utility functions, configuration variables, or any information vital for team collaboration.", - example="`new_module.py` introduces shared utility functions for improved code reusability.", -) - REFINED_SHARED_KNOWLEDGE = ActionNode( key="Refined Shared Knowledge", expected_type=str, @@ -126,32 +97,6 @@ ANYTHING_UNCLEAR_PM = ActionNode( example="Clarification needed on how to start and initialize third-party libraries.", ) -INC_PM_CONTEXT = """ -### Legacy Content -{old_tasks} - -### New Requirements -{requirements} - -### Design Increment Content -{design_increment} -""" - -MERGE_PM_CONTEXT = """ -Role: You are a professional Project Manager tasked with overseeing incremental development. -Based on New Requirements, refine the project context to account for incremental development. Ensure the context offers a comprehensive overview of the project's evolving scope, covering both legacy content and incremental content. Retain any content unrelated to incremental development. - -# Context -## New Requirements -{requirements} - -## Legacy Content -{old_tasks} - -## Increment Content -{tasks_increment} -""" - NODES = [ REQUIRED_PYTHON_PACKAGES, REQUIRED_OTHER_LANGUAGE_PACKAGES, @@ -162,8 +107,6 @@ NODES = [ ANYTHING_UNCLEAR_PM, ] -INC_NODES = [INCREMENTAL_LOGIC_ANALYSIS, INCREMENTAL_TASK_LIST, INCREMENTAL_SHARED_KNOWLEDGE] - REFINE_NODES = [ REQUIRED_PYTHON_PACKAGES, REQUIRED_OTHER_LANGUAGE_PACKAGES, @@ -175,7 +118,6 @@ REFINE_NODES = [ ] PM_NODE = ActionNode.from_children("PM_NODE", NODES) -INCREMENTAL_PM_NODES = ActionNode.from_children("Incremental_PM_NODES", INC_NODES) REFINED_PM_NODES = ActionNode.from_children("Refined_PM_NODES", REFINE_NODES) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index f92b72f7f..ce0e2fe3b 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -157,6 +157,24 @@ class WriteCode(Action): @staticmethod async def get_codes(task_doc, exclude, mode="normal") -> str: + """ + Get code snippets based on different modes. + + Attributes: + task_doc (Document): Document object of the task file. + exclude (str): Specifies the filename to be excluded from the code snippets. + mode (str): Specifies the mode, either "normal" or "guide" (default is "normal"). + + Returns: + str: Code snippets. + + Description: + If mode is set to "normal", it returns code snippets for the regular coding phase, + i.e., all the code generated before writing the current file. + + If mode is set to "guide", it returns code snippets for incremental development, + building upon the existing code in the "normal" mode and adding code for the current file's older versions. + """ if not task_doc: return "" if not task_doc.content: @@ -173,6 +191,8 @@ class WriteCode(Action): union_files_list = list(set(src_files) | set(old_files)) for filename in union_files_list: if filename == exclude: + # Exclude unnecessary code to maintain a clean and focused main.py file, ensuring only relevant and + # essential functionality is included for the project’s requirements if filename in old_files and filename != "main.py": # Use legacy code doc = await old_file_repo.get(filename=filename) diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index 528b4e8f3..e4afb393d 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -9,6 +9,7 @@ import asyncio from metagpt.actions.action import Action from metagpt.actions.action_node import ActionNode +from metagpt.logs import logger GUIDELINES_AND_INCREMENTAL_CHANGE = ActionNode( key="Guidelines and Incremental Change", @@ -455,7 +456,7 @@ async def main(): write_code_guideline = WriteCodeGuideline() node = await write_code_guideline.run(CODE_GUIDELINE_CONTEXT_EXAMPLE) guideline = node.instruct_content.model_dump_json() - print(guideline) + logger.info(guideline) if __name__ == "__main__": diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index 7004bfffc..4830076e3 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -130,14 +130,6 @@ REQUIREMENT_ANALYSIS = ActionNode( example="", ) -INCREMENTAL_REQUIREMENT_ANALYSIS = ActionNode( - key="Incremental Requirement Analysis", - expected_type=List[str], - instruction="Propose the comprehensive incremental development requirement analysis on new features and enhanced " - "features for New Requirements.", - example=["Require add/update/modify ..."], -) - REFINED_REQUIREMENT_ANALYSIS = ActionNode( key="Refined Requirement Analysis", expected_type=List[str], @@ -194,22 +186,6 @@ REASON = ActionNode( key="reason", expected_type=str, instruction="Explain the reasoning process from question to answer", example="..." ) - -INCREMENTAL_PRD_CONTEXT = """ -Role: You are a professional Product Manager tasked with overseeing incremental development. -Based on New Requirements, output a New PRD that seamlessly integrates both the Legacy Content and the Incremental Content. Ensure the resulting document captures the complete scope of features, enhancements, and retain content unrelated to incremental development needs for coherence and clarity. - -# Context -## New Requirements -{requirements} - -## Legacy Content -{old_prd} - -## PRD Incremental Content -{prd_increment} -""" - REFINE_PRD_TEMPLATE = """ ### Project Name {project_name} @@ -255,11 +231,8 @@ REFINE_NODES = [ ANYTHING_UNCLEAR, ] -INCREMENT_PRD_NODES = [INCREMENTAL_REQUIREMENT_ANALYSIS, REQUIREMENT_POOL] - WRITE_PRD_NODE = ActionNode.from_children("WritePRD", NODES) REFINE_PRD_NODE = ActionNode.from_children("RefinePRD", REFINE_NODES) -INCREMENTAL_PRD_NODE = ActionNode.from_children("IncrementalPRD", INCREMENT_PRD_NODES) WP_ISSUE_TYPE_NODE = ActionNode.from_children("WP_ISSUE_TYPE", [ISSUE_TYPE, REASON]) WP_IS_RELATIVE_NODE = ActionNode.from_children("WP_IS_RELATIVE", [IS_RELATIVE, REASON]) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 772cf0944..a6b51bd00 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -119,18 +119,9 @@ class Engineer(Role): self._init_action_system_message(action) coding_context = await action.run(guideline=guideline) - # Get dependencies + dependencies = {coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path} if guideline: - dependencies = { - coding_context.design_doc.root_relative_path, - coding_context.task_doc.root_relative_path, - "code_guideline.json", - } - else: - dependencies = { - coding_context.design_doc.root_relative_path, - coding_context.task_doc.root_relative_path, - } + dependencies.add("code_guideline.json") await src_file_repo.save( coding_context.filename, dependencies=dependencies, @@ -344,6 +335,7 @@ class Engineer(Role): return self.next_todo_action async def _write_code_guideline(self): + """Write some guidelines that guides subsequent WriteCode and WriteCodeReview""" logger.info("Writing code guideline..") user_requirement = str(self.rc.memory.get_by_role("Human")[0]) diff --git a/data/Gomoku.zip b/tests/data/incremental_dev_project/Gomoku.zip similarity index 100% rename from data/Gomoku.zip rename to tests/data/incremental_dev_project/Gomoku.zip diff --git a/data/dice_simulator_new.zip b/tests/data/incremental_dev_project/dice_simulator_new.zip similarity index 100% rename from data/dice_simulator_new.zip rename to tests/data/incremental_dev_project/dice_simulator_new.zip diff --git a/data/number_guessing_game.zip b/tests/data/incremental_dev_project/number_guessing_game.zip similarity index 100% rename from data/number_guessing_game.zip rename to tests/data/incremental_dev_project/number_guessing_game.zip diff --git a/data/pygame_2048.zip b/tests/data/incremental_dev_project/pygame_2048.zip similarity index 100% rename from data/pygame_2048.zip rename to tests/data/incremental_dev_project/pygame_2048.zip diff --git a/tests/data/incremental_dev_project/readme.md b/tests/data/incremental_dev_project/readme.md new file mode 100644 index 000000000..231589028 --- /dev/null +++ b/tests/data/incremental_dev_project/readme.md @@ -0,0 +1,3 @@ +# Code archive + +This folder contains a compressed package for the test_incremental_dev.py file, which is used to demonstrate the process of incremental development. diff --git a/data/simple_add_calculator.zip b/tests/data/incremental_dev_project/simple_add_calculator.zip similarity index 100% rename from data/simple_add_calculator.zip rename to tests/data/incremental_dev_project/simple_add_calculator.zip diff --git a/data/snake_game.zip b/tests/data/incremental_dev_project/snake_game.zip similarity index 100% rename from data/snake_game.zip rename to tests/data/incremental_dev_project/snake_game.zip diff --git a/data/word_cloud.zip b/tests/data/incremental_dev_project/word_cloud.zip similarity index 100% rename from data/word_cloud.zip rename to tests/data/incremental_dev_project/word_cloud.zip diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index 5b016ecce..c729c047c 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -101,4 +101,7 @@ def llm(): async def test_write_design_an(): node = await REFINED_DESIGN_NODES.fill(CONTEXT, llm) assert node.instruct_content + assert "Refined Implementation Approach" in node.instruct_content.model_dump_json() + assert "Refined File List" in node.instruct_content.model_dump_json() assert "Refined Data Structures and Interfaces" in node.instruct_content.model_dump_json() + assert "Refined Program call flow" in node.instruct_content.model_dump_json() diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index e0c1381ec..68d73a3d9 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -141,4 +141,8 @@ def llm(): async def test_project_management_an(llm): node = await REFINED_PM_NODES.fill(CONTEXT, llm) assert node.instruct_content + assert "Required Python Packages" in node.instruct_content.model_dump_json() + assert "Required Other Language Packages" in node.instruct_content.model_dump_json() assert "Refined Logic Analysis" in node.instruct_content.model_dump_json() + assert "Refined Task List" in node.instruct_content.model_dump_json() + assert "Refined Shared Knowledge" in node.instruct_content.model_dump_json() diff --git a/tests/metagpt/actions/test_write_code_guideline_an.py b/tests/metagpt/actions/test_write_code_guideline_an.py index f8a1ae626..f3fba26cc 100644 --- a/tests/metagpt/actions/test_write_code_guideline_an.py +++ b/tests/metagpt/actions/test_write_code_guideline_an.py @@ -185,9 +185,94 @@ if __name__ == "__main__": CalculatorApp.main() ```""" -INCREMENTAL_CHANGE_EXAMPLE = """ +GUIDELINES_AND_INCREMENTAL_CHANGE_EXAMPLE = """ { - "Incremental Change": "- operations.py: Implement the Operations class with a method to perform the requested arithmetic operation. This class will be used by the Calculator class to execute the operations.\n```python\n## operations.py\nclass Operations:\n @staticmethod\n def perform_operation(operation: str, number1: float, number2: float) -> float:\n if operation == 'add':\n return number1 + number2\n elif operation == 'subtract':\n return number1 - number2\n elif operation == 'multiply':\n return number1 * number2\n elif operation == 'divide':\n if number2 == 0:\n raise ValueError('Cannot divide by zero')\n return number1 / number2\n else:\n raise ValueError('Invalid operation')\n```\n\n- calculator.py: Extend the Calculator class to include methods for subtraction, multiplication, and division. These methods will utilize the Operations class to perform the actual calculations.\n```python\n## calculator.py\nfrom operations import Operations\nclass Calculator:\n ...\n def subtract(self, number1: float, number2: float) -> float:\n return Operations.perform_operation('subtract', number1, number2)\n\n def multiply(self, number1: float, number2: float) -> float:\n return Operations.perform_operation('multiply', number1, number2)\n\n def divide(self, number1: float, number2: float) -> float:\n return Operations.perform_operation('divide', number1, number2)\n```\n\n- interface.py: Update the Interface class to include buttons for subtraction, multiplication, and division, and link them to the corresponding methods in the Calculator class. Also, handle the display of errors such as division by zero.\n```python\n## interface.py\nimport tkinter as tk\nfrom tkinter import messagebox\nfrom calculator import Calculator\n...\nclass Interface:\n ...\n def create_widgets(self):\n ...\n self.subtract_button = tk.Button(self.root, text='-', command=self.subtract, font=('Arial', 18))\n self.subtract_button.grid(row=3, column=0, sticky='nsew')\n\n self.multiply_button = tk.Button(self.root, text='*', command=self.multiply, font=('Arial', 18))\n self.multiply_button.grid(row=3, column=1, sticky='nsew')\n\n self.divide_button = tk.Button(self.root, text='/', command=self.divide, font=('Arial', 18))\n self.divide_button.grid(row=3, column=2, sticky='nsew')\n ...\n\n def subtract(self):\n number1, number2 = self.get_input()\n if number1 is not None and number2 is not None:\n result = self.calculator.subtract(number1, number2)\n self.display_result(result)\n\n def multiply(self):\n number1, number2 = self.get_input()\n if number1 is not None and number2 is not None:\n result = self.calculator.multiply(number1, number2)\n self.display_result(result)\n\n def divide(self):\n number1, number2 = self.get_input()\n if number1 is not None and number2 is not None:\n try:\n result = self.calculator.divide(number1, number2)\n except ValueError as e:\n self.show_error(str(e))\n return\n self.display_result(result)\n```\n\n- main.py: No changes needed in main.py as it serves as the entry point and will run the updated Interface class.\n```python\n## main.py\nfrom interface import Interface\n...\n```\n\nNote: Ensure that the new operations buttons in the Interface class are properly arranged and that the grid layout is adjusted accordingly. Also, make sure to import the messagebox module from tkinter for error handling." + "Guidelines and Incremental Change": " +1. Guideline for calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Additionally, implement robust error handling for the division operation to mitigate potential issues related to division by zero. +```python +class Calculator: + self.result = number1 + number2 + return self.result + +- def sub(self, number1, number2) -> float: ++ def subtract(self, number1: float, number2: float) -> float: ++ ''' ++ Subtracts the second number from the first and returns the result. ++ ++ Args: ++ number1 (float): The number to be subtracted from. ++ number2 (float): The number to subtract. ++ ++ Returns: ++ float: The difference of number1 and number2. ++ ''' ++ self.result = number1 - number2 ++ return self.result ++ + def multiply(self, number1: float, number2: float) -> float: +- pass ++ ''' ++ Multiplies two numbers and returns the result. ++ ++ Args: ++ number1 (float): The first number to multiply. ++ number2 (float): The second number to multiply. ++ ++ Returns: ++ float: The product of number1 and number2. ++ ''' ++ self.result = number1 * number2 ++ return self.result ++ + def divide(self, number1: float, number2: float) -> float: +- pass ++ ''' ++ ValueError: If the second number is zero. ++ ''' ++ if number2 == 0: ++ raise ValueError('Cannot divide by zero') ++ self.result = number1 / number2 ++ return self.result ++ +- def reset_result(self): ++ def clear(self): ++ if self.result != 0.0: ++ print("Result is not zero, clearing...") ++ else: ++ print("Result is already zero, no need to clear.") ++ + self.result = 0.0 +``` + +2. Guideline for main.py: Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Then, ensure seamless integration with the overall application architecture and maintain consistency with coding standards. +```python +def add_numbers(): + result = calculator.add_numbers(num1, num2) + return jsonify({'result': result}), 200 + +-# TODO: Implement subtraction, multiplication, and division operations ++@app.route('/subtract_numbers', methods=['POST']) ++def subtract_numbers(): ++ data = request.get_json() ++ num1 = data.get('num1', 0) ++ num2 = data.get('num2', 0) ++ result = calculator.subtract_numbers(num1, num2) ++ return jsonify({'result': result}), 200 ++ ++@app.route('/multiply_numbers', methods=['POST']) ++def multiply_numbers(): ++ data = request.get_json() ++ num1 = data.get('num1', 0) ++ num2 = data.get('num2', 0) ++ try: ++ result = calculator.divide_numbers(num1, num2) ++ except ValueError as e: ++ return jsonify({'error': str(e)}), 400 ++ return jsonify({'result': result}), 200 ++ + if __name__ == '__main__': + app.run() +```" } """ @@ -207,7 +292,7 @@ async def test_write_code_guideline_an(): async def test_refine_code(): prompt = REFINED_CODE_TEMPLATE.format( requirement=REQUIREMENT_EXAMPLE, - guideline=INCREMENTAL_CHANGE_EXAMPLE, + guideline=GUIDELINES_AND_INCREMENTAL_CHANGE_EXAMPLE, design=DESIGN_EXAMPLE, tasks=TASKS_EXAMPLE, code=REFINE_CODE_SCRIPT_EXAMPLE, diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index 79cd913cd..693f4924d 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -85,4 +85,8 @@ def llm(): async def test_write_prd_an(llm): node = await REFINE_PRD_NODE.fill(CONTEXT, llm) assert node.instruct_content + assert "Refined Requirements" in node.instruct_content.model_dump_json() + assert "Refined Product Goals" in node.instruct_content.model_dump_json() + assert "Refined User Stories" in node.instruct_content.model_dump_json() + assert "Refined Requirement Analysis" in node.instruct_content.model_dump_json() assert "Refined Requirement Pool" in node.instruct_content.model_dump_json() diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 44478adc9..8890137b0 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -11,24 +11,73 @@ import subprocess import pytest from typer.testing import CliRunner -from metagpt.const import DATA_PATH +from metagpt.const import TEST_DATA_PATH from metagpt.logs import logger from metagpt.startup import app runner = CliRunner() +IDEAS = [ + "Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator", + "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove.", + "Add an AI opponent with fixed difficulty levels. Currently, the game only allows players to compete against themselves. Implement an AI algorithm that can playing with player. This will provide a more engaging and challenging experience for players.", + "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores", + "Add functionality to view the history of scores and perform statistical analysis on them. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores and display the statistical analysis results of the current score", + "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8", + "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores", + "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game.", + "Incremental Idea Gradually increase the speed of the snake as the game progresses. In the current version of the game, the snake’s speed remains constant throughout the gameplay. Implement a feature where the snake’s speed gradually increases over time, making the game more challenging and intense as the player progresses.", + "Introduce power-ups and obstacles to the game. The current version of the game only involves eating food and growing the snake. Add new elements such as power-ups that can enhance the snake’s speed or make it invincible for a short duration. At the same time, introduce obstacles like walls or enemies that the snake must avoid or overcome to continue growing.", +] -def test_refined_simple_calculator(): - project_path = f"{DATA_PATH}/simple_add_calculator" - check_or_create_base_tag(project_path) +PROJECT_NAMES = [ + "calculator", + "word_cloud", + "Gomoku", + "dice_simulator_new", + "dice_simulator_new", + "pygame_2048", + "pygame_2048", + "pygame_2048", + "snake_game", + "snake_game", +] - args = [ - "Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) + +def test_refined_calculator(): + result = get_incremental_dev_result(IDEAS[0], PROJECT_NAMES[0]) + log_and_check_result(result) + + +def test_refined_word_cloud(): + result = get_incremental_dev_result(IDEAS[1], PROJECT_NAMES[1]) + log_and_check_result(result) + + +def test_refined_gomoku(): + result = get_incremental_dev_result(IDEAS[2], PROJECT_NAMES[2]) + log_and_check_result(result) + + +def test_refined_dice_simulator_new(): + for idea, project_name in zip(IDEAS[3:5], PROJECT_NAMES[3:5]): + result = get_incremental_dev_result(idea, project_name) + log_and_check_result(result) + + +def test_refined_pygame_2048(): + for idea, project_name in zip(IDEAS[5:8], PROJECT_NAMES[5:8]): + result = get_incremental_dev_result(idea, project_name) + log_and_check_result(result) + + +def test_refined_snake_game(): + for idea, project_name in zip(IDEAS[8:10], PROJECT_NAMES[8:10]): + result = get_incremental_dev_result(idea, project_name) + log_and_check_result(result) + + +def log_and_check_result(result): logger.info(result) logger.info(result.output) if "Aborting" in result.output: @@ -46,274 +95,19 @@ def test_refined_simple_calculator(): raise e -def test_refined_number_guessing_game(): - project_path = f"{DATA_PATH}/number_guessing_game" +def get_incremental_dev_result(idea, project_name): + project_path = TEST_DATA_PATH / "incremental_dev_project" / project_name + if not os.path.exists(project_path): + raise Exception(f"Project {project_name} not exists") check_or_create_base_tag(project_path) - args = [ - "Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal", + idea, "--inc", "--project-path", project_path, ] result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine"], check=True) - except subprocess.CalledProcessError as e: - raise e - - -def test_refined_word_cloud(): - project_path = f"{DATA_PATH}/word_cloud" - check_or_create_base_tag(project_path) - - args = [ - "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove.", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine"], check=True) - except subprocess.CalledProcessError as e: - raise e - - -def test_refined_gomoku(): - project_path = f"{DATA_PATH}/Gomoku" - check_or_create_base_tag(project_path) - - args = [ - "Add an AI opponent with fixed difficulty levels. Currently, the game only allows players to compete against themselves. Implement an AI algorithm that can playing with player. This will provide a more engaging and challenging experience for players.", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine"], check=True) - except subprocess.CalledProcessError as e: - raise e - - -def test_refined_dice_simulator_1(): - project_path = f"{DATA_PATH}/dice_simulator_new" - check_or_create_base_tag(project_path) - - args = [ - "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine_1"], check=True) - except subprocess.CalledProcessError as e: - raise e - - -def test_refined_dice_simulator_2(): - project_path = f"{DATA_PATH}/dice_simulator_new" - check_or_create_base_tag(project_path) - - args = [ - "Add functionality to view the history of scores and perform statistical analysis on them. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores and display the statistical analysis results of the current score", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine_2"], check=True) - except subprocess.CalledProcessError as e: - raise e - - -def test_refined_pygame_2048_1(): - project_path = f"{DATA_PATH}/pygame_2048" - check_or_create_base_tag(project_path) - - args = [ - "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine_1"], check=True) - except subprocess.CalledProcessError as e: - raise e - - -def test_refined_pygame_2048_2(): - project_path = f"{DATA_PATH}/pygame_2048" - check_or_create_base_tag(project_path) - - args = [ - "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine_2"], check=True) - except subprocess.CalledProcessError as e: - raise e - - -def test_refined_pygame_2048_3(): - project_path = f"{DATA_PATH}/pygame_2048" - check_or_create_base_tag(project_path) - - args = [ - "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game.", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine_3"], check=True) - except subprocess.CalledProcessError as e: - raise e - - -def test_refined_snake_game_1(): - project_path = f"{DATA_PATH}/snake_game" - check_or_create_base_tag(project_path) - - args = [ - "Incremental Idea Gradually increase the speed of the snake as the game progresses. In the current version of the game, the snake’s speed remains constant throughout the gameplay. Implement a feature where the snake’s speed gradually increases over time, making the game more challenging and intense as the player progresses.", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine_1"], check=True) - except subprocess.CalledProcessError as e: - raise e - - -def test_refined_snake_game_2(): - project_path = f"{DATA_PATH}/snake_game" - check_or_create_base_tag(project_path) - - args = [ - "Introduce power-ups and obstacles to the game. The current version of the game only involves eating food and growing the snake. Add new elements such as power-ups that can enhance the snake’s speed or make it invincible for a short duration. At the same time, introduce obstacles like walls or enemies that the snake must avoid or overcome to continue growing.", - "--inc", - "--project-path", - project_path, - ] - result = runner.invoke(app, args) - logger.info(result) - logger.info(result.output) - if "Aborting" in result.output: - assert False - else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() - if tag == "base": - assert False - else: - assert True - try: - subprocess.run(["git", "tag", "refine_2"], check=True) - except subprocess.CalledProcessError as e: - raise e + return result def check_or_create_base_tag(project_path): From 420b10c5c37a9264e01854db889050d1490a9d46 Mon Sep 17 00:00:00 2001 From: geekan Date: Tue, 16 Jan 2024 16:22:02 +0800 Subject: [PATCH 362/637] readd logger to aask --- metagpt/provider/base_llm.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/metagpt/provider/base_llm.py b/metagpt/provider/base_llm.py index 0cd440ea1..93931f14e 100644 --- a/metagpt/provider/base_llm.py +++ b/metagpt/provider/base_llm.py @@ -13,6 +13,7 @@ from typing import Optional, Union from openai import AsyncOpenAI from metagpt.configs.llm_config import LLMConfig +from metagpt.logs import logger from metagpt.schema import Message from metagpt.utils.cost_manager import CostManager @@ -65,6 +66,7 @@ class BaseLLM(ABC): if format_msgs: message.extend(format_msgs) message.append(self._user_msg(msg)) + logger.debug(message) rsp = await self.acompletion_text(message, stream=stream, timeout=timeout) return rsp From b48a001d14a72b10245f06eebbf157d739762991 Mon Sep 17 00:00:00 2001 From: geekan Date: Tue, 16 Jan 2024 17:57:38 +0800 Subject: [PATCH 363/637] Update ROADMAP.md --- docs/ROADMAP.md | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/docs/ROADMAP.md b/docs/ROADMAP.md index 9bc62f849..4bb530bf2 100644 --- a/docs/ROADMAP.md +++ b/docs/ROADMAP.md @@ -76,9 +76,8 @@ ### Tasks 2. ~~Support Azure asynchronous API~~ 3. Support streaming version of all APIs 4. ~~Make gpt-3.5-turbo available (HARD)~~ - 5. Support 10. Other 1. ~~Clean up existing unused code~~ - 2. Unify all code styles and establish contribution standards + 2. ~~Unify all code styles and establish contribution standards~~ 3. ~~Multi-language support~~ 4. ~~Multi-programming-language support~~ From e0839822c0e83117d02f39d7ea8f40d348b09b02 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 16 Jan 2024 18:53:30 +0800 Subject: [PATCH 364/637] update test_incremental_dev.py --- tests/metagpt/test_incremental_dev.py | 51 +++++++++++++++------------ 1 file changed, 29 insertions(+), 22 deletions(-) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 8890137b0..2fd319117 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -7,6 +7,7 @@ """ import os import subprocess +import time import pytest from typer.testing import CliRunner @@ -19,6 +20,7 @@ runner = CliRunner() IDEAS = [ "Add subtraction, multiplication and division operations to the calculator. The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator", + "Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal", "Add a feature to remove deprecated words from the word cloud. The current word cloud generator does not support removing deprecated words. Now, The word cloud generator should support removing deprecated words. Customize deactivated words to exclude them from word cloud. Let users see all the words in the text file, and allow users to select the words they want to remove.", "Add an AI opponent with fixed difficulty levels. Currently, the game only allows players to compete against themselves. Implement an AI algorithm that can playing with player. This will provide a more engaging and challenging experience for players.", "Add functionality to view the history of scores. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores", @@ -31,7 +33,8 @@ IDEAS = [ ] PROJECT_NAMES = [ - "calculator", + "simple_add_calculator", + "number_guessing_game", "word_cloud", "Gomoku", "dice_simulator_new", @@ -44,68 +47,72 @@ PROJECT_NAMES = [ ] -def test_refined_calculator(): +def test_simple_add_calculator(): result = get_incremental_dev_result(IDEAS[0], PROJECT_NAMES[0]) log_and_check_result(result) -def test_refined_word_cloud(): +def test_number_guessing_game(): result = get_incremental_dev_result(IDEAS[1], PROJECT_NAMES[1]) log_and_check_result(result) -def test_refined_gomoku(): +def test_word_cloud(): result = get_incremental_dev_result(IDEAS[2], PROJECT_NAMES[2]) log_and_check_result(result) -def test_refined_dice_simulator_new(): - for idea, project_name in zip(IDEAS[3:5], PROJECT_NAMES[3:5]): +def test_gomoku(): + result = get_incremental_dev_result(IDEAS[3], PROJECT_NAMES[3]) + log_and_check_result(result) + + +def test_dice_simulator_new(): + for i, (idea, project_name) in enumerate(zip(IDEAS[4:6], PROJECT_NAMES[4:6]), start=1): result = get_incremental_dev_result(idea, project_name) - log_and_check_result(result) + log_and_check_result(result, "refine_" + str(i)) def test_refined_pygame_2048(): - for idea, project_name in zip(IDEAS[5:8], PROJECT_NAMES[5:8]): + for i, (idea, project_name) in enumerate(zip(IDEAS[6:9], PROJECT_NAMES[6:9]), start=1): result = get_incremental_dev_result(idea, project_name) - log_and_check_result(result) + log_and_check_result(result, "refine_" + str(i)) def test_refined_snake_game(): - for idea, project_name in zip(IDEAS[8:10], PROJECT_NAMES[8:10]): + for i, (idea, project_name) in enumerate(zip(IDEAS[9:11], PROJECT_NAMES[9:11]), start=1): result = get_incremental_dev_result(idea, project_name) - log_and_check_result(result) + log_and_check_result(result, "refine_" + str(i)) -def log_and_check_result(result): +def log_and_check_result(result, tag_name="refine"): logger.info(result) logger.info(result.output) if "Aborting" in result.output: assert False else: - tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() # After running, there will be new commit - if tag == "base": + cur_tag = subprocess.run(["git", "describe", "--tags"], capture_output=True, text=True).stdout.strip() + if cur_tag == "base": assert False else: assert True + if subprocess.run(["git", "show-ref", "--verify", "--quiet", f"refs/tags/{tag_name}"]).returncode == 0: + tag_name += str(int(time.time())) try: - subprocess.run(["git", "tag", "refine"], check=True) + subprocess.run(["git", "tag", tag_name], check=True) except subprocess.CalledProcessError as e: raise e -def get_incremental_dev_result(idea, project_name): +def get_incremental_dev_result(idea, project_name, use_review=True): project_path = TEST_DATA_PATH / "incremental_dev_project" / project_name if not os.path.exists(project_path): raise Exception(f"Project {project_name} not exists") check_or_create_base_tag(project_path) - args = [ - idea, - "--inc", - "--project-path", - project_path, - ] + args = [idea, "--inc", "--project-path", project_path] + if not use_review: + args.append("--no-code-review") result = runner.invoke(app, args) return result From b275f1a3f8c33ea5832d1656e26e9e4b8831b631 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Tue, 16 Jan 2024 10:45:01 +0800 Subject: [PATCH 365/637] feat: +ver fixbug: RPC think --- metagpt/roles/role.py | 1 + setup.py | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 3d5e55057..36d007f3b 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -494,6 +494,7 @@ class Role(SerializationMixin, is_polymorphic_base=True): async def think(self) -> Action: """The exported `think` function""" + await self._observe() await self._think() return self.rc.todo diff --git a/setup.py b/setup.py index ea84fe299..ca8bb3980 100644 --- a/setup.py +++ b/setup.py @@ -57,7 +57,7 @@ extras_require["dev"] = (["pylint~=3.0.3", "black~=23.3.0", "isort~=5.12.0", "pr setup( name="metagpt", - version="0.6.4", + version="0.6.5", description="The Multi-Agent Framework", long_description=long_description, long_description_content_type="text/markdown", From 112b1da4a85d1631d68b92d6fb1ba0f5f8f05718 Mon Sep 17 00:00:00 2001 From: Sirui Hong <34952977+stellaHSR@users.noreply.github.com> Date: Tue, 16 Jan 2024 23:40:00 +0800 Subject: [PATCH 366/637] Update iclr img --- docs/resources/ICLR.jpg | Bin 0 -> 456931 bytes 1 file changed, 0 insertions(+), 0 deletions(-) create mode 100644 docs/resources/ICLR.jpg diff --git a/docs/resources/ICLR.jpg b/docs/resources/ICLR.jpg new file mode 100644 index 0000000000000000000000000000000000000000..fa293f91b2f4ec23239981adb441b55f4048fc91 GIT binary patch literal 456931 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® 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%0AoQaQmQM2Rt6qf(qijf=;Y#3mGKZ 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 nL)v;5tKb(uOOvq_1O}|AJ&Bhv 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 literal 0 HcmV?d00001 From 51b3d113936914d9aa53d7034b7daec33b84ae4a Mon Sep 17 00:00:00 2001 From: Sirui Hong <34952977+stellaHSR@users.noreply.github.com> Date: Wed, 17 Jan 2024 00:22:31 +0800 Subject: [PATCH 367/637] Update README.md update news for paper (ICLR) --- README.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/README.md b/README.md index 9c88c92a1..0c676f03b 100644 --- a/README.md +++ b/README.md @@ -34,6 +34,14 @@ # MetaGPT: The Multi-Agent Framework

Software Company Multi-Role Schematic (Gradually Implementing)

## News +🚀 Jan 16: Congratulations! Our paper has been accepted by ICLR 2024 for oral presentation! More details about our paper are [here](https://openreview.net/forum?id=VtmBAGCN7o). Note: The overall acceptance rate is around 31% (similar to last year). The fraction of papers accepted for spotlights is 5% and the fraction of papers accepted for oral is 1.2%. + +

+ +

+ + + 🚀 Jan 03: Here comes [v0.6.0](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! In this version, we added serialization and deserialization of important objects and enabled breakpoint recovery. We upgraded OpenAI package to v1.6.0 and supported Gemini, ZhipuAI, Ollama, OpenLLM, etc. Moreover, we provided extremely simple examples where you need only 7 lines to implement a general election [debate](https://github.com/geekan/MetaGPT/blob/main/examples/debate_simple.py). Check out more details [here](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! From 6d5a4bf36adfb41e3555f4f71f91458e964c2d25 Mon Sep 17 00:00:00 2001 From: Sirui Hong <34952977+stellaHSR@users.noreply.github.com> Date: Wed, 17 Jan 2024 00:57:18 +0800 Subject: [PATCH 368/637] Update README.md remove img --- README.md | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/README.md b/README.md index 0c676f03b..cc78ed459 100644 --- a/README.md +++ b/README.md @@ -34,12 +34,7 @@ # MetaGPT: The Multi-Agent Framework

Software Company Multi-Role Schematic (Gradually Implementing)

## News -🚀 Jan 16: Congratulations! Our paper has been accepted by ICLR 2024 for oral presentation! More details about our paper are [here](https://openreview.net/forum?id=VtmBAGCN7o). Note: The overall acceptance rate is around 31% (similar to last year). The fraction of papers accepted for spotlights is 5% and the fraction of papers accepted for oral is 1.2%. - -

- -

- +🚀 Jan 16: Congratulations! Our paper has been accepted by ICLR 2024 for oral presentation! More details are [here](https://openreview.net/forum?id=VtmBAGCN7o). Note: The overall acceptance rate is around 31% (similar to last year). The fraction of papers accepted for oral is 1.2%. 🚀 Jan 03: Here comes [v0.6.0](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! In this version, we added serialization and deserialization of important objects and enabled breakpoint recovery. We upgraded OpenAI package to v1.6.0 and supported Gemini, ZhipuAI, Ollama, OpenLLM, etc. Moreover, we provided extremely simple examples where you need only 7 lines to implement a general election [debate](https://github.com/geekan/MetaGPT/blob/main/examples/debate_simple.py). Check out more details [here](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! From 22c48449f4536a60cbb9d09585d5bd0bab73be6b Mon Sep 17 00:00:00 2001 From: Sirui Hong <34952977+stellaHSR@users.noreply.github.com> Date: Wed, 17 Jan 2024 00:58:22 +0800 Subject: [PATCH 369/637] Delete docs/resources/ICLR.jpg delete img --- docs/resources/ICLR.jpg | Bin 456931 -> 0 bytes 1 file changed, 0 insertions(+), 0 deletions(-) delete mode 100644 docs/resources/ICLR.jpg diff --git a/docs/resources/ICLR.jpg b/docs/resources/ICLR.jpg deleted file mode 100644 index fa293f91b2f4ec23239981adb441b55f4048fc91..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 456931 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® 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%0AoQaQmQM2Rt6qf(qijf=;Y#3mGKZ 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 nL)v;5tKb(uOOvq_1O}|AJ&Bhv zd>CB#XtK4f%M{POQ^+H4CeTow;sTj$Bo zSpzMz`Z80v_zttvv?KLQn5!&y4@^X7-d_Ke8XEAK4+D)z-0S0_*povf;iIdp>T}Ay zomXU=Sn6Q9z^bmXjT6@Oytf?RD%ImZ`(&q@!Fp!z_fqKx$j>~gn@d<3AJ=x%p|K-a1HNgHSdb`$|ik^;oug9$iVrk<_Q zy>IKRz6U7xT_uj$A0=v$;g-Y)t+I11HC#2B7_0O)^~{bq+Dx52)%^Yq#rG2v_VeYp zctY%kLZR{7*`F1V`&qur!a9cBcfAZUDCRKuYA!-!nb}(mCD;W;Nc9!Xl0DTk@fI zu&NLHXnM7efH}}30Iv!rNrxE_59f0|)kWd~ifF|a58H989eq15Ifa_V!S$9fy4f*| zB+lsFZ^~p=)0IsEF6&YeZ!k@oVpQ0T!GQymegRh#Jg#lzm$1-6W4k-eQm8@OtSkin z!GZ|FPAGQ zu-9cStt2^XE;MS7u`^lW;P{V=ZzDCwAjhi3aHs@RU_B!z1iLC;VJr~H=eN7!Rx1UF zZ!@AVVdHd*JDHXG1g_SVXCIvu2s}-XP41bfGl^cHY^g_hdB_p@{xZbiP3u-|6dc#X zyxd$kS#(`{-;B8?xmK#L5584I8sQshL2oIQ3ddZCL5%2@}Aiw0c0K9hn{s>@HdjtfVO-rIZki%B; z8#ZjdSBcco=x+Yz1HL|=5#S@>QxY)yas>ZgyY~@*!1V}_yjV}Ynkb?htV1=dt0f(x zjIwyHeqhQi=W0;}iuT9HGK#A=+{{jOGelnCnLsShV{77F2U*D+mbtv}|G@J!q-PcV ziEO(VDIh@HNEmlhCpicnB?L}1+@@-ezE$41y9y;{V4$eXNlv>U@808T1n{PP5E~0i zfgK4KkAknd4&>$8d3I z5gQEmD;@X3%MsD+;XEC<+JH4KsCpM%bRmN-&R^?*F#(5dzBgHXohoZJUcJi7U^o6t4hNgQCm+FN(EGf$093os;8~|^_>e{xD z{nEbqkxQ@`3$z|h+O}uANK!-Y*UG<&74ee9A4|o4Jc)keX`T+|_7PslAH^OyB5cs< z2%)S0{T|5E8F5Qszd#h$lL2m z?v7WlZA*0m<_zm~KSLRd+*OkRrQuCLjJn(}&woQZ+c3n02ixr1pP^PTiB;OaLc3>% znP9uMwk=p_Nm;(uBLKGFv*lnUn8|w>31;+g zDSrfrr^CE?0|`+5-rGYL(zfJzry3}N_OP~Ewc{GbI2VoZ>8b|@>Gx35WE2ubYDxA2 zbqx$_yX>CJqP0s1>{L;oLOqt;_;E_6XsjrG z!D*i;cW*PrO!Jdzd;k$u5yv9uK=8`A^cIJHP(`S&PA)tmI(H2REzau62{*uB5qelr zLl%3lToe*ttuWF$VaB;IoZPC*y&4CvJFprmn{6f(q&k(?=q~tWgMM}Xb#*7HVSFks zx71m(8zYdCVoaxr1H7v~W-g${DQff{VHkI_XvBRcY~Ztm=v%y_y1`x>cZqe!a{~WY zFcP$PvixG{TiZ1ahyqC3>)CcBm&BoDRA@ z1wBYE{b`VK$A1PHd;aJ4`TrYaOrAfqqUBTZ6|~&-@)T-4PCe|wYhDxx-4VzHQlww^M{w| zN8WB95RIpkAF|7Jzk2lu(l=!Xim9vZArW}D6Tc(AaX!DRI4d65*>JnF-()|1@S&B{ zpV$cZlcV~$V{kzywjmw0B5!hW$2Aer@3Lz@@_j}ph7dG%dXMpDJ1nxSW2C&!0*IuE z;X%?3*Ka5u3FaX_Hw_7mp$bSVdL4h7Wz)y@fp?3AOwjah;nF`FdPSlad9HG6y`O*i zOd2vR-1c<+*Y{1k`q<`51I0YQhDuEB-y|IWZ0m)>Y;pgcjUJn~&0U8AtX|P8rrw*| zlzqEtjL7@hPd>n5wO?-$89&kll2>RUi;rxI0NOr&$jF;I=bBzAp&-7leEI;2Olqxz z7{Ja*EkSSAAEtb?cy}SfifztOcHbL*8~I!ZWHJ9nZgl0ASgm|bukG_AfcO@6a-rI- z^%}mc-$#^y90;`$uW#9vwy8@4j}4#QJOViMejT9TuQ&hqx^=p zm<+cp%2}S`n5fFEpgHfv<(F*1t4v@bPo`I$^TgffT%UcCl~49(b2D;#NOAWiem>87 zs{`&Ed8l1Xk0=%Xss03G?1v={n>(u5n zUbD}1&uZ{n;xFh%Nl~1=vRB_(Nz!8xAgBM1qe{eDgVdfMPA%K1%HXB6Et|x3y`O)) z)F9lXJ3dI2H{MFDIis?j`0(UOfN=9lhkJedsk}l4JFH^b>!19+{pzfztuK38PSUBC zZ|t!57w?JQ>e(=tS{0)WwgP=S@&y&v9`e^;6wq+R-F^u;h3ag}5Zy50|4edgLpf@e z^S3%~X+uq@G~1wcH@Ax{Kp+=n-OgkB7pi3W}M}W@4mCL<=8RRASg1FD~ z%t7jkXSEs#_W)evmeI#6`BjJ`DTcFzWH^E|5>f_KfHjz%(Rewz^jfY)dwo%*+QK- zNS4_t4lPR^;k_JE$@mRRWI?%23*-@Cx+?RW^Z_&Lo%G~wymh62-D*4n&V{ElN(6AP zYo9!{_YNay`czz)El(nxBe0|W**~VgoxEyQc3f zx5>9lMlut{6!<4#To%HrT223$h8K@EY_d&^BDyV5PObJB9+2X;+^s4sTXLF!>1-ug zmtJ{tcKdH#aEU~H6`zOFP^jr+m5C$BdCrhHy!idzNEOYkQ-#N=fb+@}6%bxfXVFnL zc^;OjEyWIoYfp^rb&6+u4ne*$ITgO#zb65-d;u-{v^X-oX)b5KFOFOp=&=|6sx|eN zzdRND+PjXtE&A}{Krll$ds`d)X6C!MXV$yqmcq;}Q6=Y~st#$5ljx1^SOcUNV>fH` zgsEZ^R6}Xx#b!1HIkABw7z(dr$7DHY2owHEGgmdUTBx zR{8b8ayVg4W^Of2%g@B$euJ97iR7Q-*tUls&iVUj;@*csAZGhtWpWtqH~90bO#T<3{on8R z|Ly62&)@$(LH^$>e?*?%4$7SP5d{IhnsLn=ox0l!0R+KcM#wM>ZF)#^;uJL~$6^00ImT%(qLk225pfQURJ<^S#>orR(OemW=ME54S9Iy#f7dX(HKB}zb&6;zVGdtBG3=OV|*;V36jjDi<_%yuQ;GM>X! ze&GA`3Pzr|-(3zh>z*}F9F+k@ z?V^{AS_3iI8Eq**+d?(}&AR`J{2T`L0&A@ln-e**fA~6FJ)5lrKsp=Ypv0%eoO;1* zW`WMODDmiDMbMkEF~o$r(DEJLG{Wt!K(+pon%VUE70509#$MPx0)Ew{UcYEXN1@+z z>2F%`&kluuN$>x;dH?s@{dH*cKRS1P&I|i7-zpZJMd_bTmTptxUGs)fZ^#JR&H(*r&y0u_kmXUA(~+o#1O zCkz!f)A4Fau9Fg5kf+VN0tUVF~j%9~LQv zMD72GXi*Rz(yci$dQ!MI$z%VVzQ8;yYak>rkVw@1Nf&aR$M6DIX@YGN>MJ>KS2H(by)ihc%w$(@|qA zNVXY_o19ZwzonyN zGUS&uxFoa#{sk2f{~HIgulGAm_#3=(D1$IuLEal*dWrfp)mRmtZj_vEU?C!6MR$uv`|U5hcF5KqYVHtSK?YL=@$2JstH|3H>YTKQD|e@6B9cM)dR#kpUy{s zTwwVZa^S-|ShfFw`jv;y$ltCq9RBp+?sxzz`2DUEdNczo%4G!A8$A5EZV(#Hzu998i5OYea{XBwb=>Y%0hN5G9WXz&(B zIzIv`j%lE`?T>&Cc32g!2UvaO2ailW5&BjtHSP0`c3_K9nZ zrrH7I6#gh4q9s?WdLoFE7J1$vZRnej-Mo2DEgQiIpMQ=!8~wV)6G4?Y?=sCFrjyU} zHO18AkY$$W@bXB)2jb&*vfa7z%8RFm373#x@T->U@_gQ;gQarZsa(Fv;RUb;`UqIK zSb_fXEbd=9x2;cqNDLF36d=Fg?yLJoX5io3Q8{5e8}ytf zG1~fK#rx}mc=jymQnojzlK1t3vcd%JicQ@JTix7)JH(gMIfqXbA&)@0QY7(`6Ly;RZa^{o&{6g%v%+7mB=~8p>B}=@gy4q zPrx)ZD&Bz(NekakV&G!a{AL|i{KV)}TFUs&xoiNrJUBkg;eK$Tia#_resB(uM6N8c%Jt0lsI&h@Q2sUq6$4bgWTY^AGDM{qtJ5-Me=RutC+ zDr+l(Ew%G)>`l`XRwbRoE%XYXtGGhZFqD0T1V;ZFMA`8E)gMy*8yWVS=!uTj&0lajM=em{wI6wi=DNf(-&u*oB7n7^Xhx*d#mcL8a0NQ?zGG0BNW3dz5Dy1T-0d-mRW8&4)~$g-ZMr~1mbSR)y*#}t$))Ish^z(~G6 zzFzH?lN%s+$iWYxno8^FgY+jaSWr?^A% zoiX_T03k$u(O^_PeT}P+HvZkpeCzp$tNj{teqn{KSH7SDcki(^`+J`w9t=Pq%z-EC z{{eD0|HL18`I>P*KU^eC`{DvMfIy6pXTu2zdBw1dkV+*e?{?{k5CV}r)l_4 z#4nu?{I3B#{oeBEkr+^xR1=Hu83aj8Yr|P~SJ5hm80KQFfvv#8(!J0`7mQ!z;i%98n_1=KdPSy1w60?0T- z9oaNQ8R2!_F=G{-tQZJK51e|=Q&`}l$ROR6l4v6eH>$}bstJcdqb#m@Qsk_g7q`g~HkCO31%wQX%wK2G%m4M~K5L;ybxdK*w2JfQ8P>wk7) z~cXK-;|(l3vhNcDOC^ySy9-HGJEQeGwe_P znNQhVoSV$Y+(W+_-Tu$M_t5*Z?Y$U|=?=hmTC(gzh^Mzi_^%n$EA*uVUnAe%2fQ|4 z!zcBJn7S+nFb^qiMy`uqrQM;662j}MQrm@}RA12@=z?rxg|Ta)KgI;M;5>CotOjVX zFjMyxs<+0^pIp3oKsCsEf-ANyL-cO#HyaU=P4bvyzH74daDdE?{rh`+9K$s_>=s1T zD(PQ>&}5M!v+h~t@S0m7G2^7Hw#h5?50eh1Mh}v{r1hj5W+bCl&dN5qkJ29@QYC`N3p{kNXhk&y zQEchsGWa}Ey_3+zs~p*K?^X60&QnYj-zqAS^u6Gmv_z{?pP7;&IjdAYYXb$@q@;(YmY&H>lMogM-gAkw{k zorh1>e#3MOyf6#oh8{oAS_EM_gp*`Z#o@12#-E6qbcM|SrnL2c6u3AJFTEp7nS_=w z_GP+y4Jy|*lED)Ok6)4uA-R&F&?PkzOe-@+O_41xxi{kggreELktgA0SxS4=l?X@70Bwx{CPl8N^{JylsHs1m!&okr)FV8d zx$|XNO_dsPnw@ump{n-;9dU}#e6dujtT4@{NmDl+ zI)a)emw8g_A_gcT4W~o;2^YK0_M_)ESfc6-QFjx}{UKravnt03^TD_b=*fmEebA=B znMZ%zA`qvL*tw**6#7Xx2Jqe$p^-W=brdVZ-ks_p<#~j7T8O}Ml+r;y$de9{)Z@T| zS(sOgTADq1&Dk|Y`D!$h<=R={t53eL&_=pKkx+3Ls4J#jeV6(AW~cS))HEOEN2dm= z#<{Kyl>Kel*-6Ay4l)|3jReh3IIM^Uv7csXCnOR5g8)mS7#O19+0Ulmp?+ph8b8or z&8^tID?1mn&gM+j6VX|I(*D`lI9@{$hl9zOK73RQ1~~c+5@`s#4$*c^q|>6Mc94~B zDzG*NA#eU};&@So^^$*|UJ^{GJ|d{75r5Ym+PQP3n#P3Yoe|X8fd!#3%fzx4&rLrt zUcL);BB48Pak8#be;}>g8lRn}P@EO=LL?yO3FYPYbRCtDwtiU$enS z$t5r)j?#`Coe62jW+y~T&eWbli7#tytwAp3#1?Ze`d&Epk`^iB=z^nI4tnPTI+FBj;NVZaEm1X&d{ClpQD zLm0@i0x`^6cY@!sinHJC*G|6{f~>{vlOyZqz!$Jg=gs~nJkDQDEM6(WTzq*WF5}f4 z|B+SRrEr6MP7`|S>|J+r`)veX3HP(T=W0d&g9-)fSH}rw-$L#3HLM#AEStxLF^gRj z@h%>_Zyx;q+}TEL3F$E8;gbNQ+uvd#vf%Hdb?D~1YVX{?YYU4G_pnVUyB z2WN~!&3Z>538^Xd(r@Z_7-0=zR-$8ySu5^)aJdK1cRM$G-^fXG&A-~w%zf?HDwJ7C zkZA`Snrp-nk}vr}2c;A2+SQRa!B3vzF{RzQDcs@7or!oVD7$HtB!u!A+k7oq{=ZAf z@r1h>(NP4T6Tb_cr2}o%3yVlxvLY~&LcFSW9CwVJIG|RD!9Wo(tDh53=&@0fdB15$ z`@4N1z=1K$Q{eYr<6J`q7B@pFLIwUDSYgc{RF@%RWyj;dbI=%CK2`}2HX0cVaKp?h z>uGUh3KQ2X-hsSD5eR_^?T{IUyL!D?&PsYBrsV^U(*-NbO7MPZtW1K&M04GYvl=1} z4N0hzp78?w{}wj5jgHeC%3t~9ZH;9=EFt1u-gi2QrW7h;QKYA-)=9Kf{5OHYZ&HjF zHY*zQRn;5{{-_(AklHIumI)Mp+cZ2@Nggfxg)tD=#eYT(Z}kWu(S=i>82N|T1}1LJH9KZ zBBiyEEGM~7d)xpA$zG++Hl66O!|6R3S&7D=R`Wir7cBBJo(fZQPxHm~NhW z|A+G9UGE#UEefkz8E2pxu;Ja=@Z4ENZpeY75jobRoRQ{u-_k26(E_z>6;ed<)rl#v_beizCtIHdVQY@q>EvfTXrX4Su+uX!?su}lEY%YkCxF?>X40Jj5bJ=zl zA{Kg>6~8>0gfkw%|2|XnuhAz0XkPA58k)1af4Ijj%V%Z+TZ&RqnbA`*B@3&zoDP$J zqXhphWO?)i(Rpq(Pr>7+Hxy9bIXYt*o0+8UBq8u&*iSj#-}2iw?Jws61J%2AM)2P7!6#YXyDwn#9Yy> zTj+n2+xWScDtA(_n_6@EWSMF3$`Qnn5OfPp+t*~cJjCmqFXBCDq2~+GLALm-Iy`Yf z@?DV9i+%Zi@tFSe3-xuH#Ye*PWP)?!$ zu_PoEB(B8e>>+y`-5jCpU;=N#$)l;<)e5s9qIY!gKdI*{wQ1dv8XaH+9SHEVF4&*{ zkk7P8!Z(QuD-RncVRBpAxNPk_walt)m!?@@S!gD8mnAaVb2=i366;T}Ds8?6%oVrW z(vlTPsueO8Ox-JpVVs~1omdbPQ6g8Z4}o+>Jjy0jKqSOH_d{kz&*aB@IM8x4!Lr{T zK~Z@bK+(mFSI6zQIF6}aEy<|P`efB_F=8nUJ?;%qg^kGrob3BijH^91$#-TPlJh%N z@3a)Kc%Rp#+P(21uIK#XoiQD3H;rvLy_dKhuh@aAQxZ9BnU#;$f^p-qr%cae>-7GS zCRpv!eTG2UEwEXLxCpB?-O$)&%jz_=lQnFk&-+=qSu>v_O@upOh&bT*7@pR|v|F{v z6{V|3JEO}pv=EPhxj0yfZkbu#;37$-W$X$zeIW{zOqc3L(qP$BmMV*b#i&pjWQDV{ zMVj>sZ4#31`@gcp1gx2h$wroas>Q*p<$moAm(%X+%!ST!dLPG6 zP>u2}o-R#cxxQ;Ap9$$AUcs!iGEO!(j^`@xY8%eqXqTj{JI!0!pY_5Cj4Xh7XM=zF z(ewO`*axK;M7wnH##Er_mThN*6=rLwL83SRVkwoG&{Vo$HtcZT;#WWDfLJZhPxC z#DNd1eQE_oO{{m$yZ`@ZVneyen-mvn!I(8-hxl_#Wz0?oK0V91t_!4-H&=fV zni=_;N0VA}c6K^a2#o+c?z?4WA>2Mt^2JHBrDkOXZ)-;SBYHzJqb%SF7~+8T75!P` z47zJgbo(>@5Bz@d`^Usr_-Mbb$taGip~IryU|DzGd_HI>e;tGWR`TRbL4UA%J$!BX zjJ%dG4wcGkKz86s+bFgGy>ghGna)D`=#f}X|I|$$!&Mg-*<=Ksj!RCmVT$L(Sw>n0 zBAe}Aj-kfQqF_2-?Ana@W5QaHJ~KW; z0T|hN)}`w7twNuCFu%=3kY=2#Ob(Iti<|h#%YLh2Hbq%xUvrw&m2{J!Sa$YZ>-f8U zub(}dEPL()8dkP{Rk7R*Tc%^_Qsm}N9yYyb2>4Szz3)nV+SkPJ$VHYQ$R3@KLb9o- zaAewF`ZqrH6axYpgSzq2;1@TY5O08RUwB9Rl<_7a7(-iDE|dB|m^K|F*)1~J@(+3u zbZ#n@)T)@kS)RA9*VuLMcheCn zU=!vC2DE&ckIMf5!P#rFPT<^CEVYZOJe(}^TikKn@^Qe6kjEh4)Xl;eP#LYj1t(bB%L z9bRjW1e!bMOuq!%bMTW(jrAK;Uz3gdE8}RM=2B@oYf!mOpWr^}-H)IOckT<;bcne? zV8-De?CI?02Lt)!+UO4y2{a%FsS$+v9ULB-H|%)w*_WS+;}A1Y`*C%^81AbrM)&Rw zkTJ3K*YqW^!M)h zi4Ve&is#yA9LQwVtU)B(W_M7{RDZ*oJk}hGgIZFG)}NjW+2ZQJD*)1s@n0N?_e_lktSnq>l)NB`8JV}-Nzb^`Dv_M=b>2fy?TsG&wW`39-INc9SDQ3ai|& zb(C}GW*XI_pq0vsUI5l#6M0KcWv;O{_xE(_w9ZZzOSUaGHYP=TQ$yBwhD@?P!@cp& z%vNnvu!dNQn+byrU$F!gQyB|A4XQ2101v?CFCt-0W>2gxGE*qx=k|!gtR!4n@IV8! zUu#ybcs)E=W5N$!k()*N^>G!?` zuiN~o0S#jRLoBk57UyhBd$PutH5p4V%K2!*i>C8u;s8LRI%Ax5|Ctx8NQL=_?t8U8 zSa#G;cgdqrGOS1=f;O9CWL7v6J$hpc2~vuz*Z;Y)i3s}w>N#gfd2Ysqo6}l)*3z|y zziI!K%q+%6Rt5tN;2xallIdZdM#1?(d5z@YtOc@Vkq6CrN7}mjcBHmYk#P}vu))m; zf(yUFs~#K-wssx7P35E?wCR z?A>1aRykc(q(Is*o`53ISB0puUTZJ$Hgj<{DT^NAYh&29a_EccTsfSr3l?2TwBXuv z!F=BJ_IR;g4gN3Z<|uxYxX@nUgeR{08>(zJi~2xU?73k?M-akFU^*2T7eTmpXXZVdnbsMNpYR%!5!JFokMpoO}cACi^l~ zB0is~_m6Wqi z61Q-YCTs%!bx;tC7J?9@p{?2t;cVXPOSqC;A94f74u9< zV;2$sq!jj(Qn1aU#pTiK!P2ZatP(R}!Mj%|JCE8PLK8v7T=lSHL5l*T%XZmC;8_z5 z5}QXaMpVo~D2FpLqx$;)6@0+X&RG(6)nG+0D4$_>P2J{VTM*x=2LE5 zo;Cx=n8C1^U2p)#V{mqB5{iqT73RD+Op7LF8V_w4dju(q8R6b4NaXsk68ZM0;~2KS z_tcYR*!CKGhqS_d%p?*>#nBYMKV$`7bWWqrDwztWv~jLcki~+kMB*|P6Qi94FlX5_ zm1zdrSIPO_ItD-(Ryuv6$}3&0uHjDF8^rwlf>w8i&+2aNFbgr}k~L-Il>U(}4l}q~ z$tFdSnhH=)_hrW|cvHuM1)*0&#l#r&>RP1dTa|ekW8bNs7J5Iv}XL_e9&KK_yjxw zCB;x)m!vW!Y+i1GYR#D5lmdFNR@510lAf%k+f`axN!C|fGoYxi;unQc6Y#%XNqBN4 z`0lCN`Cf4Ss=~;*a?NHkB3v43nZj1t{f!yN73Vl1dLJAEZ$v@+VSBJd621T(&XLV> z2qF_M4zB9;$It@}QE6j08A@pbiR;_Z4dznYO`$VLwWX|?zNC{ZUPG&jSVkMZJ`x_v{^CT#; z-?*t;6f3h4hUcm!26n1Vh6!zHk*+IrtA9K^9q$aq;?fP%I|&ian+Bpj+d9CopDqlD z&Z8-?reiu){!#d|=hIopGj%5p2uQTVG!scz6@l1Nx;QVb)-p>YYYCZLP`;8Rc@PTz z+enkn$nN$0s%*Y=A@ag;QBW#XcRj;&JHHPa619dLJU=G?66(yx!&~~2(@*^1g6B5D=&nhV6BU;$2$>5{Gh zHNoAP>rlElqro*ir3;%QMYCO*b2vG&zF06nuGw8dDhXNXGd1Q|CrJyh>@vKV(P|q1 zPd6S+=`6$r=JwLl231|Lr4&bFUn=kNx9B;#ZR#?(z+Bs9 zR2KZD=LS7OmkSUZqSxEJTx1)#cY-h=h1`eJ3bY)miSRxVj(7>1@BS!mCj<0WMW?Qr z4%tEOMXqkj=);Bd$;-Hb5KT!jWYtnz9}g4F3*YK!Nb1GN_HZW9Gim&2N*e$>oRb&C z8Qqu5B(A4Pp1-O&shHLKp!}j2`sHh}_jKh2ZZkRuKzg$2f%&{CgrmxBX$;xd0jQ}%im7;-7M;WuBW;1<2m>3d@kp4>4?9kR=5( z4lqJQ2RDL|pw5$KN+uqIwA{3OVDb zb>n{pp2~#F$&VCUtiY0)P3#_`bW4 zAqn9XBtO^$Xo7o)ruuR$cbTV=$w(MnZQ)c-GHovb`4c1c;3^~Y)*V(FqPY}wH^|Z* z#WAl9CI@5TUFmrN;61Ib!61L9XcVMLM&Ibpv8SYxTGJv@Ck|6T@aR+?vDI`Ma;FnV z{|*&uzKZFHP8{-qk6kRNkXtQr-l{NNy9_U~fv(T)$?4{}kH(A+0s|_P_3z!z@3tBB z<=}dAPJs`1r9_-jJd4JPNZtXlz|PQDR}d53krt$t7LlIZ2L5^{ z#a6qQCq)G)hV8R+bADYqC{N}$8zWe1h$VMmd#Y^x-1@(26Jm_7|gH>-s%~v z7I?#7o92MgY=W0oF?vOd(26f!l++%tQsTqznHok%Myt6{p`S=Js-AVHnT)_{^O)ns zS<%IMdG17*A@YLWq48L*Uo>=TSpG-6S58yV`(%oWi4Pit}a= zca*7J*}vwGN(M4W#UufR(OE8sI#}w!hW4jz@7l!4sbMl8Vq)Cx_erO8^cLKel4rx` z$SVX(6zf|lk0kpc8G(qV#E7?QY6=%#iE4W=9TNAy27Nn&vy3OX3gd0t-$)J}dfLc? zF*&XCnx~NfP_$X3A|Af7?iEe8RB;ouY|1J6rjJ7no(G2WxQ){|zVXE-LV^u9`l_`> zjfc-#A#D-$sc0`8m`Y|anqzSaaX|A4!%H}AY7)e7&rer|tW_Y5VWL%4{ z1qz8aG2OxCr-rSoCH@x`y4c4(c)-RZa8B_g#fdU#@`iR>ebOc~tQiH7BDMsY>#9%$ zoq*zU4f2U`ww3M%>4JruHC= zaVEp2dBIbx%0?D^W7!>RrkJ0EMYnPPWtv9jOZC*k6Kra2>@SjY@uQozz6bu=QP0Q( z%<^87^JGOhh54w)lkd!*62hCKJLv%`f#(O&md)A9caMFy{SfI%S7a za+{It1!sR+s@pr*TEJe7MbepxCFr(Uu_exj z0#r%z{Lxcc91fIDoE@|D6$LdUZHA@m)1w%_O-uChSH}o{YckuSaIQC*=H#a_Nq|Xb z7AGp3zO`wLY`mf>r#*vUfaU90D@IfiY*DRNP*qUPv#BP}E5DH?@3f-Vi!i3rw2!#} zD)YOkM|hMb&8bFK`Av_fIEmg!E`zm7`TnP3t^0A`f-cAT|4sNr=i^)X+7R`>NF1GF zbs2MS%)TU|vi6qWpC9(V(Aju=xv7qwImb4g2lAx>A__*ik7*L4cuF80jO3^MmvWB4 zJO+Q=dFka)QAzT6cf0SZLKUe9+XIx)bZ6Ae7 zFKb@MM`&4c>TR&QR*mF23RlLJ1(14&Nv6~;2rUC}Ius7TdK?HnbP$Y{!***XDpF5A zq~;1rh{{eke3A_Uo6Tr<32%t%jRfPM;LLAC+iB8wCNsLc0nWOO3f=37<^`svD*Yi? z&@h{?h8WXL2YbUN2U^zHY;&mxf)dax>jZ$j@@AFm95Kr5wXJfE4Q}IkxV`k-cdnA; z^?{j~`S$0f1-6&1etV<^=DnZGR&V&;!K^Zd6zK*cW~d z+PS~g0GPF|_0R=^e>dYIM#8c!XG=GUX`0>p<||ok4{&vevxNOC&JQYyIZT0Tm0GR^?Z;dZO))?vn4*T7en zNBs9;ExkK1fhmxx+`0KLrV*wlv-jbjlU(#V-qII&Ra9A#5*B9}zcNi#Dfhz)%S>Qb zo^U@4PUUoC@e^?T22rNsbv4cgs!s4-%Gl~{m^Iy(VnBinbEH<1&*AOCZ5KyyochjA z+Z{%5oKHr+)YD3E97bfb9KJ+m{(t_&pcq09IXnXwjW0`0=t~gDkl-TR!ky(_kW!vM zx+KVDzxYQx%Awtk74M?oM#na$v~7CSwm6eNCRcb%eUp=Hcellsqc@Ulpw*1_JVA|Q z9z@?pX7RTtp(TG81y{GE1k9m{ zC4bb|HiVK{MiFtZd&Naa0+I&BDN&|WgV*2O*;QrwfP^o-Z39bOJd*O($7ziv%bnFr zXcQGZSIQ9rb`b)vW_ZI*BNH{FwbNrJ%0+6JmKv-6^9@o7blaOKQ`gO@O#^k&9~o`v z-h#ddrB8u~HZVRwP|uoR0V(!(e^eDMV3&R5Dj5?%Xep`eX3E6*>xy()fK4?akS#JJ zoOKe#IP2rWtH;Aklm*}WNLkToF>TxDaIZ_%*0hx#664@19p4v+;>K=EqgG2i+&rt) zRPG!u_~4D3jurk8Q*PB)%se=k0p%^U5h-*o%V@^*l>LE5SAwwKWJjeefRo&&)>%_5 zY}t!Ldo(eQzqFGB4s16gG`ND&U>@TUKv(v5WUz^W|Q7~_>F)EVbFjRI;T){GixvP7a{Lzv7D9K?gF%8vDdl%2U<(db>)K$Dgx^hzc z=)xZv-X+O#F++m0)&c?iOG#jwc(sv?CX+&!xO<|Xuc}y>&)5=Rj*q{j=rFyvS<~w( zC5gE=YBL{SMT^s_*SRjG#_O@%Ie*?27NeoJDtjBKS|w4eXGCeBRyUMFf6Cn5)iCqb z9kwvVab$QDyvR}!E?x0407KPRXAdqt>_#FvDPjC;018um#5@K%dg5v=$!Utl9i>oQ z$Y#wDIdo50z*vco>e!cmEK{ ze!yVbJaRKbZGPkVGbSXJN?3kASPa{uhjKAKc558!gfENH$+||I+;DJ+h*iHzl4=z}=O$cm%_FOCiz_8ppcFL;4R(!RN5Fb^)awFf8d9nq*@;IZb@N>Arp^$I8>+1CkVE`W;t<~w*<_I>d z$COQEj0&sT31sc1eCT<5b^cFVHE4yU#XeMUO|+EEvrZ&iI^M`;*e3sO+f>4{{RE-Ov7FO4w=d7H(G#Hva&M5VqU|elhCo#52ALY1m@@iN%JNz^?mYIvB6XCdb$EFN7 zCdhc+GcA#|hb2Z=E=I-tSMB60NEc8C62WF_MmT%%CHjF4V55o&q61JkLvYtv zG1kaY@(XT@;&v0R6)EHj))h@aZ5^=Yk77<1-n(C_tW?YlwP`g!m91qKEbadkgN)#2 zt;#X*w>Aaw$OKAYoS@jL&5K53P2dX}E6uz~j_~SmB=)T~Pj9&KOz1d)sy;50b$Y-dM<~zYChyXYI<4HvZ2ka~VJVPhWMOL? zp_a7TfIiLjrGV{HQ`sv_V` zB7*IW#n0$I`*{8Zz&cRtVhR92+v39e0~0-=(F$q0l&P8M8Cgl}pQ;%5A0Z=GB`%_3 z3FpzBtjpR(`(IuNxfIfBV2ib`3@B^KC}bZnu8*Hech4ue>Z;qzu;h{12~2n~<`=Yd z%oV6QSu6Mu@eV!Acg>Vb)@ZudY7AZD$=H63IERh`=O^ida;I?eXhMe7$8e|za8t$c^ z+2x!4S|lf#h4JoF#<&C#V@pQ9D4M%EZxqc5;Y2h;Kw+t zJSsrb{Afi7910TCWS&2Q2=woa2?GOvf>D+(+D52dK5K7B-$uBrT|P~c7NLy^$zF2= z83(?yrMQ;2EgJY}<}Tr2i;Ri2wlW0>sCjtyyddw@ zE~T4YAMdj%nZ>nEQXy%P`6yu!s`G#sm5Bhe!g{0#gT686{x8N1mx!^V(9JzKsVs^e zH>o3KEaIIJX2Yo#%I2Qp8ezWs-2VV6?|Cq^4boYnQMS91BPsMntpAq3d^kW}6 z9%4iO30_S$_S$#@O>&(PGv4LyJ&d&;rTt9GR>|(bWrki-X7$RKIy9yNtIlVs<0KE& zYDd-T$#^-ntk*=z`FbQtU3t=20cQZNJ!WW1s4A1FD&B|X4pV5c*Nha=t_2%^QqfB` z9_b2j%&d2uyFIDg_S0=!2DtGr8aJ1#u|agEe~_v4wf~)Lx8q736;5&tnObt!6(dhX z<6&0f7%;OqV6pLBbPB zGF(FC{0B%we>kLCL~d&pI3WXsSGrXf$0>R}baS(jp0HnD54>Ouecp&`R5q1xe zB+&`r7LtwbMr5MF`~}X&@!?FPiW1{Ym;^;xhQGi)NL74)BSoPxfDkAsA+`r+31&vX zBzx2Z4l}Sfegs~|sA6c#dR`%U|1!gxa0cVsxDGCx^icw~m z5*B&+>NcHUk0X(HcIWHnliSsU-UQBf`&b*uJ;v1Edo4SJhrkOjTCK#6cgsf2liAOr zbaMZNk;mz4pY)68>hjd>p|ayMab$ZToP*@Q$lWU>WSx{h#|!6xZd7n3AC4tAC0?72 zgiZB};Km|Y41Q&oSqSAP7y)ZG;QgxX2S!Z?50Dl=X0X4J-5e)2NueTf-w|o{=5|7x z+Pf;ueMQy5e~;-|8br=W9b2EN+IIV*vQl}kn?y;?_ct}b8!pC8zXYxRB{kp?21z?P zdo1<|y9U4JJ|{v1_g~PYCeQWlypmSeRd7l#YNDLilL1+FvQBhgl@l7y1|37j<8vgG z#H2LZ-5d#0ffe9YdHg&ekN0vXujgqslvYg9jad8|^wwJq-sl1zBleQ0_iWQ8+VwV` z&M$b;L#@OH7Okle$>RNnUiKgvVXoM|A-4r$w;@3S7rQ55B+Ruplg~Tm&0-gW$MS?v zuQ6flD#|97B_9_%0}hW&l4~+FGG3&}8vA+V{G;z#`L}iP7oqi5t@EObckQ+ROefVi z#n@5kbs;vFY5B!sfF8LWZ;Q|7+lSii2>v&l3DRNQRE%!iM7c3+C=YYZHYvMUC?GkT zEG9{Jx`xGsFbVNETYHQ~ZeU!hKqCBrF#%w&a=2abT0qTU%(LjPF93cI zUuF*i56P!bA{`z#Hu=bkkp%&IvfjH(<8Ev!O+;uMu3zzVLcrz@CpbZ)$AeZ51><^< zQFpj`+P6-$uu)sOuS$c9(SUzRq(&c`Wy9BZSnpi)+k_w%E>RSQg>$S7je2+@Vlg7z1T8#& zRYYD15bm;-AJ~TT%7f*bj+of;=twFhZF*}2S29Gvr3!jL20d12GL}mrh}hh>RqC*c zvPs)KoFe74Zk%J-^ZlJ2`Tk`zUq2x!msz4fhGaDDK0lr0c8o+t8-yi=#z~lX$Sjr$ z2n`96d~Q-Pr19AWujoWbZ2zVSmgbexjLuZO%c4PtV(`fgP_&ZL1wBb(;W-COOj(5e z$P&7mP!Ylr`-`t7RUM#m7-M`(uRG*A(Zj4ckY;xPUw@2D&;BSDWs!H|S$|KXN!HvL z4VpNXCLuOKT&V|%%>Q>8hj0ul!Se@0-0Z+MofltV zX>p1#H+SIlb8b(LP6^u!ee1P@&zaZg&`Jh3v-PQq_OosR)ABXEVqA<|L9tgn72Z^E zNDZ+4D|HGQElET$&F5;xI3j-`+SPyDsPW`j468Q_yfraFFQPB^{ z75&AK`u~2$$O@3w579Hqw_EKT_ZMjQsi{LA!QnS_8EbiUhK#7-rXy5gZ8i4m{6hBU z_;*M)V#pGpesylivPZmei#)w)?h+VPX@+jCgmphCEU$T)wZI^}q$! zv6&do0IsjX+bnBxCqo+;FX$pVBA$44(GBvGRAQc2vhjgJCUez6c-g3FJV!?xA!Vwh zM2E!@${-1Yr=h~3b0>}S4F)9Rp>W+Q^CyJ~>J3Gh*5&BYI11F`Z(Z-s+%~s+7X4@( zzS)k1zpj;I4Elr{XQMpDg6_L0|Da!dBz3G^ncxO%HkpAgj->-8TO7CHsnMJZ(DrX~ z?)l4;cn-bVsmd#I&j0W$0^!=Ud`DH2o)@EHWbE=(=IItCMvM!|2;^0?jIvpg$J*BS zj?=9c6km*4Q_gSc50$ZOG1@A0{3vVcBP`fts|o!;fl^+9q?DcfuTlPC4>Z1?6N|eo z>roYpzTGHPYZ%2ddsdwcn}~Ot>z4i;4>s4+=vj5z^!v9I25{Xy<7HJ!GA_Xfij+&} zad6&uv5Vdp@*p0oVP*&AR25hV+!g&|PoMEMjqNKUA3uLe|4f=#G4sWb*{#ay{i8{n zO4=271^hXrZlLE!JN|E%T=uBQY`y=2>DT zs%oO3F8o)~sss>^>y?2>2HC~9<*CJkZn7jTVYog`n`lCrwy&%6Tc+-+n?6s*5E9B3 z7uEcf8@uM9v9pkTfCh_izwDxpSe>xz>S}1?y7|#JFMaoz6Vptr2vpWUeyC=!wx^0} zxbb|FJPy!VzSC?qkb%W?UIbD7m5ug91J^ruN3+Z5=>BAiP!i0LR(sYS zV10R2x=S~{G8@bjc)`VtXiVK>ubHPAOV4ib+p{fk&0$=Qj+&YRf}u1 z9=mpxNm?iG;Z?Sd#`8uAHrqEID}Xt$U0gyMD^k~m*X;);BU9u8^km>-_5RtCf6?Hz zbjFNv|E>&d^?jKm7w;X&Ph9TE?@8;s%~|FTm3(Mkq)e4OQi!TRQ+uO3%|#JI;{fD} zSLtE4-j4)1P?Dy9x(!!Ai!(Ri=sxPnO0QY~!i2fj{xFyIWccb_M_MCzyK_JY>;U{z zqi%4C7BRYYan$shOLC?rs%c%jXIfrW4KfWlIZnTLFuSz2Uo;0!$GX|RLTA~qPj$VJ z2^+g6zmG5;aZ3d!mTB>Kvlh3ngb&j2TY34FB%1ScogqC=dBeQP1h2Pp891CclKaV7 z@&+a7=eKP2J#&dIj=0x>?jm%;fxr#zW(V>Em(6$F^z;+$*@YEX@(Epv1vk?kJaD`wLsOS6 zu?DXXwoYbVwZwu9Iqd1K7qe?$<|e=o!9MWmBD04iainU!RT7usI8Q-zW=!PKtQExA zv4`rp@C`b>fZ0)x$f5%%$5*9OfPL6IJtBe$^^m&l_A$bCwWLkVrg7TsTVOxzQr@#E z)-)NPr>AKpsBFEbuGkw&rRHpJC15^c(H%s5VX<`mbDZzo}-oRj$sfcp3}I{Tc^yTwi+)zG5>-R zG5FN9fu;N-B17e_-#GW3FqEzkPBoFLc%GD6K$Z4yW98bd8 z0RjglsSX;0IXc7icJfq!&& z*3^LGIB8eDB|Pb%M!STG8_Cm-6_}Rahh%kHHvgm-w6#K8gF`z~!OeneeWetCZHAId zsV9@Wcqozxku>%pBsn43RJ~EZd)1bS0p+(V@;U^Iq&gMm>7Y(@C=@)E0YZHoOip{V zP31dDB3J#gDx6~njX@7*FNC)%gP^87#?_Bk7vgy>D!rj@r^5u$yn^`?Alg&EPdWY25 z+oKuy^e%V(i&xvL>Y}lmhrw1_x5p^t1Bv_x|?XK3-wEnQ?-PMz8Os1u0< z(u_7{@p5f_Lb1_4MJBDm6?Jn!NyT-c;GA-rR}rTtcb56n6OJjK#M~Ii;-B6O0~^MI zQhUokS#Pe+oC1nXbR-`6j{xXchsWw0n4-_4=zkH){~?Sehw*{^p{$8$A@_!CKANW( zraI7&WeE9~D|t@GQjR(xJTtM<)( zGAJmAkJN32>%j|%gK6ghNCmiAF-6xJj6XaK0IMgGDWMqEy1n!uz(JVRaU>kR=yHzt z^OZhYB$T=4HY@%oQT6TqYzb`9ffQB(vuk<_sIqHs9dfX;ecyZ}^CEa=od9T>d8F}< z+4lnQzg2@{ociS~XOoJFjzve(o;o;{ACejx<1JAgQ(l3qyuIgzF?ksr;q8ikaMR9l zCYdt{gR!ihQ1t~g14K=JntkG6fio?shQ{h}l}dHJGkY|eE3t+(tKqnXqGE2F^P^$@ zf1{z|ewtuLj?*yxHLMy;gkqm?j)BKaAx0Pl>mbEfG4MJn-t;K5MsgHNFh9Ya8NaW2 zpd^H3%oHJkpjAbVL;u|kWJ^s1BTJ>o{i#Bzz*HAX1c|_35JH;GXArbbj{2{YCDARh ziu=;RMNC}4rXxLiSH6j>J{x!9qDB1!Lq+v>C~%PKMm;r=H#>7THKs-Wq3oCh8iS?) z8pc#S(NDY%*N#n9u4H?ky?(1hQfFf@1N0~%{0cr;tS~5P1cO>F&$X(LyfS!mY5`N7 zWq|p_39}4%oj|>(1WxiQj)_bBCOh0FO{3JmZu=m-JUva5TyD}0JU1siM-k2`9647H z(g{sP>F}#TXTotJ1%af1Ny4AJxWZJgr9~dxma3UK&TD_eKk#)($rlDLbPm|cT!o25 zT@fhKmX)p`j)A8@e*iB>(oiVSAkH-a8;Yn7r$yt8wY&G5vlvf-YO#q)8wtlteNmaQ zE@K?EEDeYCoL+zq&2(K~nRC|Vf9?!~M82h>{v}a)qFbX1EG~b;zPQ|A(?Ex~I#obb zQqpl7)TEpbf`cJKS+?xR(Ps>y8%jY^yq*2s9|*y&P;fJ-&J{tkc!pl3s?yLLqo~r6 za#|WOcA4{L71y0a0Hgo+kC)2E^-f@&IgNO{TT0+CGzL$*4F()eU!ek8Wm>&u#D9_2 z$gHzVe38}&-!E%hR5-W|Nw@5;&Dhu#&uFXVl~(#VYl0icb;=_ zT!wn?rs5syL)57f)m^! z)2GhN-1n=g@63;T>eij9{OImoUENf(?OFTT>sgC@ElMJ7Ia>0G@S3VVm)yzZ_XyEk z#*H{0?*PWxy^78!sG21(B{!WWVOos6#Pt*)16`)W6+R-wM{$b^1!*g6HT z_yR`I&X{K|8v5jgPU)Noc5wZzQ=||l1@-}_zbC%LzrM%DH$ZD!1yN)PsG=4PK`wTV z`qCQ37nG=cvEx~FX^QYwyilEtQ}%5wPjLQA^r8i)*~pM9m9URVi4^m4$7&od&0?v$ z8n`3)N>j?HG95giz#3|H2+ zJkh|aXt59uY*b;ktNsG9HL}a@^z6!uxb*9pV%m39)7q6WfusS4S;b2F zNF|>$eElH>eK1JFn&p>eEm3-yTt6xqK?(*c%`bHqvEGAX%*-a zJLUBupLV%b81V;f|0G-vpb=)Pb;~dg9)u4r8PZBj!3DUNXB>>Bp36(Vuz)N-`_<4f z@;9l)U@_Gt^ycnj*f~AkRDYRCw-91_a1RW2S|I=D;pU(J5##{AlB+s8rafseWj#yb z0No~6etKt`6>iVlnu1ZgxglXLf4_$6)eVMw7Hgg&js}2K!D?`It>%cQh~^T7vZ{c# z;>`Xo`45d z6Vsm|xE z;z$$?@_)J|RIhBmLxS8A#&~4cwb36}`Th~)U0(d7%Dz0!lKYHmY&7F@(w^0ElcSAJ zKGaX}IDmd8fz-|lcl!{^Ve5_TKq%JW@L^UxU8Xgy^NW{0RIjmMo?(dv+UNxxU^bxY zp5dTHLXk-?I1*n1RR;4Li5B|Eup*~;3PU3MG%WseGi&TdcK`uXQAkmf#KN{bn)g+9 z#(bbm8dY?BmyY*flZ2|EvA}|w)!dTj&0k0 z-&OV%j)j1qQ=i@VOVg(f=g@OHg~i4%5&4pf10r8I6Xkzd&OTAn?8*$sT61}1 z6s>K3TzAIy|2L@F84Y6A>y};Yi1$&>TP;c^xu$ZHThOKENW1SLDLK3HOOZ9tMqFab0@cx(6!MHCEuP%ragRGq$M@;& zYqzs%0GX6s);fYwo;Tgyt?U;gX8mRrbLK^l*Ho(Rjx)aic>@ra=0x$Ma`0B=A83RpC}3W= zRQ_913TVB-O7G>_UNt^GvPp&Q8+b2DLcM*fjtGhm_L@XrG}D|+xBrEJHs9Rm8`pIb z%e@mbtw?o9Fz<9ZN{(F*l&waLNrr)c(4MY8`dNU4K_X0b5}ah<>V|3Eui*AhG3;21 zmpmpfD3&%luEwHvW25fc1!dx+{FdGEd;7;PC9T?TKa@J1>bXo%HBqyp#YRz#cg>Nl z!Jg%`@_VqalP0~ickPItsNiUMes^4z=zy}=(@RG;3f*cdrQ|b!QCRi>`gG_9*tFQG za^zdNBjMulp^Pb0C=(avQj#R~q7!REW((qTyc2A_3P#aJNm=@!M9Y61N&lI5{3Vf& zAEnOYA;Hc`o6$neVH(rw1O0hTktihnW1ibN$>%7(evY>@&5`rTOZ(kKP=0)$;Y7Iv z7S%$rQu3le@fFN=2@FJvuFt9VT%D-Mwi)jH8FnbM*yQ(1{TYyQ6Tp%Hm*h~LdL^8a zq)bikuU^EcXLC~UY)ux7-A#zHNGZP>53b}HCckQvrV;g$0Ab~GZ!t`-W~K~SdO=~v zVd17gmW=+KIau2Ol;k zFRFiTDx0<`V2lb+*O_0=Q&e9l|2Br3vPf1UdR9rFrpuBtCdQHWMVvp}1ggXZJmgRj z5GTTUik8mHE02Gup3NsQa8^~86|Q>hft>LMhA&*A^+iUJEMr)Iu)K|2P#-r@HaFFi z;wxeRPHI%{7dul*X%=sinpozlkge}NlMRs3D7D4^J{3M1M^yf?xp)j9I?nPBZQ*Pq zyZBhFJSoZFDa;zZkC|TeH_f;PS!;DyU^Ng5m0UkPpxFKJxPLhBJdsA~V!R@dW zuOXRNiBW*JG4G$hT2Vre8>3iwb_PwzgRKmkvBFI4eeqapbqb0XbfQbtb7Yp$$a2h@ZFU2C-+fi$(b{>@Z80yl#0_=S(!sv;>M@aCukoP6(F*R`70L3%%tUEk=|35m-y z57c<_f}!F=_To3goa7#*AzQuYDTR@Nc)XiI1w7T1L6nKrt0%d(oNGoZl%PVDMN1Di zrf;-uE?^Mx+gCy%O-kB&nQcz{;CKZdM#ke8Q2ziK84*gnBJ0Fj-7nDDvh)_uUF$!t z>Dee8G8i}M9NW?$RNBcxNgAm*J{{yu#U7nQ3mkWw6n`GE#4^K^2zY zg4O0t#~22xxJy===(a?9#Eoid#;j zc9?uB@ZjuqB{siCR}rd-76I=m9L};M8?dDlipPg1gJs%(cQ9Z`t7!rf|8rUWvz?9a zAST3PNsJ6{zA^+=-`R~O>c1SV^#`m~#max3VVU+V3vv8v3fWV2-Rlc2UyY=@K$@_p zmVXHmkX*p@+yhDj!27N)(?XFXg23*fGH7bc&4QP!XuN=Vdp!@O1VIjNmPo@b_9I0D zy&rq>dKV@hRZeCI57p=l5^%W9KC45scuB#!7TT%@xQ-?;uvrv7#5m*bXJ6jfO%Fl6Jhr==g{|IZAw`A_-RHBV0EejO z@g#k_>KL%-Z&W=Qw^qg~Wn?O;OtO8tphMEm8O_aO{;0P5k73oj}NEpQn;v5b?pPN`&PmR|t<990ezoZdA^46Tl zqfEvW?5OFIN6~=9k^fCp4(ia>#>*^}bT1|uEve?Es>QGLA5MfPWg%EbiXZ_HI9VO` zYY-t$v69JSyJ!WIZw(WmM?R|A__Znd$_5Q#zT~zS_q}6*BI@)S(#Jt3k86x2&Pad( zTg*N;$elBPIZxx8U=(DG*XW===uX*Z53Pg(`YrC471b>HaL;VkSj<F&U6)hRx-d#_VLi3=LN)V{9-{3#`RC)2*{S17ef03I(a1%zZ$g*R-<=hjYV!nP2 zO6qw2a?u@V2tF>ed&1pqbxo0rfxO-|S=`)SH^)#Z5^d&BX|Js%TIqdBqP{BX0*3+# zw+3OSF>8>hT=1)R1$f$gqd*||wX3S4>a4?MS6y9O|8Zt*#JP^d@ZT==F}6c5Z=&4p zp|^T3L4CWB{TY`-3yneq=_89M}Au% z(`I;sk|nzHd`~`VT@~byDFg){u>v5KSC+KZxBdiI;WNuSw}QT@T5Zr#x5gXRCqsa- zIDZo{6$V=Up#LiU(|E1C?e%y{E6u8ba8Z6~!<}=!Tk^QLZo;wm)l|;XuWf^8#`Ua@ zy`SH!g)^NGPqweHD58jtxX(VCjm=XCL;$!Kn006|3MX^W$%&h?4znF)q?!FyUY6oo zvXy@?je(*D`O;`%;)6KYISJw(U%$LC{&(<@|Ao^$ibW8#Cj&b<8hoYXI=}AE=Y~c+ zilj@s!Gv9L*j)zXw-5EZwcq4$39CLSn`qbOx69kaInpLZ^xnchZGxm2IPG|&MA7Mx zGbXX09k+dQAqnq|l#Z-Gu{e#+3q)yToC@5`nb>7S5Z=gM88gLrX|cPL7}%}kpH)m{!iB&2I%Fs6^dXDeDCic3 zi{-YhdQ2G8>D(}>3X@{gFl5D;%q5WR_a{2_UT#+`dE=I*m+Vx(za<9A8mmykbal_(2^QP8zR zKZ!Dki&cOj{*1YSLvqTa`oWa^>`HN(v0ojdzkN9GeppCn>Tcyh%Q#95HG0rwq|*6$ zq}GWKqLtMs#DIHiKX;OPl3uh4fO5k0dz0Z!gx2aDPTgvHy^q&J(*dwHX7=mAVV&P& zIYPJ=Ok($&(~#4W#DGcg2G1t49eGz5ItMAQ#5I}=_z;0RBDdml6)7exk9ExAcvQ)Q zql2nURg&<-Z_fp3LYcb)(Ml-|)@r`Fz$2M~B_lqzMM3s{OcOp-H2fpv^7BNVX7F1{ z0}GGp&!nw%8+MVOGkJ6*YhX5=Xce$+aW*(v=kYW6MauTkDLb#v_h4aR4m&+!vJm)4 z0dtR&0yewxME1gcRflQvrJ(Y!7vlr-(zZmwE(DokcioAaq{j0iiG5K@TZ{{52MWlH zCgua_4J$bxYsS~_%h4K4kxia-+=uMxS;`i_^Wu{YSc*Pjo)DzTbAHMsu=2Ojz9;Td zM}2ltV)AXcU#sIQZ?d*7Ee??9dzmIpNot^0KKO+>pIeQU8^eB*-aIG~yeg#8CAS9) zc2@jqG2TiMZc9v7LqLo3N!m1O24juK*XuT#)`LUDX(f68kc}pNmgo}>dm|+Zb@n>h z{hkP9*#O5k{TY-coe5-Mu1O#u=9?dr!TR!*B&yU;z1C!t9Ume;p3JE`JUom}oJI<8 z0Yy<#Pbdm@^Q7+%DCRKiO1`E0S#FuuK55gLWS(Zls)B}sO99Z5qmmRlr+|MPqQnF46i~)Ki}t5;l01*)9w8v1ijbOzohkR*|TgeD|AXVitoZv-jXu^njQH!HNpAy3;~ zUcKMd?x7hLS1cS)5i3lm{xniklIQA}b`m0cNc5y)tr0c-!74weI%v!ZsPD=S2`kgj z2e{3W3BtY86TiEFjJGRAr@`0N&8mq|N-JLYzXYLk)zRK%~*5b2Vn%*nH2w~Z8p*N7>L9;sL5PsU=` zzw~^fBC+!CDJo9+!pM|9upBbF&myHFFff4=uf#Z#lFxy{kT~3$z?;DBSYtKL2@5Z8 zB2Lhin&a9t>uJ0$Pjy=U4GmpkBivFA4*U7WH+GVl-rat&rhQty#sotgsNX$3HCag> zcCQ&zB@){=F+ zB#eST5-nNFhb^{IykEkt$vg?c6DGvImwdK`tlMu+OhY?m(|IP1s^bMrzd~JD{nn{! z3PV^)rif_{sVSoPyeg3Fo>p$VefdykKs zE}m6hV>M817v{{`2Hr=_S&Vn{*}FC(&z!Wt=mpXBM~{$DxXw;u_lr+J1I<%7q2%wA zi;o}Wd-&_zN4^`siN1kU+;`voZx`Ed-xQA?0}Q|4*FNL*mAAiw=SMgnIDF*ooWuE( zOee?j|NZpeKKVOO{;r0bH7U4Nc)4$R=q2JeJ14eH;co2 z0*gsqUbm#*INHAIv$mRiNK?9M>9+++x`>e_4;HSL!@m7@@&gzBc87V$GYG>4^18^C&W3 z9nBnYpML$s&0(A}`K4*Om-)3slKjlFs{K0F@5bspKSB!lnDxA7hM)_oSe%QVMvNy+H@OCFtTgoTHY|0dZKZGGn zckh;U+!cj{Nc2y+>510zmdnxoE zu%$kt2N9QmO018CV2~4nVWN{^k~X> z4I<&wxt?{C17SWgom(4VvJ_>>hTFSD6KmdoJ)^1Q9uHSitL6opAS6ryk2XZ2qaoC} zO%mny2r?;7Ks-;yC1YAD-J=q6Ezx0`BOU*l^TGV*2NZCZt%iPPOdi=?E1nvPH# z5Y@=S8YGKi-Grm>ux)9Jq1G~UkvM+2=bynrk||`>?0*-O3!K&U8qq&rSPDp zu%?8?FC~PX^?UWN@*^v{2#~I0z?FEEWJ&9B3u`HR=1FPDsBlnW??*gfMdPZZqS_|^ zUq&tdmAF%ADf$nXwFt;YjA}+9ws>N7eGJ_%?>Rm5k}q;^TvuB8EjdM^qQAmZ%j@S<2_(LM z*p4n)ju{n{2OuG=?exWEq)+wX1_1fgjj9`Kmhb2_gCeVSXYJnUSfZy^*GF#H?gNjG zf^8mdCY||s{(v>XC%;SgJ){tUrt90s;)hb88P~GOJDuB@4$or=LDM<+g2D#d;%hm4 zyK#^BsFY1f9_Nl4=hszrws=dr@|Yt#wXNyZ2M&FP6;kJ88VY+3D# zITJVd1E#uHH^G;`?SoWrxR6P^I+-=QG1IxPdCrGKE3e;;w8@ve=C=j-;0E~!6#uvB zCjEZ{DEv>$MOA?AvKK=xI$Pa30W?~7aNsFPvSp31C{_p4TXE0TT6fMb;! zZ#3G8_z;)lc)I`#y_^P}4@g*GaaY&x1yx5306Auo7|G(>X-o#Hm1;@RDypSV5%<88 z+K?bo&RBU`f0a;Mz*Kf_l8mWvadlL(Ts%_JFfMBlU?nl;Ys}AJm1~m#xQhDZFbNze zRONw-Lj*Cr?EWIBiZ7up+DZvkbsaGURxC*~<4I}Y2pwJuQPDm`xXU$O4A^j5un5w1 z;$eEd%TI;=1Ya0dzaFf_RI9(B*pT%(-wSrTcvA+qJB+0tKC1iYiO*wXmXnKM5!%HJ zrDc7k^W%GD*>~m3FL$K1*kkVR%@btvRM%NaaZD2W7v!L3(QOHGVb?ptv(2%ubAEf8 zVLKaf{sR_|Gg_Z7m6uK!>=Dq&b?@bV{{hE3K{%3Uu9+WiVT~G&g?h!lqD$+b2{#Q7 zj(Z(_NAcCZtY(Mt0wcWscQ$2K1x=4eK%?~uxXRJj3R@9S8MP z$(#{Wo-)=s_FPf|@fPmx#NX83g&noF=M zR&|ll`3-^l5T5!V3pZf-r)AZY^d@@Cz#8r0wlG==wFc^jL^%BNVoF)LhSAlWhgeQK z4OA{RkWk&&86%J)tPl`yzv{gg$h;t=$V-S2Z!kap8De9%Ct>o4*xpB4aXts{Y=73Z zZX`D}8YeXO8ZJlvDz!g_V~nyl1t&kkIV0nc*RkOM|1n%%l6_kJ2dthJ>!YFL517f~ zg8aSuAF$O__&;D;dMAd#J@XwqqqW4>UMXAT1kJZv&ue6m&IgIhqr20B?_1n&nb%OL zNgh?jA9MC;87G4`_Hd=@!OotZzXQS_-ncAv_}2$lp~#E)Oy#>^(he zhT$Fs0lFN`r%K9Wk$l5Jh|NNAuQJc!4d@QhZP|1H zGS*F#-UZ!P%|Qy))_|o~VAF$sxt~vcge*z1F+eeEhI(K@EFKvEoiv=$7d4JazupY( z2wSo*;cH_i+TtK#IZe{{ikX+>SXKY=y~6NiVE;W@5x}(!qNNz9F})kR;FMpb$(>~J z_Zrj4elgxjie{%8ZZ(Y6W)b)?a3>ibz{Hduw6hs4QEns1Uw1o}AO66w^|(;=UE-b7 zU%iQoaygj}+!H^)?wS_edWo3q+)yw2T(bnOu9Z}v6A~KD4m@BD!TV-~`RAb3tGrHZ zH6%O^n<3VX5?b;EqHXhoWco>3DSb5+Did-iWlr4aX*@!G1YE<a>7=clq{H|1gy2(CHtud z+#ibm0b9W$I@j?A^j7t~+!O|!!6#KbDRkN=+&nL6eiiug45)6OyiVFH={2xC&Mf9s z5iy@bNKDNgRGb9y{j&20f7si~lI-S!H4S@$cshl|QqAkGJ+ZM3&`uX$8Q{aLs^YP(9BHeh!5!eo$md|6utmW^hBE)8 z8yd}sAiB|H)9+~$q*sJUBJ^Jks%r}otTKImsq?fwpGc2h2-+xTl(mnA!Su%5^;{=7 zv*VX_X8H+LM4Oqr^*>-%$DJ0NLS)waHI5K{Xf`3=Stjvo55|u!l_?jN5oz>!-Fvc< z=K%zUjRZkqx^8`%bQx>od}MA3vf_5N^KGKpaKzt%MiatxUvJJ4P(3av&$BNR5h7X8 zx)hqrVv{}6T-q86e!q@@E=?&=K!Ac_{14c_uENf@-^aywiCFR?GWVGkDUp=L7=WH` z8Y~2C$}{=It;G0cu9!r5bBSp2<1CDEla!uZ% zSP1ZV-WO%+@)R!1lZjO|JG3QPnX-NYEq5EMb@*>+vUJCDr#{udniBtknsX2)jUc{% zSB=Egx{)7^o@GZ_`dBtgqWipYNgUVp8-Lm2_G_yvCIee$njF0ot$myCtHrD#L(^#Ha2OLnz-HM1<8i1oTkLuglMfLvg1ip64PLAU*Wo1TtMs{Xx2Nq=#aVwdI_&fKNf);A#oN0^O)E z?*;c%PX`7&9d6BfHjQcNC2|H;6xHkWzpDf)zln*GR&`M%v0AsRWPC=63Uh3GxqLcL zm@61e7@0BbwcC+jUXkl`3l_hs4i(tW+`zRxUG;m|@oUW4mq%07Us83{=(ayUsaqfE zx@dDgKg-!qi7nl~NToX58C~a8WZde0dj7Q2?!40uGW)t-H7^!;>%ofs&gb<~m2!Ra zd;jp~=;&nQ#ZJ5bX#2T<556nfk!hdR{yBtD)Aj$|x*=rvnDmFfeI4Cj-WuH&61pdf z*LL8EylvcRaee?NTL1b#TGx&bx>rK^+d_o7e?R%Rm;TO>zpLZ_+#~+(41agUzp>+Q z-296f{^F(oM2&wV!haDF`a9me)JIr8_zwfWm=ZnR=Ct1b$IEDRKvTfy=lA{Vv;P3t zM6)2I-L&cYyy&O4sLsBjf48IWx#;H_+q$9O2zt`nzqizH%(287Jq4w(M)8mIPgy*E zf84WxWMof36-o}FkTdr*Y0%+F;+Z8g)XRfQQsoseEk$~_LyTDyRH;xygW06Pj^>a8 z`-fR&;{-QZw%qa(iObUaTkf%3#l!V@g!C=HOJimo7R?DXZ zBSbR&zJ3qF<35*p{Pd=;mLSsrsh9;s<-4n9h_us&%d}W@hu#>2q&KDo`;07uDQOM? z-t!jRoq2!P*X{*M(bZsG{zs`5NZqnb`|WA=Z1fKpo=|-f%;Re^(hcZD2Y;!V?qs&r z=Dd`*Ktf*9bxg_Ir_Z>gm;w!yWfZouQl|rOhQh#r{Q)bVPosikGi|z*itr(iZy~>6 z*XH4-z-ag19#jnIWCpQ4{(u$q?j>9Qn!=uXoUe)3H&^~BC%Rzu1brD?N#xyStvoZn!kKP){BrsOuHsH^=vN!P4ZD%#p;ikeANdBB|KbnwbJqvQ zB>nBN;{0$lO$!Fxud5$Bk7{OeMdOm$_%S~s$i+qe|ZFL~MH0HWJ>i9A_e2hO?Qby#D z4SsS_qkDI4*385pCZt~cQW@)U-$u@pU2+?on6 zoDPU^7zq}5<9o>;8luT+gtKEb!8^8L!`}yF2yNiZd@tG-csFE25+ZVY>mtJ6TXYyy zyaXH~^$c@>><{4;bJhG2r1? zN$=KBs8onO{v*EE!k6ls*Ds+0$K(LIYeEts*K6ph$5;3L_tn;2L5x3OHg7`TLDlTL z{%1LT3jh`xy}J(|(_oR{;?$|zdQ5D|(j=#!s>RF(V@pYm3^E1Ni_pI7$R7FB}J7dy(edFT^BiiLMQ_+_;ECsJ;OKp5qCy%IdMtb~Nr-JjnE^?d}r=F@o2md1CmM%Xv5zN(Z|8dXgfvH};ZXf~ndVy24+%dXQ4 zA}*6*2@-k-ri!RXW}(d*Jf*GbrqlJK%Qnm@{A-c^M+i978^x-R%j30?tKd4`s``w9 znc1U|u?F*mT1|`(3pOR-FOU&muOoXj6O)z-4J)br` zGc{3HZ0>V{Jtx`m4k+Yu5}FTs0ZC9d<@sLU(N(=Cop;Pi^u!%ib7*Rv%v8mGGOJ(f ze;}s>*gdGp7-bw2WJ|vh)A4-KGb{7}UXZ{V3F(`y^ZOo|!0}lZ6bq@kR>+x5hD_0A zHq7#d&6*tTCjIze|F**!3mE2wb8^gp_4rXugwJwTw#{t&%W=bJ$9H`lPF95imb!J)p;v!tNYB4RRvy$R+&^5F*QM9tL*OLymNCONYHE$TA<0MANZI zD+r-j0yW#4=6gaS&|FsQqrx&aeD`fCgeMm*DIH>)ErPH`kXnB zpI@6^?mFbXvj*F-Mh#9$dkAB6Y$MAK1BEl4>9M%%7IC5kQbLm3BfC8-1*FH}Mxa^~ zu_0iUCea)?CYRl_%id88;>L04unT9}Ajx_vjB=v1cmTsCF@7{SK?s24Bym%DOBkuM zalCd0G+EQDPmRSevFbl1XO-bZt{z!}PVUSR&q1BBVsG<+nTy498mLe(a`r>gBijM& z7S9dh$(j=gvrR?~YXQE+JusTZOFOP%0*B;%#X_;&=3OjdH}$Mwfb@CVZ@o``RU!8C zJsZap*1-{rQh&?K^QO1QPZ?S4-rQ%``CgYUPGD@p-0DG_{3}Csl1ohV^8(;HOrX>H z6s*o$F|y~zk*}ciAjve3c&CF3HZ(G&Y1_m1Tvdv5No|R|w6P2ub<41x* z{iT^u8ax8%e68P~MNM5hFOvu)nKOEQiqkj5(bwmtLSa#@G~4x53q zySrkAd@ni>kZEy{btuX75Y~DYNXA67kT_VJkTh^*0>*jfy1~4)jq&My37qT6=@4O9 zXZ=p*2YqrEUpL=dfmKAxXhrRCku&tOlHNV3u_-o(&%Gvsz9-USF>zrPB#^2X(_fRx zBcd*YzM1LigBFmE8xbH`F9*K3GW;b}LHXU>i!Vl?b(pBl!5U{gIl}apS%R)AiO_hV705FM`H2^Gh1~ll@uYKAwV}N0n;i zfg63Z?+R!)ZWGU{+X7~3yXx>gWiR~!D>9q2w_BXo;4yzu%>j0r*itIUkj3weI-RqZ zVbwHHR`F>v;!IL3z%{LhuSTtVN}1Bry|{XZ1g%kI1C(%aA@z9uh_-I#v$6;o__<;v zSqk0oxaB>W3{%xqO&mARuidXFZi?ut{D0h4dK<8J{#+hfpk*Xl+S>6?CUdR?BMID6 zqu{@n!UL>#oBZ%T&$dmf`kkVr-0bIaW>*&1xpD{g84DW%lO3Tx5S9>?8{5O+eJ~%2 zUY1=8$#&0xes?1talVVj=TWw9*U)sEu_@}J?Nyksd|u%A{aoUQ%S-3W(}KGVKo8Qz z8{{3@KH5ZG0;+@;b$3UHHTn|n5bR8v!KPjZX0v=5VJRyR&j>=78XZ4w`#K(z3832RC{l} z&d{wmrA%1SWz&l4C`KP#p+!G$CiwOAV|+`%=gt%4n`Fzyh@_$Ok;da`!28Y}*&Ao{ z-Lf9F+y+CPHcVFFYls%xV2MN`8bR`yegQEcnJS7_&a|bSUeK$t*^l&{qr7`vFkzUF z@H8Xr{*Y!1mh=*#P%9es_>rn7*u-(&zJOFPWG_E}g9x5n+;W9H=bFR(i9Zg8Bn zV6L&A4Z5Q&m3Qz{eQ@4kG`(dTBfn-WdzePtS*kSowwNO9X;k|S@$D1aDatM~0{O-u zX$OM1TCENMu){6td4_Yy-CEb;z?At&;`%Lx^OW*&p8Jhwy&)+u@9caw@|$<)y98;0%YpN-sg>kUAZGz3yzF^E`(qdRqLaLv| zj3f6DGP&IAUVLH~jk2vYDU&=eT^I7Da<33wf#dIBf)R`Xt=8@)OOVJHy6fO)C_kdn z#8~^OCfC3rkfiV|DEaWt4=s?msM2CRr7K=DD|;|{5Q^2B+N+OqO}X3Yy!^y?WbaZP zG8|r7^D2TC))K;wEQjW!SXi4aPOE;=6sXv@ix5FPx1q@5Qu;?D#weO{)Ch_x;tqKi z*QZt)#fWS^wrGniPX-}I@kF4L;50)drNFy9pOvNoRjwwt8hTHRw=0geZr-zQRwXXd zXlPa~2ymJ<$QAT(nx*!Ze4$vHIFIRE^>jssowVt4PO=9!4L+6eG;Xo(%@!@yy8b%Y zZ8y7H=5XO!`vV4*af#tIXVw=FIf*wmtEbOir$ezj;UdJOH`py`V!B>0N~+dw#P~c+ z;VJw|-qN!iV(QH5qpqb#0FhDY6bzNnv0JHUAif#Ru}M1Y+(ABX8I;>&)x@+ptWEG6 zc(SVLQ+h%(>&8esFnQ)^L8-p%z6usfQrq=HNCTfGZo2=IQ^4$wF16I9IQWn;ac$$) z3PXSNn`a0NeYQIL3#I8oUOilr`+^o}DLVS9D)>0ryYD7RrIB&#&&+QnLF3uP5ZVvl z%xtiY(wHv}zMBO;7*PMra!{o_nH>K5_IqCU_zze!BZ%m>Uz^CbVk|aILAg!qVmCuY zmef46LGGcf6PLnXYgV$N>Yh%1Z~*ui(d*Ek&nG<8Pdn1pV6(=7mq6`9X0<8v?YjH* zPIm3v*()UTt5}wE8g-Y-fJ*#wA*R=+A+k=$1lVZ?M8Ol29I|}!VmHlgv7BwW{tmUx zmpBF#Rh5(j@-X0d!CIx96@{Z$14f0#c!4Bz=RyuW6jpQ652g+JmQ`gRalrfrmx@>p z3T#AiT!RN;0uD*Zahd(5kX=O&GJ1!clnxO&#Y}v7ki#DN^2FNE{2h;8+WqiSvR1cc z$oWaBs-GTuZ}NUuX(oCOI?N=ySYHB3`8I86D8T&0)kEgAxLp$j4=m9>ND{yAX*~&XaO0IjvQ_Ia$E}$ z4LN&vyu0ok73=ETN17f)s8XNT>y8ya&ZtLdAn>w2LgekF`&wT~FNFors!`wvk=?>|i5AjOpg!27}<5V8o-zvKER3qn2; zqV%<19i4#q!vmJVkK4D~zWN_)U;9ul`+oRR34Z(|k9L5Iblh~TMLLMUT!4M~B>NDO zd931ROMus$CVRY#U>M59ON!krSCQA@#9a+tyiIYz^>x_|U0I-<=Jz8d(cvGkJFLyo zYbJsCOa)r*RZj;&&0M;^C3{oxsjBmZTCuqZ%U5x%adaHu$tuCOgfw)@QZaZR#_?BVUTxfQ}KroQvG{Re9&L zd&}i`TRZRD)7fgm?=Q_WdXguB(21aHlK{%7!J-56{!Hlz@_q9bhic+Db?NMq z)vfMW3A6Bns%ISXD+ z$M7+yT+#jgeR)XVgQR~%xZ#gRCdhAAHmB9L!($_Bzm_{>a!3@_ZE`AlA_Zf{DEgt~ zTCoFdXnclvWr;hoS^(E3VyGY0tOXuu0m!^hv#+vGeS2|8d)&^OlnG`5o_Y5X6szVF@7SZ-X=mB@}Al}a(u*W{3+(Vs{x{b?OefCAY zi{97nk*>44jv0>)`=%Z2Rek4}Bc2)O7*m`X*V98-X$HhGjcYiI?Ds0{!9aZpzbkyip^5LB^pq$=eqx5E1Fz|`O zQyZl^?baBfW#3{5x{^kcO^M~3>R2#Pfs7)6HA;M<*_f<#Bs+&M|H+!NlQzOvECs5c zI9_A|@B4jz^zM-Rx|2GT8t{XcL3_PdR&}pLJl(YV5>vg{Bo#!vrITn{&eA!`zCH5= ztv+jHTE(Hcxp<@u&C%hjmol^YVYkMbujPP4vn`(9E{r%o)?S|{r9<4B6uBqmYf}Yq zHLrp&trlwHA4$^@(X=ctuw5qN)=<_p^S4;FsenN#dkyvO6m)qrwJm}B4WY4W2$7ZU zUZ{>m`XpZ^op?wr8;L5j0aOp{xr&0RN^gv~a!Srb+)$6oDRVqE>^$lu(Qa$!r;wq?ZfgCy{$v zJ6~NX1ie|Oq3C9YlEH8Iwz)cA8HogH$r4Zm#gg~m;<`zFxH~l-m~;);6zxjJQ|6}G zZR&g{&SX7wX5R}yEOnvygr{!_JCx(QgU%YovwC;!)|Tq^JE zK~Eq#;F5q>sy|>o&q>&pDs9POXTz$haW_2D>r12hH{TF$l~^ZsO&2689GUdEVn9 zz~H?8U}9O3Lz`8UeXV~A?}I9Q2q4m@k0tSksiom-E_0Rvdq<);fln>V=}6@3Bb*Pe zDPgTU=(eRAf3Ponzce|;+T7}R|1c?rFe?~h%RzxhZ)SE&dn0^pDOTB@REKrQd~|vdyVu5_KVWT(N)X3lx+Y8m!O*wnRmC?s@j0r~ z<4~jm_>&}2$uaK2L;(?rWG%9cgo(aGwvN7sZdt6>;bwX*mc>O(Ih=(36gj1P`Mh?^ z0zIcdQ}(D(Xa5&~<7!zQSTmez^<7E-@w^gMHV-^GA35BS#pT?NTifc6jK)Z(^j&7% zS$rgt;nz3$$sk79Oy{cz!WXJEGsSToAbZEVfnJgqV|3onSL)tp@}k1^3o8;yn+f-i zY!9veyRNLST_enjwd|)w%KJ!#gstwWG%`FA(i2NJYN^(A?j`lu9FK;DCIfAZLk9H| zkeq}nT*M`bJTM&0i5+(t4rkobUdFl?Z0BzGht-W`j8*KD1?lRx{4HY+&Q;&D_Z61Yci$`NtdbE__b5D`Elg(> zd*!YDQ8NRFWM}xIyqO4O{l&)_k3~6i&v?BKj}yJ~?$l*nza?NRiBslx)tY`Ea1JeT zhYBrq|x;0Ep8#W?sOQWch)$n_*L2@baoD(8}P1ROaLjlmrp&fi|2H+dfUv`TI5 z1C`UX%_AcrFm29eC%c}Cx#n%lqw~DPSK={%G`cHaFLZ`g0-Z_BZO+;%-+e;WW*s=7 zlW8C6n-1h@l-Z^1hR7-16kJ;kcs^Sm-Wtjc4%y`m%F$JbE?4qIL_?fcF+$!gEN|}F zPcYwtlc;>DYcvK#)1o#(^+&K~5s`%Mhh>12+dl+Yx{qS_P*ESIJ7gjog6=I~`D*v6 zdEzD8Gj*Vf(WsK8u%pMsyWLsIhkpJfCJ;#gtjx#imsNeiPABK?O}HUUYamI*Nep_V z*Ct9%u32rcbXK5}rGHbO@iAQxEv@M+CAQTb0}rdl(kM3h-CgO3Pxrba7Lb9_l zy4fO@(j~=9+}j1CK=x?>Y#6e$S3fq2_ zD^viQLjyy7oArIztySO?(Ou0YOut~Ev6mZVaESkCnsP0FDlIS+2?U^}fKFVB{jH?E zKx^mqIZyX%1d}iWiXSL`cO=bC#rhHOIyrROQg~Rt<0m=k$>pz2Wnq*(gHhhgiVY#b zoIv^%r9!RbWy-5FG(Vl~tXN}Y^{J`mASd0B^<7*Ig5;SL3|W@*Y$=hTOFIqfI`O;} zvJK3T9J%CNCR-F1eY0@O+79fX;n_>E@?t24<+UCGQ)|20Eg7ZwA0@+=z<%NG%kRk} znR4Xk?mCaA!|(LJ^Lm!YWUxG@WT#*(6_Awo()Kp56cU=?<#W3iTA+eM%Lb(_e^ohZ zoz%`C(ZHB#P__vOv&$ePLw;**oQ$}i{hs}~No}3vvsf^la=&Hk<$*qWpb$9NziB)B3b=*lu zlpv#~YW2R7`{;0=V4Ts3uMwU>87FM&`8DySLNN3_+2F~?u$pzzsJPDeOa!&tGIy;b zUd1w+m6n%xXd|xA1iT>?<}Uz74m`4s)Gdd`02^~HJ>jh*uC#Z=vB8zaIqbe0o4X2> zvFLb2e_?b^7&2YvoR6zdn%%QuiYE4=;heBrzKyvdqmVe1gzw((>tx*ze=38sFbd-;q*jqQ(eha@WAMcJX124uCk{Ap zIeVEV&P`E)64xe-MQIP!5M0C2?5j!p9x|4v@;zR*YzK%2o6JZvM~A#&l^#jE{n@FL zSk5EkH>oCCIU7|!%(<(u*y<#t;a1cf8|CtP9+&D9(U|WqnUA?+c*`o!i{qU^+Li@t zk9S*hTU-+E<$GfriWAql9g4x&TYmV@KfcVUG#OBr0K(ML5Th{T_$@sFrU@_;5HTLb z6fz@^l*~yy@#cu;j7_-#wx51my89!|5FdH^J;bzz-0B7Ek+4P z*fR?3-KmO|?0o*`x4KK3iiL(1d-{-;>Pe9?QEeTn`v|(@1%I1kt7FZqOiKq<8n{V6 z&0~p{zefyyuN*N=5Zg%IX7`b7k<(RYrgBRfQQ;eX(-Rp-hdYr zjj3yVvi0md^-b5d;>k7kFHCstKt%j`SBOSXnYvTA{!D+uYbM%g@7j)urRnwM_l77) zp~PEa$($XA!I#cVRm&wanY6J#J+vIhD0bq-@v`K4T}-uR@jy>@PYqdu+1}8yipbIe z_7V=1vUsOy36FeyhhR`VU6B?iytKf6R>VcTA^Yq~RLG|j8B~$gd2ks6-B8pCt4|oa zn%?x-vO?O<31*V4tG1A8(_&hOBtv%$)&-d`$;J)$C_2<3jiC4#-k5FnzS?=hr_OWB zYW3tsN@t^f;;4YcJ~fvn|Ee4U;xjDfaQx;bZYdU*o>ZG_F)00=Ozf*Hij^bfH*Y5r z#$Wd1`<;K3>8eV4AH^LV>fJNnCaIUWu4&jV0|fFcB2y1rxhRfM+0;)5sSk&XbtXWT zy#9a-WGs#|@RXOhS6=O*A?<95Q<`|QmInCF-g!)8%7NZ8vLSEz>ZF*%PU0?_1WpE? zK`nCECv>c3PL+B?VmRmI_o`!pX<(y6QF&&H*-n|hRlOaC6$9(aF@YNFoyH|vCzNtD zl&1-=<9t%0xMYj!XoGTGb*JDZs$Eu{{}*OOYc@=%`}@P(VREi7gGgCo)T?9pt$Zou zlsDTW(s@mAf|rrQCvu{+SHD}P@hv(?mbViWq1hLcN%xh zWXDJ}Po%aRG@J^+wQYByVYLI66|YheCrWy|;v_`0=$I>iv`lA%oO$ygiiE3PkVG1a zI#T=LkiIR=ovLi0A!^85EA0kUEQ&Yp(o4HjeBALS_A}d-lauSz=PhYjkf+EVgR$cz zz|1*4i%~P+P@V9qSTAAXW^+Mydu-Djg4<#E{gxP_bhsXRj=0yY+ZXMh^}!NDCViIh z)HbxJ0`T~;gKBzk2k}J|!Fixtnzfj_;+z5M&K{vA9EU~CCx~E$(r3wJ@gB$XPYk{s zyXhD}|Io@%(05-d7bwo`mY=R?#igq3mUj+he|ija;}361EMfCvkK>8q`>E%`;PweS z-Ybv=Xi$`{TzRP;yKQxqt^rO2KEzot9G`BHO;dp%lt_5BPZe{XzQBRXe%%FT?^vyV zmG6kTY7!rdmecs4WeVe_i%sL{DSXpL89vgpc|eYhN?`$D+8vU*lcV3auvTDFa4bdK zyS_=Ch#j$Z<6O9NaGs)4CoOSP8q4h%@~et$^iK6Q$Yy9<+_On6poOHNl}&;ylr3jx zyM&2?e$VbI)O;k*$}bK@9oAlKp@Pi5AcoO$0W$Y~(J7r|BnmS2J?SIYj-KuEPX};6 zC(&Fb&=GPxFsnPv*$l!@tTfqKYGI^5?&XN!Pv=vQlIIeV2!f4S6F9=eMcAP>CC85R z1scxNTa4v6^DFZrOTFDCb5tW{z#(!@vzn2FD9(vxIqpFK2JeFt0dV}eB4SfO^Lfp> zD!`$k?}5P%xK)#3S{9}DzIfXUr?E#BEqhIE4U@}ozx0U}#{vE{yJN44{@aM6;(Aix zRhVt=*VX=CwTo9DCfn=)qbSIfVxBxxJ|bDdDNC~HwT?&$e(ziS-lDOU(Va}f^SYF;a)mA^__{=5b#gsn6sO3&vtEWB07-FFG%J_a^N z=3x7w0#2s<5|!FBJZX!i5$LnDGaM~EH-TH?ESQdIi!*e<>O&nA$|+CM>a^OjPNz)y zpu0sHt}PcO`CqA?$iMh;Opcg}2(|+OZa`i#TW2q+AFYNwTZHm|?zDcg2nj$SUGYx! z)!SM(XZ@r5PPj|07ea`Bh(qY~jCV?rM&o%la{g#A%eS?ap&xZGH*IUaYG|)?+Ho>U zpGGC+d}oph4JWPdCDE$rveo&7XJ~LOuQI%t!gJ-<9ZedO@|`*dtK4u=6YwXr?Cc5a z9V~5k`0q~g#?k4*pxkz|4cnxM(xa1d$04?Bjp86U5o9Iqk66v-)RZJ=<#pp>)RT>i z_tMI#23jU;dV|{D@6bMW2*95->#7xq>h-^bH<@y1M~!1U z;IT2eP#FFlP%ZA5o(O~6UC@zucH2Gk&4Fp9qqJS}&o;u4kra6(CwUSdg}6=)`=VA< zDo$P@z>gDG>dR`P(SQ`^V6IKBwz!EJz6&jKfV9TNe1i2k&cy)C#pFUPOmtvOeb+uc z)IqKi6;b>E{_`P2tcG)F2yD8Hm7IVVLCU)npyjy?1_>;S&`&gSsswNCv@+Rt*IXYv zhu&Ozh*67yl2bhtT-V6Db6B(a+|P3`h-$AQ)dmkL+vR1M0Q;DoUeovjeP_bT|rScX4ou&}Q&8~GfL8T-pLuds^I z7m4)=4RlinW{X!Rcu5y-mhhHaloO@O7QR)tr(c9-nf3Ji*tzfu0Y z*c+C0J$~l!Bg<^k-$CT_`;bE+Us6Yxh*e@AE|O!WU2TpBMsg$(NtE`2i#0w(=#Is4sro;NpR?1ARc}7#W|u zcl}r;``lUE{LV6SY~zs|eSW0OU^m*xxy^$#_5Lr87+v_18F@3Y_{ErblJ!zJxs6y3 z1{mC$R(*5Rg|`W84LPGhNx6ju^_-ljGzAU&MwkFqXHj1I>NF;y(UAuBt}6B^d!(ga zKxRNF@8nC-mHuN)CiALT!b=nZl$oCR3hL!mONHm|epTkr5d?V)7;6*26i4T$gd$ge zQW(RVX|jd`$_Sx&%jXIJ$@pY95HLEho<7o60LBuh^|$B7tUP;a84Lv4;3;Hio%d5?!%7Z&5#Z51H&*E8#7Y9Nlf^wmAW+k9%{LK0f}roBPBO zZfwcy2gj1OA)6@Tufk5V#LRE>svs6wUCbp3wHv@7IeZa#_t>e_Pb&;?`ii7HpKsz_ z#Z$VgsWLGkBd2T{o5`yC{*SH#z?)9SCU%K=)8WTrmgClD@z^N{_4^tz(`;8+n`#Iy zkmR)L*Y%@|nS_H8dqpiYg!41%Hk{0>WM<^?GqTq`pUMh1;r?C3NDnJ*c6h>xL&k!8 zQxZ3saB8CN`imM23pj-13n^{6}5(k@5s> zKvfiaGHDYYQIGcfhuM1L)}L#3JJ|O%u}JnCAI^)9_zIV%XebYT5Q}e{*1n-x?{CkR z9ukSYA354M`Bon;w1GEE=wR0@h!Z(yJJ%4?7bs?jRV%UmnJ~{^kE47#`HVeVd^z8( zeR6G@M-)eUSeYe|r^=1eC!8=s8F})D4N-^5O}GVtJbK6UOS?(_*`d@{TnxoGr)Qf0D*FP7iB*&`fdOAs>jd zz}#?jcXHrB<@b?wQ#XfPrX`C4yBHY)Yk4dMl+&7FqC;Q_u&3)(0qm|b6z4bT> z6nHTt4%6W@f1@s_5snoJ=M0YFV11EE``|cKEN3Rk7~6mFVU2JcEgIXVOIbW?)(gv% zUHtTHo-IQf=%9W}5+ACE61p5(;|7nWSSxOe6a68lNLw}~l+Q0!rvIy{1wDVp3zv+< zo%nhDNu++Wr;8cGfuoRt|Hqw`TenpB^si$*9nv;dMQO~|zDr0S7;P7%xO81`LX#&F z`Y*U;lN(Go#SL(NtFh({!zu*+Mzca~Mzs5NA_A1k{;d$HB9l_YIFeyiM4dz{U$C7{ zQl6#Fdz`-L*tC&(e28p&s(l@~YD0Wd?gi$YZ2QDCXYD+2barC&$(p;0a`ARzp;1Rg zCY+1^n?{(#cqqE>Z5j}^Y&*L#9ohO<3_Nio%LXgivyqo~n`mQ$`=2vJEG#7J5<|Fd z3T?-7Or3GNs%19_@;8D!mI#U^^vgg=6Gz{GWT5(3&s6>${DNN$9+ z5(#nfYqBO~y{0<5y~+u$*=CNlPG(%AY|908+N2ahhpN6N0%3zTi#;0@JG&wsFB{*( zrtj9A)?#Sd#4Ah=S(K?+qp~~FL@Dnu)vlj;K5!MeosnmI)UP@;yD@FF%Mak*L8R^= zw05=SJeo>2CdN-szs>{bn)F3a16zjEm{a(Hbq*?Mj33m+`cPh2h(tX`shA@Xt+C2w zv5|L(FqCq|!6<2}=xbU2;sxght>fN_gT>#u?h{idP6(t{ewDcuwu+XQI`uFHVg#I5c2x!Foe*Ulod=L4O3IqNqYw)4u_q-NrkWsX2o*8$TWe5x~9R48LbjEptyIuh%w3Li4#EV)Wnl5NgF_1@y_M4kXUiAiD7Cglj_`!3&9 z+wuO2Bo4s+d2c$M%3Alj+#j;d#j5>#=MPvqot{kkW-IlT*bm<3vC`*~Q#a9>?9Iyr z5c^%O&UPGNy2ib)A4LsaVfeCT?T{(OzgpWr!KuH5nu2xO>&=dTmPtK;mq?x@KZh)0 zx$&kuh`Z4;tJLloH`(eH8CzSB!#82|3SN885H3_Z`}*F{9z-8nf3`z zZNrOOo)j-o-1}S;V&%Qa)w*HiUYzSto&|=0IAM-7lPK^>f0<85Pi|WxEpej)Ch}FH zmxvmTmAVt>tfGiNp#lyj*io-|nM;zzncWrLU?}T`iQFOJ;*I>2o@n67$=COu(4Bf; zBx8_1#>UmIOijpA#p7ml5}m_uR57LiWc)M6VtCr) z+Or}>!~HsI%0+B8`bL<0O$JePTeew^ghH=bu~3&%Ee)_k4)hVm(z&WF@`c>ACAU2>Ph1 z>;ZhXFtib>9~&a-Sg!U`{PU8@U_*u4Vo!(FktbGT?pmrM_e_pB_oYR+GE}vn+m?M| zUR{g~e5oK~kt-Zu47Cv>RKC3`?u;19vfrH6F4UP**_kUO z+3>S0a_>e2%KVQYc&E%9Vfe$+Vt~t=N$?LQuxKkG83kK%OVo;#+Leob&nfN@r3+cW zsD`A+t2xS{lk$;K@LlZnn0cR$Z+@WJDutLD{mqQ7U~}HoG0t2Q4XRw5#qNfpPiN9P z!Jz$twb^<)t@xPKL+vdrLYoV4J}Qx)@HvVEV3qIOQgSD<*UUzV3(T@a0}+lYjc70q z_{9hFJUcR!@WZYYTf-w6c!PV|66c+Y3ABGYP$?Tup}>@emUK1gHKf6F)iA2haUejf z<(J}Uyz{EJt(WED>C>o_cu?-etcCx=xYuwnc)yJ!l+HB49>r~dIw&q1)F^M(b;>V& zJ@j-j2rJe2VS#R)ow6Da@$=9{&hEZVlFETx?Avt2EyD(Pn$G zfb!`q5yvo)YIc2ErVC+*qV}4A{XW`*^q>vmW$`FW;&W-W!AWtzdHD)B*q~OW=T1Yz zA5R5Puyp6hE6-o0^m{nUSvGQ;WLAG8CJmPkndZk2nb65KQ!`3H z>JYWbIrG{r51+rVmJ-g4R3@_~S+f;@tncT6uUl<7vqt)ab_CEjtd--;v^e5reZiAz z11GRw@)qCqC6TQ8h2+!Js9v^Gb*@fI*rB5 zQS2I6JoRMCWl3Pz&*3DKid_UWX`dwUO}}}T$OBW4{yb61)7w>>U(Y0|(L2V=`Wr>m z1W$aL(OY;Y&ifq#+IP#cr#a_2cbJaao>CRWn<0^+h3`%ug_nA7%Zg0{-#G@4PBD3c z(Jm2Lj14CxrPZWuBABR?2AduFsXUnSqDoihF>ZedMZdHRETlC0tZ1`9vxCALsX7~{ zW2L4T(ZB%DOmRZPOl0trc+*9dG%<#~4qX1FghCBgp*+K$WTI zI6t~0?fEpR3?&Z&^<1mDppIf!AwTHqdzP3^81D+|iKShT%}2uS-|_BUCmCq2B9TLB z1aUEr-MdF-jFR~zf)6t>x5ybmG<6U49?R$GbzA;rb>1 zemLu&;$%szUAz`-i?`-A)1hnn0Cv~{ zP9OSu4P}#7!un>BK@4@8;BQNh762L)Z-jB()OI?6S|j`cJkie>Y!cEiK-m@mhdmTo zAW$l;l%UZP!W5v;?tL{E!|NF8{};xyxQ3zF-{KCM9h@YH8DbIpF(jr~LWjdO*#z^1 zAiz|h!2WK}fzt)y51r!nt{M7djH9ymqdE@@Qlm6@tnwVqCvhXijToQvM=4(L;C#7+ z67}gs!KkLJ?HBpbD;vb6Ii${MuXRP3cqGd%@iH{CPhnO=7A*IkkSB}9Wo@BNEW1sd zlGx4x^Hpk@mb0+pR7Te`56{F76;+95ifKJIsTkYE9V;H|v$z43oUDm&F-`6_Xn6j! zQ<@jIT31YjtyOpTv^FBhj_MKt^fvTxbWr`SFh5S)tAs>}KS6i-*eFd@`Ia%F;mH@F^@qF8cLC@Ew-22A|U=(?4o+4Bm8C z<{8$C(6q^i6bxy3=}ZgZsUzb|&dtF$WdB3P< zN{#dSc-+$v`911~!>ww3Ea^(zR!1_8C&UM`Q}mZO8KVYH#wlmGM%HI_NUu3%TJP&P z&n|jDS7vJ4pTn=Vs}kFIk`z}X?`msq83pxfuN&@7)Xf|RnQJ*`_%^!hvy8>EUK#K^ z;4Nla##y1?qANat+!Jm#bJF4cP05@UkS)rnfFq@7}dwjv38-k9{x+ zj*laHI#|?tsEA<7&p$JZag$o6UvQ(vSPu)SEMbOpiI4x3KO0C)mC8uS=fF99+|bRk-D$C=2aGuVQlTV*tIoPMkF%58U2xx3ym*P&FX6>o*H zwbEWgI^b|#AA=w4Ef`S{htZN!cYvH6*z>t}4MtzpG}}_|^M?rpRT-3_!Ggvp)P_&j2UiuVgVcgtjzUkVJUfc=d&r-x;%Er(RzwmBNi+4+3 z+#(%#X~(P#1eKMGM3)YPs%KZK2OQ6>W`iB zl8cR}-;wpd{s@f9x<*Fbl=`R|*nC^>;r@WDgsi{X%W9)rQj5RVEaeIRvdZ$)UVebt ztsPiPg`v>Kwm+ccRE|KMtF(XMGf#i$>wC;im>%pj5A)Q2}vv6lgDP1DU5sDLlxWi78`OJEUK>k4{#?4!S&Y(mkHc{ z<7I5+1GPjy6|qx4M2g_Me1QGvd1CG*kF8{0J7d8h)Tx15hsU&K7J*H)Et^9jTwgE8 zr(ACpo@s*-(Pn>Q!IU7i+H1nB$Df*yht$<3aGb&8yl9@7Q@$Of3DO#8aO%vOC)^Tv z=DWMuk2Lc9l8Pc&IZrz$ab=FLB=aF7Z;sYkg1L0hm!*Hj_0JzjO~#}!uc#7}6kpbw zyEm+#*0-8DbyA6gdtG7K7B^w1BVVQF7X*gge)|5(Bs!gsi)lmUndP>dNADUe?Z4mJ#oC|;@|D3>~*4AGfun4~U# zNM$u+v>hSk-?7uAxwq}D?Y2zL@Y*|hyk(-IDy#Sk_?gHi_CvyA64mQ6kR8j#0Mm4L zovFmCXAl(2wBh2ujjN`+QPtL^M}xh!*qiH+G&?MvCGRzqYO4OBVJptP@i1YtbUmxW z0fV03iX?}yG^p_dcH04wzqdItclbs>I{fR+* z-Mh?38u6Nf2ibSQhq2fBL{M7dM{5;9$+Jnnt+NEjYMs#mHFlkOw>$6$hk^bQJ2uj%mP7Hc2idI$X!6qYDEQ1mGTfe@iK-7T^Bu^? z6Xy8gW*j3d-<2{^e}BK$r5D@3;b-2CShhv!Zc&$7wckA_b29YR>%r z@z`f%wzxU#v>?BksZ^&t48hqVfzxpbn%A`A(%lRD;M(aS^KD*sjdbg3LJtTXqqHrv z-ue&zp#}q~-9-KaujAYQ=i~sd7U(_`ss?kXX3vwz5I0>-Y-1gst%W>!<+Da z$1d@_>Czi2qK&0W@oji*tk#ZHblMB)33Hy?hX(8hUB&OJ8gJHl3?CVv&TWLHw>Rvb zGb2BZD=_7q*SYv=HJ>4H@uT&Ju2$g3xW?0JA2P=t}lxj=UCYtot}UJ@w=Z>H(2F(uTeh>FIIwW z%=&7GT7%R$l{6pLdD@4S=;AL-w_%0fx5kgU;AlL;tY(Lnn?vDD1b|L!Dl*wTnRs?+ zsO)6JYjWWGP4E_m_Z`h<#G|vdeOv+8tf_upPD2ie%T45M`@CaMfx-sDpcRnU8Udu9Gs(fI z3E1Ti*nPZPBUm$p)*KIrGfBrS{zj~^VH5`sU-Wx|Z$k0~(4}}z&dBdmIA#{FuDfjv zBqsr5eD@njKzgq0d2c!wtG*NHo|#*DDYf=_HZyx@l>m$Q6EzHnieEh$Vmu}fO-p!O z;7R#fWxvcc$JJYm74Dt74nv0|F-04oezgL)!a8JFf{AbA9%Lmb$!l}}jQVObzgi%n z5Kt=$DI=`4cd#R}bBj?bK1o&Hr6+r4wBhY#-Oak0GrU!a+N!@l8k+ zD&u?m0HxlCJdAKi!rAj%;aw3LIL(iL8R>lRT{b+gf1G|UF6cfKa5#B$(LH?&F=5Xx z6!&6T5^Y(csX8nrN3#6L!S;vc?8lVmdh$|<`wgg_LR30sq$^*2T3TsDsMXlNG6W6fCt(=^ilO4}dD>R!}1L&i))fZN4-QN+jbDu?sa(+IzM zSuK67#j^ypNBB%vLRghO4msXvRyKu3awk|(N4SPWD0j^yjL+%uk%@$E!f8a+I)LX* zXR{hh3hiVb*kt|8jdI9yHaY||wr-E-r2DRYmoNfC;YJsGp(X#F;ybme?=^;ak5cr) z1SQ4Szc746X$$JE?HbwMgxAp}yDR*Q)K~R8)M~#aM&NO_P={kI7hDf=DyuuW&BsWI zTOA8dE2}${-+!E?zUs`i(YjqWF{vRqR0;Xa z19Wk&{~7h`YqR%_s_1#Wx8Gxkt=0YV*nHoD4PO3Bc8CsR#zo*_mkr+mz8$rX=BhMM z1Yb`eo`*5d!#v*APuw4x%cJ=b*lnVE~9e)A1BFFJ5oW2bm%p@6sbO4zQtYSbECmT&!R7Izrw~zAjGjwgdMR zgK&ZKs`p%YM&5dR+LIK#(fgx1<#~`J05>4$TohPxpZ+Y_`RWh0A()0H!fY1vY@p~P zQoH-uoMduiL%uV8vzwPXUKJw0+Cic+*eEs>hoFX9{E^U@WQL?xF+a~%x`@(pJK`WT za51hd0@Dp6Q--u!)X7WU)9|>a)+*89&eN5|U&%ZO&w(^k{)3c->_@iQUnTC680jSa zY7diI?vS6)UD7|DaTZn4M`Y^PMfIA!6SWn$(!Qv-7ugbEqwu?1kEeVlicTch0@c6H zi_uuYBQgWKj>6cAcDLnHDb>fT+UAM4P1jfy((nc@VbCC7(W`#gh^q1@(7JgbhoVj( zo=od2{D?@{)m7(}p?m|#eh>3UPu%Wiq)4xPnEHmJt$zSuwidsaY1X;XRqvpPWloXq z^1PM)a7-J^Kv&PLOsz4GR9=v?d*@J|<51sunzWbzCFm3=SwJm2PU9^MEeJsD@&6o7 zpwGdz76h+fK%zSAlbn9kUXXV?`sBz*wINbCLhKzfpD~e2(%rX#`;hWpu*}hRFGnY_ zurdi_B1g5~no@uvNU(4Kp_TUWWr=z$S$7IP%lAT#v{{{c3+Qy9xGjDa1a>-e-nZyw@n^GPyf#~g;bV8fMO3`>jNq!9;w zAP|QfrlgVV^U9yELb^?72nH^qJj+mxcB59%>UsPru*-rFx;C7oJ)BRkTn+t&IJxnbNzvs_(zB?`y^(4N)M#m|okD)0B)jo!n@U2| z#d4PL_}hhG(vk$AlJup=Q-G77APd54MfnyRyhtt| z3mns0NL#%NmbXlS$`iUra;v?LyHNh^q9qx3)fA1BAU8Njepfk2|0ZxZ z6?NipXD9%N>Ym!~hu06;^Vc#QZP$wS8*S%JF)Tc7^N4izT)!eyB-!6~k8~ktZFM=p zl1;y@jo+vf^X`nALPzk8FpBU!|f5*a~aQNd>vo2Fysq<1s(CoH`79-b9F{(I^0I6 z<%PRa0~jmckYllMg2tn(V@r;ztAD`2oKV1SHNrG=Hocx*OkPZe1PC?@HeKjtXQj#u z^FQWUINA_f5L<{&{}X+M{P{C|GA-(RC~)~7BCh#67&Kz*W~lH@g7vSW5@$@8gr@5K zQt8bC9&w|#wV(xexeok>^(r6)Ji)gGx%UKhI%I;`ft>ZwkOa&bzsm=5gCb<@F5fg% z3oZ$YmUlKJue(Xx&t2YsVZM@neG>~34B8O&6YWkNdb^%f8;r-L`oZtoW26KZrd)5YRt=genz2W zCtbjN?fLffRTB1r4Yeq0U80#@w5^%#HT=Qjh5*|6b_qZpCwGSgBMl-ACN;~2htNTz zP`BZ+$M4SLAyL5VG2X&tv7SBN=06NZvhMS|L&vv2|0``9W@^;EUQo}#} z;s^fc+;4%ca)3Kz+RK-hxDvE}#Fx?7xcS2mcQ&h63_P4Z%M+9B*+4E#Fwg z?-E<^K?6y`w?G>LFm&EC0t(UbdmbSbm_77b&>th25$bu@FxVQnas87}p4xQy`-D}B zYc3%%jI>N^YUG9`He{4@EE)Yq*++bhX05hU!TGJ$|E8^mY{@O&Mg&@fUL5Ej$YIY1 zf}uYZ^lsh)p?8chgDLX0$z9aVyfPx`A$AItC3VXs9hr-w>W@-zaAj z+`W&rXiLF=oaSEcES3{`zT7uK8#1&QZ^H*=g+fDM_Po=2kE8>z8+3rSDC~9{QE5CB z`vBGDD;b=j=Vgt+hNRUx0f@U1wX>TX7|OdGBG_hr+&kaNtJ>lENE+cQYXLWen@QY8 zF3z80m}D=QB&u8vw6#7y%W0E0qJ#7P`QAJz)as>+hx%axTArn z+3(ZC&;x`w?q0OTZ){U_?OD@Da^p*=sJ4};whK9pD8vPsMsqZHa9a-?xAn{=+rta%!G*=c+1!B@5cH+n{ny8IGRgT=tA=~A6x@duddoLom znn(d;o0~m$!qXLbr`F^Ks^bn-(ktW zmZ@Rm_?-Pn7W6zka7$=9St9@o`%c$DA>wqIZ_V(j`Az7bX7o9RFEQ zHhnu=t)m@*NYMBY8Xp%?g(3~Lg=yWOL+T2ki5Qzg@W>o*x6hV~k%M?@xB-tvXUCcG zow~!3Irwm-e5s^HBHuB8RY%1JKU0sI0%O=sXPGm$ym*wGrLbj{s(POfjo9-a)?AD| z(sv`C8OQs?&T9X{1f+9u-dNmcTROQYG<_!b>Jb-ixn{ezvVKa}(Rxni%L>7I#+i|! zjEAxJw>mIbDiF;AagWp(S{MfREkZletKZ$>oyA>n_u)UCaJu0?%`O`{ncd~P+ZCnd7NNvo!^72iXr7ld2_{8SES3P3lXh3?1KFfn(bUJE%3sjP zHhALocoa=8G@Ta#3>2}xullqFox67y+30-asw`Woq8K93gKqQk{@Sx~pI5iw!|(g~ zk+!C;q~(#m<&o+C90sRHE^h8rr`udu^?fdX_2j(;+6A<%0X5$O#QgEJPX({=r2kd;&W! z<4=1#&+CJXd!U&BXyyRgm_R|$>VV$XpXPX*$H(V?e3C|Uoo-!fbZck#qwJyPFgTOs z&1Axmkc3*Xg1aBUrxw$s?;|wD@D&%tr`5q%Jo=0cHERZ5ykPH1z-}I6X$>xXO57(3 z?0D^DK-!%S{Y7}<$Z&SRy*Y*?6!xQa)#_m>1R`~j`-2E2i}?`RueF^w7O;wzwA0D* z!r|&ae=S6pr1BS1qIcn7F&VHG3j@^O-F%!L$920j>GkZZTid@b?0Jy4k@8d)<^}WZ zZEVI*%h>psgD-}&vkLa1L-nNULaIF7iGOHyafGy%`YrMg#%~WWFmP{h4O6Yl&-4hV{f|JL~)A@gpg%FBE?!t-pKl0(ijY8~9@C`%lGr=& z$)ei2P!gT^3v&dKgz%=>zg$G`lE`G@3 z&65O98#=bLJo&87Od=vFRjLh&5jd;ie<@#ro^xIWkO1{Xo? zlBI8nz87EyV5~15$?r4^30w0LDp>C;yUNqZc2fpJO z=K0k#cW15#M(hSSL>+q{6qpb0+(XmA_?3PVy_!IYDoqk3?LZz>z!EmudRst0#L=54@AKxZT^-ok=gcWU&!L z?)Ro9E%Q;YLTn`;-AmZHbz@EJ_zdijF&!9s8*W4BNT2VoN+H{@RZc(l56}yRP%zu z%gG5VQ;yAxtx+$SR);x~ZOXAxXoSX=Xi&yYU^0B_oy%sp^EW2Fcmo8AZS}|*2eNH$ zX^4^BN%Q>_UPv?ja?8MGN317iKFO03J6~Dyp;#MS(S1f7{JU8EGFt)EC&w&K;<2zX zO1oG(A}#ZTO+d>9|B)1;4xaJZRh|8C%wqlNwMG?-6F;1w>dk0R%nl^MlO^ehw5GO;#4*D;L8xbLui9qGPaCJ_zyr% z7coSn2D}Au7h$ZGxvFlHx5Uz>-@6~O`Jzx1i)|EE0hi@iIo@f54c+gahv9Df`*-ao z{Zaqk#spMdW6_2-Hfgc9d}+7i8pD|iAE{I>RqD0fb#(OJEKdDl=ga%NRN*eob&S*R zxa?FMHLPI{d%SY9h|j@whMA$8%mVtx^mn-0oeP$9b@%gXT6-UFx$eo?OLp576)FFy zmC3E!SQ@LG@-HZ#z;Y60XEqwU*h8@8dIp(Z>1B$XBJ*NPyOxKhxXb}-FRAHGKX7@! zFbb?@QAM*6yo;}WWfT`SqcR7bm8(~NZ9>muSXo9{W~BtY-+;*Jxi>1&06%dkz>Nm^ zSZZyLQ0&a?5T+5wXOs^37HNslXu2sFN6nLws z2!Ai;Zcf1%Avumpc1QtZJnJl?TqsR=O9zO)E1hSwDNV2yjBPl{ufk}w=(1L4y|usl zUddj|%ZDx2(e+x=3zG1TOX^>a6V!b-v6_q^PpTe95y1Yb3zXz3_bCs#vuTuRApox) zZz2jU8z}BWxJLjeVhUUOZI(q~MTbt;PvBci$HUd?J@j z*QnVT2_8`7jky=j&Q^%hN#*c~_CC+AlUE-ltD>IOf`K&@G9dxWktrYMM)oUoSmj6N zC=PHDQYG>osFeJxANGIhhyAbOY6f!ur6A_|`Sq0eO>Fv2rI4hk&+g6U>2CNn>=EO5 zxo?ETBj2Y&_#G?*8JM*H`O7&Yu*r5mwR|o%!9EJnu9Ih7lg+UjMImL{Jz4S5Yeuu z*TXuO(T!K(z7?Do$s{VD+xx#TrkYpij(KkXzWd)g`L~_?+Z+Bxh<{P@-?8D}@$%oO z;NM8;-{|q*82SIgyx}j*@P|Lp#mz`$f2eZ}=q?#79P$!qL;k0EDlJiH?UEe2_lkA< zHTY&(Fhq3dFHGU@mj_{AgOVESBi?yI8!0S#QqHy-K^v`>WhlcHy3u77x+v`dUBfwj ebF@hGYv1$#?~CdG>%{9{q3i#9`rclDA^#1lh5Hf! From 75cbf9f087810d8d4adceeda9e66b809aebb2d33 Mon Sep 17 00:00:00 2001 From: Sirui Hong <34952977+stellaHSR@users.noreply.github.com> Date: Wed, 17 Jan 2024 01:52:01 +0800 Subject: [PATCH 370/637] Update README.md update news --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index cc78ed459..ba3ea74db 100644 --- a/README.md +++ b/README.md @@ -34,7 +34,7 @@ # MetaGPT: The Multi-Agent Framework

Software Company Multi-Role Schematic (Gradually Implementing)

## News -🚀 Jan 16: Congratulations! Our paper has been accepted by ICLR 2024 for oral presentation! More details are [here](https://openreview.net/forum?id=VtmBAGCN7o). Note: The overall acceptance rate is around 31% (similar to last year). The fraction of papers accepted for oral is 1.2%. +🚀 Jan 16: Our paper: [MetaGPT: Meta Programming for A Multi-Agent Collaborative Framework](https://openreview.net/pdf?id=VtmBAGCN7o) has been accepted by ICLR 2024 for oral presentation! More details are [here](https://openreview.net/forum?id=VtmBAGCN7o). 🚀 Jan 03: Here comes [v0.6.0](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! In this version, we added serialization and deserialization of important objects and enabled breakpoint recovery. We upgraded OpenAI package to v1.6.0 and supported Gemini, ZhipuAI, Ollama, OpenLLM, etc. Moreover, we provided extremely simple examples where you need only 7 lines to implement a general election [debate](https://github.com/geekan/MetaGPT/blob/main/examples/debate_simple.py). Check out more details [here](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! From 03012a81fdc0ffef457a1a19ba32bf7d5769011d Mon Sep 17 00:00:00 2001 From: Arnaud Gelas Date: Tue, 16 Jan 2024 21:44:45 +0100 Subject: [PATCH 371/637] In some cases when trying to create tests, metagpt crashes. Adding some more safeguard to handle the case where code_doc is None. --- metagpt/roles/qa_engineer.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/metagpt/roles/qa_engineer.py b/metagpt/roles/qa_engineer.py index 81082ef59..0e323893e 100644 --- a/metagpt/roles/qa_engineer.py +++ b/metagpt/roles/qa_engineer.py @@ -63,6 +63,8 @@ class QaEngineer(Role): if not filename or "test" in filename: continue code_doc = await src_file_repo.get(filename) + if not code_doc: + continue test_doc = await tests_file_repo.get("test_" + code_doc.filename) if not test_doc: test_doc = Document( From aa969e0d9bba3b1f7a0923f40faf6c96e154eaab Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Wed, 17 Jan 2024 11:20:42 +0800 Subject: [PATCH 372/637] skip Selenium web browser engine test if the browser is not installed. --- .../tools/test_web_browser_engine_selenium.py | 25 ++++++++++++++++--- 1 file changed, 22 insertions(+), 3 deletions(-) diff --git a/tests/metagpt/tools/test_web_browser_engine_selenium.py b/tests/metagpt/tools/test_web_browser_engine_selenium.py index e38905b85..1b1439d29 100644 --- a/tests/metagpt/tools/test_web_browser_engine_selenium.py +++ b/tests/metagpt/tools/test_web_browser_engine_selenium.py @@ -1,6 +1,7 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- +import browsers import pytest from metagpt.config2 import config @@ -12,9 +13,27 @@ from metagpt.utils.parse_html import WebPage @pytest.mark.parametrize( "browser_type, use_proxy, url, urls", [ - ("chrome", True, "https://deepwisdom.ai", ("https://deepwisdom.ai",)), - ("firefox", False, "https://deepwisdom.ai", ("https://deepwisdom.ai",)), - ("edge", False, "https://deepwisdom.ai", ("https://deepwisdom.ai",)), + pytest.param( + "chrome", + True, + "https://deepwisdom.ai", + ("https://deepwisdom.ai",), + marks=pytest.mark.skipif(not browsers.get("chrome"), reason="chrome browser not found"), + ), + pytest.param( + "firefox", + False, + "https://deepwisdom.ai", + ("https://deepwisdom.ai",), + marks=pytest.mark.skipif(not browsers.get("firefox"), reason="firefox browser not found"), + ), + pytest.param( + "edge", + False, + "https://deepwisdom.ai", + ("https://deepwisdom.ai",), + marks=pytest.mark.skipif(not browsers.get("msedge"), reason="edge browser not found"), + ), ], ids=["chrome-normal", "firefox-normal", "edge-normal"], ) From 6c954b8455a8f9f3383a31fee5aa1da3758fda5e Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 17 Jan 2024 11:40:16 +0800 Subject: [PATCH 373/637] update context in write_code_review.py --- metagpt/actions/write_code_review.py | 22 ++++++---------------- 1 file changed, 6 insertions(+), 16 deletions(-) diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index 444af51d9..27e4c280e 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -7,7 +7,6 @@ @Modified By: mashenquan, 2023/11/27. Following the think-act principle, solidify the task parameters when creating the WriteCode object, rather than passing them in when calling the run function. """ -import json from pydantic import Field from tenacity import retry, stop_after_attempt, wait_random_exponential @@ -15,7 +14,7 @@ from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions import WriteCode from metagpt.actions.action import Action from metagpt.config import CONFIG -from metagpt.const import DOCS_FILE_REPO, PRDS_FILE_REPO, REQUIREMENT_FILENAME +from metagpt.const import DOCS_FILE_REPO, REQUIREMENT_FILENAME from metagpt.logs import logger from metagpt.schema import CodingContext from metagpt.utils.common import CodeParser @@ -139,7 +138,8 @@ class WriteCodeReview(Action): async def run(self, *args, **kwargs) -> CodingContext: iterative_code = self.context.code_doc.content - k = CONFIG.code_review_k_times or 1 + # k = CONFIG.code_review_k_times or 1 + k = 1 guideline = kwargs.get("guideline") mode = "guide" if guideline else "normal" for i in range(k): @@ -156,24 +156,14 @@ class WriteCodeReview(Action): ] ) else: - docs_file_repo = CONFIG.git_repo.new_file_repository(relative_path=DOCS_FILE_REPO) - requirement_doc = await docs_file_repo.get(filename=REQUIREMENT_FILENAME) + requirement_doc = await CONFIG.git_repo.new_file_repository(relative_path=DOCS_FILE_REPO).get( + filename=REQUIREMENT_FILENAME + ) user_requirement = requirement_doc.content if requirement_doc else "" - prd = await CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO).get_all() - - contents = [] - for doc in prd: - prd_json = json.loads(doc.content) - product_requirement_pool = prd_json.get( - "Requirement Pool", prd_json.get("Refined Requirement Pool") - ) - contents.append(str(product_requirement_pool)) - product_requirement_pools = "\n".join(contents) context = "\n".join( [ "## User New Requirements\n" + str(user_requirement) + "\n", - "## Product Requirement Pool\n" + product_requirement_pools + "\n", "## Guidelines and Incremental Change\n" + guideline + "\n", "## System Design\n" + str(self.context.design_doc) + "\n", "## Tasks\n" + task_content + "\n", From 8831177a22ac3f3e89ddce6bf8528919f2826722 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 17 Jan 2024 11:47:16 +0800 Subject: [PATCH 374/637] update test case in test_incremental_dev.py --- tests/metagpt/test_incremental_dev.py | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 2fd319117..ed8fc7e78 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -27,7 +27,6 @@ IDEAS = [ "Add functionality to view the history of scores and perform statistical analysis on them. The original dice rolling game could only display the current game result, but the new requirement allows players to view the history of scores and display the statistical analysis results of the current score", "Changed score target for 2048 game from 2048 to 4096. Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8", "Display the history score of the player in the 2048 game. Add a record board that can display players' historical score records so that players can trace their scores", - "Add limited time mode. The original game only had a default classic mode. The improved game should be able to support limited-time mode, allowing users to choose classic mode or limited-time mode from the available options before starting the game.", "Incremental Idea Gradually increase the speed of the snake as the game progresses. In the current version of the game, the snake’s speed remains constant throughout the gameplay. Implement a feature where the snake’s speed gradually increases over time, making the game more challenging and intense as the player progresses.", "Introduce power-ups and obstacles to the game. The current version of the game only involves eating food and growing the snake. Add new elements such as power-ups that can enhance the snake’s speed or make it invincible for a short duration. At the same time, introduce obstacles like walls or enemies that the snake must avoid or overcome to continue growing.", ] @@ -41,7 +40,6 @@ PROJECT_NAMES = [ "dice_simulator_new", "pygame_2048", "pygame_2048", - "pygame_2048", "snake_game", "snake_game", ] @@ -74,13 +72,13 @@ def test_dice_simulator_new(): def test_refined_pygame_2048(): - for i, (idea, project_name) in enumerate(zip(IDEAS[6:9], PROJECT_NAMES[6:9]), start=1): + for i, (idea, project_name) in enumerate(zip(IDEAS[6:8], PROJECT_NAMES[6:8]), start=1): result = get_incremental_dev_result(idea, project_name) log_and_check_result(result, "refine_" + str(i)) def test_refined_snake_game(): - for i, (idea, project_name) in enumerate(zip(IDEAS[9:11], PROJECT_NAMES[9:11]), start=1): + for i, (idea, project_name) in enumerate(zip(IDEAS[8:10], PROJECT_NAMES[8:10]), start=1): result = get_incremental_dev_result(idea, project_name) log_and_check_result(result, "refine_" + str(i)) From 2b522ffccbf48a8f9295793960e1ea02bdbcd927 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Wed, 17 Jan 2024 11:55:09 +0800 Subject: [PATCH 375/637] fixbug: engineer action_description --- metagpt/roles/engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index c83a776c2..7c91ec6f9 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -297,6 +297,6 @@ class Engineer(Role): self.set_todo(self.summarize_todos[0]) @property - def todo(self) -> str: + def action_description(self) -> str: """AgentStore uses this attribute to display to the user what actions the current role should take.""" return self.next_todo_action From bcd143d220392aff33937c01e856469fa1f3636a Mon Sep 17 00:00:00 2001 From: Sirui Hong <34952977+stellaHSR@users.noreply.github.com> Date: Wed, 17 Jan 2024 13:48:44 +0800 Subject: [PATCH 376/637] Update README.md change paper link to arxiv --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index ba3ea74db..026920700 100644 --- a/README.md +++ b/README.md @@ -34,7 +34,7 @@ # MetaGPT: The Multi-Agent Framework

Software Company Multi-Role Schematic (Gradually Implementing)

## News -🚀 Jan 16: Our paper: [MetaGPT: Meta Programming for A Multi-Agent Collaborative Framework](https://openreview.net/pdf?id=VtmBAGCN7o) has been accepted by ICLR 2024 for oral presentation! More details are [here](https://openreview.net/forum?id=VtmBAGCN7o). +🚀 Jan 16: Our paper: [MetaGPT: Meta Programming for A Multi-Agent Collaborative Framework](https://arxiv.org/abs/2308.00352) has been accepted by ICLR 2024 for oral presentation! More details are [here](https://openreview.net/forum?id=VtmBAGCN7o). 🚀 Jan 03: Here comes [v0.6.0](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! In this version, we added serialization and deserialization of important objects and enabled breakpoint recovery. We upgraded OpenAI package to v1.6.0 and supported Gemini, ZhipuAI, Ollama, OpenLLM, etc. Moreover, we provided extremely simple examples where you need only 7 lines to implement a general election [debate](https://github.com/geekan/MetaGPT/blob/main/examples/debate_simple.py). Check out more details [here](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! From 4e13eaca6e6cd6475509ea2aa0daeae32e0e0a73 Mon Sep 17 00:00:00 2001 From: better629 Date: Wed, 17 Jan 2024 16:28:13 +0800 Subject: [PATCH 377/637] update zhipu api due to new model and api; repair extra invalid generate output; update its unittest --- config/config.yaml | 1 + examples/llm_hello_world.py | 4 + metagpt/actions/write_code_review.py | 6 +- metagpt/config.py | 1 + metagpt/provider/base_llm.py | 4 + metagpt/provider/general_api_requestor.py | 6 +- metagpt/provider/zhipuai/async_sse_client.py | 90 +++++-------------- metagpt/provider/zhipuai/zhipu_model_api.py | 59 ++++-------- metagpt/provider/zhipuai_api.py | 65 +++++--------- metagpt/utils/file_repository.py | 1 + metagpt/utils/repair_llm_raw_output.py | 22 ++++- metagpt/utils/token_counter.py | 3 +- requirements.txt | 2 +- tests/metagpt/provider/test_zhipuai_api.py | 41 +++------ .../provider/zhipuai/test_async_sse_client.py | 10 +-- .../provider/zhipuai/test_zhipu_model_api.py | 23 ++--- .../utils/test_repair_llm_raw_output.py | 32 +++++++ 17 files changed, 156 insertions(+), 214 deletions(-) diff --git a/config/config.yaml b/config/config.yaml index 6dff55b4e..f41f7d276 100644 --- a/config/config.yaml +++ b/config/config.yaml @@ -36,6 +36,7 @@ TIMEOUT: 60 # Timeout for llm invocation #### if zhipuai from `https://open.bigmodel.cn`. You can set here or export API_KEY="YOUR_API_KEY" # ZHIPUAI_API_KEY: "YOUR_API_KEY" +# ZHIPUAI_API_MODEL: "glm-4" #### if Google Gemini from `https://ai.google.dev/` and API_KEY from `https://makersuite.google.com/app/apikey`. #### You can set here or export GOOGLE_API_KEY="YOUR_API_KEY" diff --git a/examples/llm_hello_world.py b/examples/llm_hello_world.py index 76be1cc90..219a303c8 100644 --- a/examples/llm_hello_world.py +++ b/examples/llm_hello_world.py @@ -23,6 +23,10 @@ async def main(): # streaming mode, much slower await llm.acompletion_text(hello_msg, stream=True) + # check completion if exist to test llm complete functions + if hasattr(llm, "completion"): + logger.info(llm.completion(hello_msg)) + if __name__ == "__main__": asyncio.run(main()) diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index a8c913573..3973d089b 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -157,9 +157,11 @@ class WriteCodeReview(Action): cr_prompt = EXAMPLE_AND_INSTRUCTION.format( format_example=format_example, ) + len1 = len(iterative_code) if iterative_code else 0 + len2 = len(self.context.code_doc.content) if self.context.code_doc.content else 0 logger.info( - f"Code review and rewrite {self.context.code_doc.filename}: {i + 1}/{k} | {len(iterative_code)=}, " - f"{len(self.context.code_doc.content)=}" + f"Code review and rewrite {self.context.code_doc.filename}: {i + 1}/{k} | len(iterative_code)={len1}, " + f"len(self.context.code_doc.content)={len2}" ) result, rewrited_code = await self.write_code_review_and_rewrite( context_prompt, cr_prompt, self.context.code_doc.filename diff --git a/metagpt/config.py b/metagpt/config.py index d633c7d28..e837b329b 100644 --- a/metagpt/config.py +++ b/metagpt/config.py @@ -144,6 +144,7 @@ class Config(metaclass=Singleton): self.openai_api_key = self._get("OPENAI_API_KEY") self.anthropic_api_key = self._get("ANTHROPIC_API_KEY") self.zhipuai_api_key = self._get("ZHIPUAI_API_KEY") + self.zhipuai_api_model = self._get("ZHIPUAI_API_MODEL") self.open_llm_api_base = self._get("OPEN_LLM_API_BASE") self.open_llm_api_model = self._get("OPEN_LLM_API_MODEL") self.fireworks_api_key = self._get("FIREWORKS_API_KEY") diff --git a/metagpt/provider/base_llm.py b/metagpt/provider/base_llm.py index d23d162c8..a50cdacd9 100644 --- a/metagpt/provider/base_llm.py +++ b/metagpt/provider/base_llm.py @@ -89,6 +89,10 @@ class BaseLLM(ABC): """Required to provide the first text of choice""" return rsp.get("choices")[0]["message"]["content"] + def get_choice_delta_text(self, rsp: dict) -> str: + """Required to provide the first text of stream choice""" + return rsp.get("choices")[0]["delta"]["content"] + def get_choice_function(self, rsp: dict) -> dict: """Required to provide the first function of choice :param dict rsp: OpenAI chat.comletion respond JSON, Note "message" must include "tool_calls", diff --git a/metagpt/provider/general_api_requestor.py b/metagpt/provider/general_api_requestor.py index cf31fd629..500cd1426 100644 --- a/metagpt/provider/general_api_requestor.py +++ b/metagpt/provider/general_api_requestor.py @@ -79,10 +79,8 @@ class GeneralAPIRequestor(APIRequestor): async def _interpret_async_response( self, result: aiohttp.ClientResponse, stream: bool ) -> Tuple[Union[bytes, AsyncGenerator[bytes, None]], bool]: - if stream and ( - "text/event-stream" in result.headers.get("Content-Type", "") - or "application/x-ndjson" in result.headers.get("Content-Type", "") - ): + content_type = result.headers.get("Content-Type", "") + if stream and ("text/event-stream" in content_type or "application/x-ndjson" in content_type): # the `Content-Type` of ollama stream resp is "application/x-ndjson" return ( self._interpret_response_line(line, result.status, result.headers, stream=True) diff --git a/metagpt/provider/zhipuai/async_sse_client.py b/metagpt/provider/zhipuai/async_sse_client.py index d7168202a..054865652 100644 --- a/metagpt/provider/zhipuai/async_sse_client.py +++ b/metagpt/provider/zhipuai/async_sse_client.py @@ -1,75 +1,31 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- # @Desc : async_sse_client to make keep the use of Event to access response -# refs to `https://github.com/zhipuai/zhipuai-sdk-python/blob/main/zhipuai/utils/sse_client.py` +# refs to `zhipuai/core/_sse_client.py` -from zhipuai.utils.sse_client import _FIELD_SEPARATOR, Event, SSEClient +import json +from typing import Any, Iterator -class AsyncSSEClient(SSEClient): - async def _aread(self): - data = b"" +class AsyncSSEClient(object): + def __init__(self, event_source: Iterator[Any]): + self._event_source = event_source + + async def stream(self) -> dict: + if isinstance(self._event_source, bytes): + raise RuntimeError( + f"Request failed, msg: {self._event_source.decode('utf-8')}, please ref to `https://open.bigmodel.cn/dev/api#error-code-v3`" + ) async for chunk in self._event_source: - for line in chunk.splitlines(True): - data += line - if data.endswith((b"\r\r", b"\n\n", b"\r\n\r\n")): - yield data - data = b"" - if data: - yield data + line = chunk.decode("utf-8") + if line.startswith(":") or not line: + return - async def async_events(self): - async for chunk in self._aread(): - event = Event() - # Split before decoding so splitlines() only uses \r and \n - for line in chunk.splitlines(): - # Decode the line. - line = line.decode(self._char_enc) - - # Lines starting with a separator are comments and are to be - # ignored. - if not line.strip() or line.startswith(_FIELD_SEPARATOR): - continue - - data = line.split(_FIELD_SEPARATOR, 1) - field = data[0] - - # Ignore unknown fields. - if field not in event.__dict__: - self._logger.debug("Saw invalid field %s while parsing " "Server Side Event", field) - continue - - if len(data) > 1: - # From the spec: - # "If value starts with a single U+0020 SPACE character, - # remove it from value." - if data[1].startswith(" "): - value = data[1][1:] - else: - value = data[1] - else: - # If no value is present after the separator, - # assume an empty value. - value = "" - - # The data field may come over multiple lines and their values - # are concatenated with each other. - if field == "data": - event.__dict__[field] += value + "\n" - else: - event.__dict__[field] = value - - # Events with no data are not dispatched. - if not event.data: - continue - - # If the data field ends with a newline, remove it. - if event.data.endswith("\n"): - event.data = event.data[0:-1] - - # Empty event names default to 'message' - event.event = event.event or "message" - - # Dispatch the event - self._logger.debug("Dispatching %s...", event) - yield event + field, _p, value = line.partition(":") + if value.startswith(" "): + value = value[1:] + if field == "data": + if value.startswith("[DONE]"): + break + data = json.loads(value) + yield data diff --git a/metagpt/provider/zhipuai/zhipu_model_api.py b/metagpt/provider/zhipuai/zhipu_model_api.py index 16d4102d4..a7d49623a 100644 --- a/metagpt/provider/zhipuai/zhipu_model_api.py +++ b/metagpt/provider/zhipuai/zhipu_model_api.py @@ -4,46 +4,27 @@ import json -import zhipuai -from zhipuai.model_api.api import InvokeType, ModelAPI -from zhipuai.utils.http_client import headers as zhipuai_default_headers +from zhipuai import ZhipuAI +from zhipuai.core._http_client import ZHIPUAI_DEFAULT_TIMEOUT from metagpt.provider.general_api_requestor import GeneralAPIRequestor from metagpt.provider.zhipuai.async_sse_client import AsyncSSEClient -class ZhiPuModelAPI(ModelAPI): - @classmethod - def get_header(cls) -> dict: - token = cls._generate_token() - zhipuai_default_headers.update({"Authorization": token}) - return zhipuai_default_headers - - @classmethod - def get_sse_header(cls) -> dict: - token = cls._generate_token() - headers = {"Authorization": token} - return headers - - @classmethod - def split_zhipu_api_url(cls, invoke_type: InvokeType, kwargs): +class ZhiPuModelAPI(ZhipuAI): + def split_zhipu_api_url(self): # use this method to prevent zhipu api upgrading to different version. # and follow the GeneralAPIRequestor implemented based on openai sdk - zhipu_api_url = cls._build_api_url(kwargs, invoke_type) - """ - example: - zhipu_api_url: https://open.bigmodel.cn/api/paas/v3/model-api/{model}/{invoke_method} - """ + zhipu_api_url = "https://open.bigmodel.cn/api/paas/v4/chat/completions" arr = zhipu_api_url.split("/api/") - # ("https://open.bigmodel.cn/api" , "/paas/v3/model-api/chatglm_turbo/invoke") + # ("https://open.bigmodel.cn/api" , "/paas/v4/chat/completions") return f"{arr[0]}/api", f"/{arr[1]}" - @classmethod - async def arequest(cls, invoke_type: InvokeType, stream: bool, method: str, headers: dict, kwargs): + async def arequest(self, stream: bool, method: str, headers: dict, kwargs): # TODO to make the async request to be more generic for models in http mode. assert method in ["post", "get"] - base_url, url = cls.split_zhipu_api_url(invoke_type, kwargs) + base_url, url = self.split_zhipu_api_url() requester = GeneralAPIRequestor(base_url=base_url) result, _, api_key = await requester.arequest( method=method, @@ -51,25 +32,23 @@ class ZhiPuModelAPI(ModelAPI): headers=headers, stream=stream, params=kwargs, - request_timeout=zhipuai.api_timeout_seconds, + request_timeout=ZHIPUAI_DEFAULT_TIMEOUT.read, ) return result - @classmethod - async def ainvoke(cls, **kwargs) -> dict: + async def acreate(self, **kwargs) -> dict: """async invoke different from raw method `async_invoke` which get the final result by task_id""" - headers = cls.get_header() - resp = await cls.arequest( - invoke_type=InvokeType.SYNC, stream=False, method="post", headers=headers, kwargs=kwargs - ) + headers = self._default_headers + resp = await self.arequest(stream=False, method="post", headers=headers, kwargs=kwargs) resp = resp.decode("utf-8") resp = json.loads(resp) + if "error" in resp: + raise RuntimeError( + f"Request failed, msg: {resp}, please ref to `https://open.bigmodel.cn/dev/api#error-code-v3`" + ) return resp - @classmethod - async def asse_invoke(cls, **kwargs) -> AsyncSSEClient: + async def acreate_stream(self, **kwargs) -> AsyncSSEClient: """async sse_invoke""" - headers = cls.get_sse_header() - return AsyncSSEClient( - await cls.arequest(invoke_type=InvokeType.SSE, stream=True, method="post", headers=headers, kwargs=kwargs) - ) + headers = self._default_headers + return AsyncSSEClient(await self.arequest(stream=True, method="post", headers=headers, kwargs=kwargs)) diff --git a/metagpt/provider/zhipuai_api.py b/metagpt/provider/zhipuai_api.py index e1ccf0de5..a6f77477a 100644 --- a/metagpt/provider/zhipuai_api.py +++ b/metagpt/provider/zhipuai_api.py @@ -2,11 +2,9 @@ # -*- coding: utf-8 -*- # @Desc : zhipuai LLM from https://open.bigmodel.cn/dev/api#sdk -import json from enum import Enum import openai -import zhipuai from requests import ConnectionError from tenacity import ( after_log, @@ -15,6 +13,7 @@ from tenacity import ( stop_after_attempt, wait_random_exponential, ) +from zhipuai.types.chat.chat_completion import Completion from metagpt.config import CONFIG, LLMProviderEnum from metagpt.logs import log_llm_stream, logger @@ -35,26 +34,25 @@ class ZhiPuEvent(Enum): class ZhiPuAILLM(BaseLLM): """ Refs to `https://open.bigmodel.cn/dev/api#chatglm_turbo` - From now, there is only one model named `chatglm_turbo` + From now, support glm-3-turbo、glm-4, and also system_prompt. """ def __init__(self): self.__init_zhipuai(CONFIG) - self.llm = ZhiPuModelAPI - self.model = "chatglm_turbo" # so far only one model, just use it - self.use_system_prompt: bool = False # zhipuai has no system prompt when use api + self.llm = ZhiPuModelAPI(api_key=self.api_key) def __init_zhipuai(self, config: CONFIG): assert config.zhipuai_api_key - zhipuai.api_key = config.zhipuai_api_key + self.api_key = config.zhipuai_api_key + self.model = config.zhipuai_api_model # so far, it support glm-3-turbo、glm-4 # due to use openai sdk, set the api_key but it will't be used. # openai.api_key = zhipuai.api_key # due to use openai sdk, set the api_key but it will't be used. if config.openai_proxy: # FIXME: openai v1.x sdk has no proxy support openai.proxy = config.openai_proxy - def _const_kwargs(self, messages: list[dict]) -> dict: - kwargs = {"model": self.model, "prompt": messages, "temperature": 0.3} + def _const_kwargs(self, messages: list[dict], stream: bool = False) -> dict: + kwargs = {"model": self.model, "messages": messages, "stream": stream, "temperature": 0.3} return kwargs def _update_costs(self, usage: dict): @@ -67,21 +65,15 @@ class ZhiPuAILLM(BaseLLM): except Exception as e: logger.error(f"zhipuai updats costs failed! exp: {e}") - def get_choice_text(self, resp: dict) -> str: - """get the first text of choice from llm response""" - assist_msg = resp.get("data", {}).get("choices", [{"role": "error"}])[-1] - assert assist_msg["role"] == "assistant" - return assist_msg.get("content") - def completion(self, messages: list[dict], timeout=3) -> dict: - resp = self.llm.invoke(**self._const_kwargs(messages)) - usage = resp.get("data").get("usage") + resp: Completion = self.llm.chat.completions.create(**self._const_kwargs(messages)) + usage = resp.usage.model_dump() self._update_costs(usage) - return resp + return resp.model_dump() async def _achat_completion(self, messages: list[dict], timeout=3) -> dict: - resp = await self.llm.ainvoke(**self._const_kwargs(messages)) - usage = resp.get("data").get("usage") + resp = await self.llm.acreate(**self._const_kwargs(messages)) + usage = resp.get("usage", {}) self._update_costs(usage) return resp @@ -89,35 +81,18 @@ class ZhiPuAILLM(BaseLLM): return await self._achat_completion(messages, timeout=timeout) async def _achat_completion_stream(self, messages: list[dict], timeout=3) -> str: - response = await self.llm.asse_invoke(**self._const_kwargs(messages)) + response = await self.llm.acreate_stream(**self._const_kwargs(messages, stream=True)) collected_content = [] usage = {} - async for event in response.async_events(): - if event.event == ZhiPuEvent.ADD.value: - content = event.data + async for chunk in response.stream(): + finish_reason = chunk.get("choices")[0].get("finish_reason") + if finish_reason == "stop": + usage = chunk.get("usage", {}) + else: + content = self.get_choice_delta_text(chunk) collected_content.append(content) log_llm_stream(content) - elif event.event == ZhiPuEvent.ERROR.value or event.event == ZhiPuEvent.INTERRUPTED.value: - content = event.data - logger.error(f"event error: {content}", end="") - elif event.event == ZhiPuEvent.FINISH.value: - """ - event.meta - { - "task_status":"SUCCESS", - "usage":{ - "completion_tokens":351, - "prompt_tokens":595, - "total_tokens":946 - }, - "task_id":"xx", - "request_id":"xxx" - } - """ - meta = json.loads(event.meta) - usage = meta.get("usage") - else: - print(f"zhipuapi else event: {event.data}", end="") + log_llm_stream("\n") self._update_costs(usage) diff --git a/metagpt/utils/file_repository.py b/metagpt/utils/file_repository.py index 0ddca414d..11315e595 100644 --- a/metagpt/utils/file_repository.py +++ b/metagpt/utils/file_repository.py @@ -55,6 +55,7 @@ class FileRepository: """ pathname = self.workdir / filename pathname.parent.mkdir(parents=True, exist_ok=True) + content = content if content else "" # avoid `argument must be str, not None` to make it continue async with aiofiles.open(str(pathname), mode="w") as writer: await writer.write(content) logger.info(f"save to: {str(pathname)}") diff --git a/metagpt/utils/repair_llm_raw_output.py b/metagpt/utils/repair_llm_raw_output.py index a96c3dce0..b71def136 100644 --- a/metagpt/utils/repair_llm_raw_output.py +++ b/metagpt/utils/repair_llm_raw_output.py @@ -120,6 +120,15 @@ def repair_json_format(output: str) -> str: elif output.startswith("{") and output.endswith("]"): output = output[:-1] + "}" + # remove `#` in output json str, usually appeared in `glm-4` + arr = output.split("\n") + new_arr = [] + for line in arr: + idx = line.find("#") + if idx >= 0: + line = line[:idx] + new_arr.append(line) + output = "\n".join(new_arr) return output @@ -168,15 +177,17 @@ def repair_invalid_json(output: str, error: str) -> str: example 1. json.decoder.JSONDecodeError: Expecting ',' delimiter: line 154 column 1 (char 2765) example 2. xxx.JSONDecodeError: Expecting property name enclosed in double quotes: line 14 column 1 (char 266) """ - pattern = r"line ([0-9]+)" + pattern = r"line ([0-9]+) column ([0-9]+)" matches = re.findall(pattern, error, re.DOTALL) if len(matches) > 0: - line_no = int(matches[0]) - 1 + line_no = int(matches[0][0]) - 1 + col_no = int(matches[0][1]) - 1 # due to CustomDecoder can handle `"": ''` or `'': ""`, so convert `"""` -> `"`, `'''` -> `'` output = output.replace('"""', '"').replace("'''", '"') arr = output.split("\n") + rline = arr[line_no] # raw line line = arr[line_no].strip() # different general problems if line.endswith("],"): @@ -187,9 +198,12 @@ def repair_invalid_json(output: str, error: str) -> str: new_line = line.replace("}", "") elif line.endswith("},") and output.endswith("},"): new_line = line[:-1] - elif '",' not in line and "," not in line: + elif (rline[col_no] in ["'", '"']) and (line.startswith('"') or line.startswith("'")) and "," not in line: + # problem, `"""` or `'''` without `,` + new_line = f",{line}" + elif '",' not in line and "," not in line and '"' not in line: new_line = f'{line}",' - elif "," not in line: + elif not line.endswith(","): # problem, miss char `,` at the end. new_line = f"{line}," elif "," in line and len(line) == 1: diff --git a/metagpt/utils/token_counter.py b/metagpt/utils/token_counter.py index a1b74a074..885eb37d7 100644 --- a/metagpt/utils/token_counter.py +++ b/metagpt/utils/token_counter.py @@ -27,7 +27,8 @@ TOKEN_COSTS = { "gpt-4-0613": {"prompt": 0.06, "completion": 0.12}, "gpt-4-1106-preview": {"prompt": 0.01, "completion": 0.03}, "text-embedding-ada-002": {"prompt": 0.0004, "completion": 0.0}, - "chatglm_turbo": {"prompt": 0.0, "completion": 0.00069}, # 32k version, prompt + completion tokens=0.005¥/k-tokens + "glm-3-turbo": {"prompt": 0.0, "completion": 0.0007}, # 128k version, prompt + completion tokens=0.005¥/k-tokens + "glm-4": {"prompt": 0.0, "completion": 0.014}, # 128k version, prompt + completion tokens=0.1¥/k-tokens "gemini-pro": {"prompt": 0.00025, "completion": 0.0005}, } diff --git a/requirements.txt b/requirements.txt index 0a54236f0..93ad653dc 100644 --- a/requirements.txt +++ b/requirements.txt @@ -50,7 +50,7 @@ aioredis~=2.0.1 # Used by metagpt/utils/redis.py websocket-client==1.6.2 aiofiles==23.2.1 gitpython==3.1.40 -zhipuai==1.0.7 +zhipuai==2.0.1 socksio~=1.0.0 gitignore-parser==0.1.9 # connexion[uvicorn]~=3.0.5 # Used by metagpt/tools/openapi_v3_hello.py diff --git a/tests/metagpt/provider/test_zhipuai_api.py b/tests/metagpt/provider/test_zhipuai_api.py index ab240260c..8f06fc717 100644 --- a/tests/metagpt/provider/test_zhipuai_api.py +++ b/tests/metagpt/provider/test_zhipuai_api.py @@ -3,7 +3,6 @@ # @Desc : the unittest of ZhiPuAILLM import pytest -from zhipuai.utils.sse_client import Event from metagpt.config import CONFIG from metagpt.provider.zhipuai_api import ZhiPuAILLM @@ -15,35 +14,16 @@ messages = [{"role": "user", "content": prompt_msg}] resp_content = "I'm chatglm-turbo" default_resp = { - "code": 200, - "data": { - "choices": [{"role": "assistant", "content": resp_content}], - "usage": {"prompt_tokens": 20, "completion_tokens": 20}, - }, + "choices": [{"finish_reason": "stop", "index": 0, "message": {"content": resp_content, "role": "assistant"}}], + "usage": {"completion_tokens": 22, "prompt_tokens": 19, "total_tokens": 41}, } -def mock_zhipuai_invoke(**kwargs) -> dict: - return default_resp - - -async def mock_zhipuai_ainvoke(**kwargs) -> dict: - return default_resp - - -async def mock_zhipuai_asse_invoke(**kwargs): +async def mock_zhipuai_acreate_stream(self, **kwargs): class MockResponse(object): async def _aread(self): class Iterator(object): - events = [ - Event(id="xxx", event="add", data=resp_content, retry=0), - Event( - id="xxx", - event="finish", - data="", - meta='{"usage": {"completion_tokens": 20,"prompt_tokens": 20}}', - ), - ] + events = [{"choices": [{"index": 0, "delta": {"content": resp_content, "role": "assistant"}}]}] async def __aiter__(self): for event in self.events: @@ -52,23 +32,26 @@ async def mock_zhipuai_asse_invoke(**kwargs): async for chunk in Iterator(): yield chunk - async def async_events(self): + async def stream(self): async for chunk in self._aread(): yield chunk return MockResponse() +async def mock_zhipuai_acreate(self, **kwargs) -> dict: + return default_resp + + @pytest.mark.asyncio async def test_zhipuai_acompletion(mocker): - mocker.patch("metagpt.provider.zhipuai.zhipu_model_api.ZhiPuModelAPI.invoke", mock_zhipuai_invoke) - mocker.patch("metagpt.provider.zhipuai.zhipu_model_api.ZhiPuModelAPI.ainvoke", mock_zhipuai_ainvoke) - mocker.patch("metagpt.provider.zhipuai.zhipu_model_api.ZhiPuModelAPI.asse_invoke", mock_zhipuai_asse_invoke) + mocker.patch("metagpt.provider.zhipuai.zhipu_model_api.ZhiPuModelAPI.acreate", mock_zhipuai_acreate) + mocker.patch("metagpt.provider.zhipuai.zhipu_model_api.ZhiPuModelAPI.acreate_stream", mock_zhipuai_acreate_stream) zhipu_gpt = ZhiPuAILLM() resp = await zhipu_gpt.acompletion(messages) - assert resp["data"]["choices"][0]["content"] == resp_content + assert resp["choices"][0]["message"]["content"] == resp_content resp = await zhipu_gpt.aask(prompt_msg, stream=False) assert resp == resp_content diff --git a/tests/metagpt/provider/zhipuai/test_async_sse_client.py b/tests/metagpt/provider/zhipuai/test_async_sse_client.py index 2649f595b..31b2d3d64 100644 --- a/tests/metagpt/provider/zhipuai/test_async_sse_client.py +++ b/tests/metagpt/provider/zhipuai/test_async_sse_client.py @@ -11,16 +11,16 @@ from metagpt.provider.zhipuai.async_sse_client import AsyncSSEClient async def test_async_sse_client(): class Iterator(object): async def __aiter__(self): - yield b"data: test_value" + yield b'data: {"test_key": "test_value"}' async_sse_client = AsyncSSEClient(event_source=Iterator()) - async for event in async_sse_client.async_events(): - assert event.data, "test_value" + async for chunk in async_sse_client.stream(): + assert "test_value" in chunk.values() class InvalidIterator(object): async def __aiter__(self): yield b"invalid: test_value" async_sse_client = AsyncSSEClient(event_source=InvalidIterator()) - async for event in async_sse_client.async_events(): - assert not event + async for chunk in async_sse_client.stream(): + assert not chunk diff --git a/tests/metagpt/provider/zhipuai/test_zhipu_model_api.py b/tests/metagpt/provider/zhipuai/test_zhipu_model_api.py index 1f0a42fa6..15673c51c 100644 --- a/tests/metagpt/provider/zhipuai/test_zhipu_model_api.py +++ b/tests/metagpt/provider/zhipuai/test_zhipu_model_api.py @@ -6,15 +6,13 @@ from typing import Any, Tuple import pytest import zhipuai -from zhipuai.model_api.api import InvokeType -from zhipuai.utils.http_client import headers as zhipuai_default_headers from metagpt.provider.zhipuai.zhipu_model_api import ZhiPuModelAPI api_key = "xxx.xxx" zhipuai.api_key = api_key -default_resp = b'{"result": "test response"}' +default_resp = b'{"choices": [{"finish_reason": "stop", "index": 0, "message": {"content": "test response", "role": "assistant"}}]}' async def mock_requestor_arequest(self, **kwargs) -> Tuple[Any, Any, str]: @@ -23,22 +21,15 @@ async def mock_requestor_arequest(self, **kwargs) -> Tuple[Any, Any, str]: @pytest.mark.asyncio async def test_zhipu_model_api(mocker): - header = ZhiPuModelAPI.get_header() - zhipuai_default_headers.update({"Authorization": api_key}) - assert header == zhipuai_default_headers - - sse_header = ZhiPuModelAPI.get_sse_header() - assert len(sse_header["Authorization"]) == 191 - - url_prefix, url_suffix = ZhiPuModelAPI.split_zhipu_api_url(InvokeType.SYNC, kwargs={"model": "chatglm_turbo"}) + url_prefix, url_suffix = ZhiPuModelAPI(api_key=api_key).split_zhipu_api_url() assert url_prefix == "https://open.bigmodel.cn/api" - assert url_suffix == "/paas/v3/model-api/chatglm_turbo/invoke" + assert url_suffix == "/paas/v4/chat/completions" mocker.patch("metagpt.provider.general_api_requestor.GeneralAPIRequestor.arequest", mock_requestor_arequest) - result = await ZhiPuModelAPI.arequest( - InvokeType.SYNC, stream=False, method="get", headers={}, kwargs={"model": "chatglm_turbo"} + result = await ZhiPuModelAPI(api_key=api_key).arequest( + stream=False, method="get", headers={}, kwargs={"model": "glm-3-turbo"} ) assert result == default_resp - result = await ZhiPuModelAPI.ainvoke() - assert result["result"] == "test response" + result = await ZhiPuModelAPI(api_key=api_key).acreate() + assert result["choices"][0]["message"]["content"] == "test response" diff --git a/tests/metagpt/utils/test_repair_llm_raw_output.py b/tests/metagpt/utils/test_repair_llm_raw_output.py index 1970c6443..1f809a081 100644 --- a/tests/metagpt/utils/test_repair_llm_raw_output.py +++ b/tests/metagpt/utils/test_repair_llm_raw_output.py @@ -128,6 +128,19 @@ def test_repair_json_format(): output = repair_llm_raw_output(output=raw_output, req_keys=[None], repair_type=RepairType.JSON) assert output == target_output + raw_output = """ +{ + "Language": "en_us", # define language + "Programming Language": "Python" +} +""" + target_output = """{ + "Language": "en_us", + "Programming Language": "Python" +}""" + output = repair_llm_raw_output(output=raw_output, req_keys=[None], repair_type=RepairType.JSON) + assert output == target_output + def test_repair_invalid_json(): from metagpt.utils.repair_llm_raw_output import repair_invalid_json @@ -204,6 +217,25 @@ def test_retry_parse_json_text(): output = retry_parse_json_text(output=invalid_json_text) assert output == target_json + invalid_json_text = '''{ + "Data structures and interfaces": """ + class UI: + - game_engine: GameEngine + + __init__(engine: GameEngine) -> None + + display_board() -> None + + display_score() -> None + + prompt_move() -> str + + reset_game() -> None + """ + "Anything UNCLEAR": "no" +}''' + target_json = { + "Data structures and interfaces": "\n class UI:\n - game_engine: GameEngine\n + __init__(engine: GameEngine) -> None\n + display_board() -> None\n + display_score() -> None\n + prompt_move() -> str\n + reset_game() -> None\n ", + "Anything UNCLEAR": "no", + } + output = retry_parse_json_text(output=invalid_json_text) + assert output == target_json + def test_extract_content_from_output(): """ From ff10c9bdda72cce908da673c29a5980105129797 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 17 Jan 2024 18:10:30 +0800 Subject: [PATCH 378/637] change name: _display_markdown -> display_markdown. --- metagpt/actions/execute_code.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 6d9135ec3..5b6cba57d 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -104,7 +104,7 @@ class ExecutePyCode(ExecuteCode, Action): code = Syntax(code, "python", theme="paraiso-dark", line_numbers=True) self.console.print(code) elif language == "markdown": - _display_markdown(code) + display_markdown(code) else: raise ValueError(f"Only support for python, markdown, but got {language}") @@ -269,7 +269,7 @@ def remove_escape_and_color_codes(input_str): return result -def _display_markdown(content: str): +def display_markdown(content: str): # 使用正则表达式逐个匹配代码块 matches = re.finditer(r'```(.+?)```', content, re.DOTALL) start_index = 0 From 20f31fa027b32181cbcddce8fc7b24cdb2bb0a0f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 17 Jan 2024 18:17:52 +0800 Subject: [PATCH 379/637] pre-commit. --- metagpt/provider/openai_api.py | 29 +++++++++++++++-------------- 1 file changed, 15 insertions(+), 14 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 7bdb4bfbe..3edd89835 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -9,8 +9,8 @@ @Modified By: mashenquan, 2023/12/1. Fix bug: Unclosed connection caused by openai 0.x. """ -import re import json +import re from typing import AsyncIterator, Union from openai import APIConnectionError, AsyncOpenAI, AsyncStream @@ -28,7 +28,7 @@ from tenacity import ( from metagpt.config import CONFIG, Config, LLMProviderEnum from metagpt.logs import log_llm_stream, logger from metagpt.provider.base_llm import BaseLLM -from metagpt.provider.constant import GENERAL_FUNCTION_SCHEMA, GENERAL_TOOL_CHOICE +from metagpt.provider.constant import GENERAL_FUNCTION_SCHEMA from metagpt.provider.llm_provider_registry import register_provider from metagpt.schema import Message from metagpt.utils.cost_manager import Costs @@ -38,7 +38,6 @@ from metagpt.utils.token_counter import ( count_string_tokens, get_max_completion_tokens, ) -from metagpt.utils.common import CodeParser def log_and_reraise(retry_state): @@ -166,7 +165,7 @@ class OpenAILLM(BaseLLM): if isinstance(msg, str): processed_messages.append({"role": "user", "content": msg}) elif isinstance(msg, dict): - assert set(msg.keys()) == set(['role', 'content']) + assert set(msg.keys()) == set(["role", "content"]) processed_messages.append(msg) elif isinstance(msg, Message): processed_messages.append(msg.to_dict()) @@ -198,9 +197,9 @@ class OpenAILLM(BaseLLM): def _parse_arguments(self, arguments: str) -> dict: """parse arguments in openai function call""" - if 'langugae' not in arguments and 'code' not in arguments: + if "langugae" not in arguments and "code" not in arguments: logger.warning(f"Not found `code`, `language`, We assume it is pure code:\n {arguments}\n. ") - return {'language': 'python', 'code': arguments} + return {"language": "python", "code": arguments} # 匹配language language_pattern = re.compile(r'[\"\']?language[\"\']?\s*:\s*["\']([^"\']+?)["\']', re.DOTALL) @@ -218,7 +217,7 @@ class OpenAILLM(BaseLLM): if code_value is None: raise ValueError(f"Parse code error for {arguments}") # arguments只有code的情况 - return {'language': language_value, 'code': code_value} + return {"language": language_value, "code": code_value} @handle_exception def get_choice_function_arguments(self, rsp: ChatCompletion) -> dict: @@ -230,20 +229,22 @@ class OpenAILLM(BaseLLM): """ message = rsp.choices[0].message if ( - message.tool_calls is not None and - message.tool_calls[0].function is not None and - message.tool_calls[0].function.arguments is not None + message.tool_calls is not None + and message.tool_calls[0].function is not None + and message.tool_calls[0].function.arguments is not None ): # reponse is code try: return json.loads(message.tool_calls[0].function.arguments, strict=False) except json.decoder.JSONDecodeError as e: - logger.debug(f"Got JSONDecodeError for {message.tool_calls[0].function.arguments},\ - we will use RegExp to parse code, \n {e}") - return {'language': 'python', 'code': self._parse_arguments(message.tool_calls[0].function.arguments)} + logger.debug( + f"Got JSONDecodeError for {message.tool_calls[0].function.arguments},\ + we will use RegExp to parse code, \n {e}" + ) + return {"language": "python", "code": self._parse_arguments(message.tool_calls[0].function.arguments)} elif message.tool_calls is None and message.content is not None: # reponse is message - return {'language': 'markdown', 'code': self.get_choice_text(rsp)} + return {"language": "markdown", "code": self.get_choice_text(rsp)} else: logger.error(f"Failed to parse \n {rsp}\n") raise Exception(f"Failed to parse \n {rsp}\n") From a6d56bd7481f22200fd60bec256b705845a7ae28 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 17 Jan 2024 18:47:59 +0800 Subject: [PATCH 380/637] 1. add mock.py in tests\data\incremental_dev_project 2. add mock for test case of ActionNode 3. add path of Guideline file in const.py 4. update engineer.py --- metagpt/actions/design_api_an.py | 6 +- metagpt/actions/write_code_guideline_an.py | 295 +---------- metagpt/const.py | 2 + metagpt/roles/engineer.py | 21 +- tests/data/incremental_dev_project/mock.py | 466 ++++++++++++++++++ tests/metagpt/actions/test_design_api_an.py | 108 +--- .../actions/test_project_management_an.py | 148 +----- .../actions/test_write_code_guideline_an.py | 309 ++---------- tests/metagpt/actions/test_write_prd_an.py | 97 +--- 9 files changed, 599 insertions(+), 853 deletions(-) create mode 100644 tests/data/incremental_dev_project/mock.py diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index 3890e656a..3e65cee1b 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -39,7 +39,7 @@ FILE_LIST = ActionNode( ) REFINED_FILE_LIST = ActionNode( - key="Refined File List", + key="Refined File list", expected_type=List[str], instruction="Update and expand the original file list including only relative paths. Up to 2 files can be added." "Ensure that the refined file list reflects the evolving structure of the project.", @@ -56,7 +56,7 @@ DATA_STRUCTURES_AND_INTERFACES = ActionNode( ) REFINED_DATA_STRUCTURES_AND_INTERFACES = ActionNode( - key="Refined Data Structures and Interfaces", + key="Refined Data structures and interfaces", expected_type=str, instruction="Update and extend the existing mermaid classDiagram code syntax to incorporate new classes, " "methods (including __init__), and functions with precise type annotations. Delineate additional " @@ -108,7 +108,7 @@ REFINE_NODES = [ ] DESIGN_API_NODE = ActionNode.from_children("DesignAPI", NODES) -REFINED_DESIGN_NODES = ActionNode.from_children("Refined_Design_API", REFINE_NODES) +REFINED_DESIGN_NODES = ActionNode.from_children("RefinedDesignAPI", REFINE_NODES) def main(): diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index e4afb393d..ff59ee7d6 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -5,11 +5,11 @@ @Author : mannaandpoem @File : write_code_guideline_an.py """ -import asyncio from metagpt.actions.action import Action -from metagpt.actions.action_node import ActionNode -from metagpt.logs import logger +from metagpt.actions.action_node import ActionNode, dict_to_markdown +from metagpt.config import CONFIG +from metagpt.const import CODE_GUIDELINE_FILE_REPO, CODE_GUIDELINE_PDF_FILE_REPO GUIDELINES_AND_INCREMENTAL_CHANGE = ActionNode( key="Guidelines and Incremental Change", @@ -123,271 +123,6 @@ CODE_GUIDELINE_CONTEXT = """ {code} """ -CODE_GUIDELINE_CONTEXT_EXAMPLE = """ -## New Requirements -Add subtraction, multiplication and division operations to the calculator. -The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator - -## Design -{ - "Refined Implementation Approach": "To accommodate the new requirements, we will extend the existing Python-based calculator application. We will enhance the Tkinter-based UI to include buttons for subtraction, multiplication, and division, alongside the existing addition functionality. We will also implement input validation to handle edge cases such as division by zero. The architecture will be modular, with separate components for the UI, calculation logic, and error handling to maintain simplicity and facilitate future enhancements such as a history feature.", - "File list": [ - "main.py", - "calculator.py", - "interface.py", - "operations.py" - ], - "Refined Data Structures and Interfaces": "classDiagram\n class CalculatorApp {\n +main() None\n }\n class Calculator {\n -result float\n +add(number1: float, number2: float) float\n +subtract(number1: float, number2: float) float\n +multiply(number1: float, number2: float) float\n +divide(number1: float, number2: float) float\n +clear() None\n }\n class Interface {\n -calculator Calculator\n +start() None\n +display_result(result: float) None\n +get_input() float\n +show_error(message: str) None\n +update_operation(operation: str) None\n }\n class Operations {\n +perform_operation(operation: str, number1: float, number2: float) float\n }\n CalculatorApp --> Interface\n Interface --> Calculator\n Calculator --> Operations", - "Refined Program call flow": "sequenceDiagram\n participant CA as CalculatorApp\n participant I as Interface\n participant C as Calculator\n participant O as Operations\n CA->>I: start()\n I->>I: get_input()\n I->>I: update_operation(operation)\n loop For Each Operation\n I->>C: perform_operation(operation, number1, number2)\n C->>O: perform_operation(operation, number1, number2)\n O-->>C: return result\n C-->>I: return result\n I->>I: display_result(result)\n end\n I->>I: show_error(message)", - "Anything UNCLEAR": "The requirement for a history feature is mentioned but not prioritized. It is unclear whether this should be implemented now or in the future. Additionally, there is no specification on the limit to the size of the numbers or the number of operations that can be performed in sequence. These aspects will need clarification for complete implementation." -} - -## Tasks -{ - "Required Python packages": [ - "tkinter" - ], - "Required Other language third-party packages": [ - "No third-party dependencies required" - ], - "Refined Logic Analysis": [ - [ - "main.py", - "Entry point of the application, creates an instance of the Interface class and starts the application." - ], - [ - "calculator.py", - "Contains the Calculator class with add, subtract, multiply, divide and clear methods for performing arithmetic operations." - ], - [ - "interface.py", - "Contains the Interface class responsible for the GUI, interacts with Calculator for the logic and displays results or errors." - ], - [ - "operations.py", - "Contains the Operations class with perform_operation method that delegates the arithmetic operation based on the operation argument." - ] - ], - "Refined Task list": [ - "operations.py", - "calculator.py", - "interface.py", - "main.py" - ], - "Full API spec": "", - "Refined Shared Knowledge": "`interface.py` will use the Calculator class from `calculator.py` to perform operations and display results. `main.py` will be the starting point that initializes the Interface. `calculator.py` will now also interact with `operations.py` to perform the arithmetic operations.", - "Anything UNCLEAR": "The requirement for a history feature is mentioned but not prioritized. It is unclear whether this should be implemented now or in the future. Additionally, there is no specification on the limit to the size of the numbers or the number of operations that can be performed in sequence. These aspects will need clarification for complete implementation." -} - -## Legacy Code ------ calculator.py -```## calculator.py - -class Calculator: - def __init__(self): - self.result = 0.0 # Default value for the result - - def add(self, number1: float, number2: float) -> float: - ''' - Adds two numbers and returns the result. - - Args: - number1 (float): The first number to add. - number2 (float): The second number to add. - - Returns: - float: The sum of number1 and number2. - ''' - self.result = number1 + number2 - return self.result - - def clear(self) -> None: - ''' - Clears the result to its default value. - ''' - self.result = 0.0 -``` - ----- interface.py -```## interface.py -import tkinter as tk -from calculator import Calculator - -class Interface: - def __init__(self): - self.calculator = Calculator() - self.root = tk.Tk() - self.root.title("Calculator") - self.create_widgets() - - def create_widgets(self): - self.result_var = tk.StringVar() - self.result_display = tk.Entry(self.root, textvariable=self.result_var, state='readonly', justify='right', font=('Arial', 24)) - self.result_display.grid(row=0, column=0, columnspan=4, sticky='nsew') - - self.entry_number1 = tk.Entry(self.root, justify='right', font=('Arial', 18)) - self.entry_number1.grid(row=1, column=0, columnspan=2, sticky='nsew') - - self.entry_number2 = tk.Entry(self.root, justify='right', font=('Arial', 18)) - self.entry_number2.grid(row=1, column=2, columnspan=2, sticky='nsew') - - self.add_button = tk.Button(self.root, text='+', command=self.add, font=('Arial', 18)) - self.add_button.grid(row=2, column=0, sticky='nsew') - - self.clear_button = tk.Button(self.root, text='C', command=self.clear, font=('Arial', 18)) - self.clear_button.grid(row=2, column=1, sticky='nsew') - - self.quit_button = tk.Button(self.root, text='Quit', command=self.root.quit, font=('Arial', 18)) - self.quit_button.grid(row=2, column=2, columnspan=2, sticky='nsew') - - self.root.grid_rowconfigure(1, weight=1) - self.root.grid_columnconfigure(0, weight=1) - - def start(self): - self.root.mainloop() - - def display_result(self, result: float): - self.result_var.set(str(result)) - - def get_input(self): - try: - number1 = float(self.entry_number1.get()) - number2 = float(self.entry_number2.get()) - return number1, number2 - except ValueError: - self.show_error("Invalid input! Please enter valid numbers.") - return None, None - - def add(self): - number1, number2 = self.get_input() - if number1 is not None and number2 is not None: - result = self.calculator.add(number1, number2) - self.display_result(result) - - def clear(self): - self.entry_number1.delete(0, tk.END) - self.entry_number2.delete(0, tk.END) - self.result_var.set("") - - def show_error(self, message: str): - tk.messagebox.showerror("Error", message) - -# This code is meant to be used as a module and not as a standalone script. -# The Interface class will be instantiated and started by the main.py file. -``` - ----- main.py -```## main.py -from interface import Interface - - -class CalculatorApp: - @staticmethod - def main(): - interface = Interface() - interface.start() - - -if __name__ == "__main__": - CalculatorApp.main() -``` -""" - -REFINE_CODE_SCRIPT_EXAMPLE = """ ------ calculator.py -```## calculator.py - -class Calculator: - def __init__(self): - self.result = 0.0 # Default value for the result - - def add(self, number1: float, number2: float) -> float: - ''' - Adds two numbers and returns the result. - - Args: - number1 (float): The first number to add. - number2 (float): The second number to add. - - Returns: - float: The sum of number1 and number2. - ''' - self.result = number1 + number2 - return self.result - - def clear(self) -> None: - ''' - Clears the result to its default value. - ''' - self.result = 0.0 -``` - ----- Now, interface.py to be rewritten -```## interface.py -import tkinter as tk -from calculator import Calculator - -class Interface: - def __init__(self): - self.calculator = Calculator() - self.root = tk.Tk() - self.root.title("Calculator") - self.create_widgets() - - def create_widgets(self): - self.result_var = tk.StringVar() - self.result_display = tk.Entry(self.root, textvariable=self.result_var, state='readonly', justify='right', font=('Arial', 24)) - self.result_display.grid(row=0, column=0, columnspan=4, sticky='nsew') - - self.entry_number1 = tk.Entry(self.root, justify='right', font=('Arial', 18)) - self.entry_number1.grid(row=1, column=0, columnspan=2, sticky='nsew') - - self.entry_number2 = tk.Entry(self.root, justify='right', font=('Arial', 18)) - self.entry_number2.grid(row=1, column=2, columnspan=2, sticky='nsew') - - self.add_button = tk.Button(self.root, text='+', command=self.add, font=('Arial', 18)) - self.add_button.grid(row=2, column=0, sticky='nsew') - - self.clear_button = tk.Button(self.root, text='C', command=self.clear, font=('Arial', 18)) - self.clear_button.grid(row=2, column=1, sticky='nsew') - - self.quit_button = tk.Button(self.root, text='Quit', command=self.root.quit, font=('Arial', 18)) - self.quit_button.grid(row=2, column=2, columnspan=2, sticky='nsew') - - self.root.grid_rowconfigure(1, weight=1) - self.root.grid_columnconfigure(0, weight=1) - - def start(self): - self.root.mainloop() - - def display_result(self, result: float): - self.result_var.set(str(result)) - - def get_input(self): - try: - number1 = float(self.entry_number1.get()) - number2 = float(self.entry_number2.get()) - return number1, number2 - except ValueError: - self.show_error("Invalid input! Please enter valid numbers.") - return None, None - - def add(self): - number1, number2 = self.get_input() - if number1 is not None and number2 is not None: - result = self.calculator.add(number1, number2) - self.display_result(result) - - def clear(self): - self.entry_number1.delete(0, tk.END) - self.entry_number2.delete(0, tk.END) - self.result_var.set("") - - def show_error(self, message: str): - tk.messagebox.showerror("Error", message) -``` -""" - REFINED_CODE_TEMPLATE = """ NOTICE Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and Guidelines and Incremental Change, ensuring the integration of new features. @@ -451,13 +186,21 @@ class WriteCodeGuideline(Action): "meticulously craft comprehensive incremental development guidelines and deliver detailed Incremental Change" return await WRITE_CODE_GUIDELINE_NODE.fill(context=context, llm=self.llm, schema="json") + @staticmethod + async def save(guideline): + await WriteCodeGuideline.save_json(guideline) + await WriteCodeGuideline.save_md(guideline) -async def main(): - write_code_guideline = WriteCodeGuideline() - node = await write_code_guideline.run(CODE_GUIDELINE_CONTEXT_EXAMPLE) - guideline = node.instruct_content.model_dump_json() - logger.info(guideline) + @staticmethod + async def save_json(guideline): + filename = "code_guideline.json" + await CONFIG.git_repo.new_file_repository(CODE_GUIDELINE_FILE_REPO).save( + filename=filename, content=str(guideline) + ) - -if __name__ == "__main__": - asyncio.run(main()) + @staticmethod + async def save_md(guideline): + filename = "code_guideline.md" + await CONFIG.git_repo.new_file_repository(CODE_GUIDELINE_PDF_FILE_REPO).save( + filename=filename, content=dict_to_markdown(guideline) + ) diff --git a/metagpt/const.py b/metagpt/const.py index a57be641b..2ee1267d8 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -92,12 +92,14 @@ DOCS_FILE_REPO = "docs" PRDS_FILE_REPO = "docs/prds" SYSTEM_DESIGN_FILE_REPO = "docs/system_design" TASK_FILE_REPO = "docs/tasks" +CODE_GUIDELINE_FILE_REPO = "docs/code_guideline" COMPETITIVE_ANALYSIS_FILE_REPO = "resources/competitive_analysis" DATA_API_DESIGN_FILE_REPO = "resources/data_api_design" SEQ_FLOW_FILE_REPO = "resources/seq_flow" SYSTEM_DESIGN_PDF_FILE_REPO = "resources/system_design" PRD_PDF_FILE_REPO = "resources/prd" TASK_PDF_FILE_REPO = "resources/api_spec_and_tasks" +CODE_GUIDELINE_PDF_FILE_REPO = "resources/code_guideline" TEST_CODES_FILE_REPO = "tests" TEST_OUTPUTS_FILE_REPO = "test_outputs" CODE_SUMMARIES_FILE_REPO = "docs/code_summaries" diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index a6b51bd00..bb0a2e857 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -26,6 +26,7 @@ from pathlib import Path from typing import Set from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks +from metagpt.actions.action_node import dict_to_markdown from metagpt.actions.fix_bug import FixBug from metagpt.actions.summarize_code import SummarizeCode from metagpt.actions.write_code_guideline_an import ( @@ -34,6 +35,7 @@ from metagpt.actions.write_code_guideline_an import ( ) from metagpt.config import CONFIG from metagpt.const import ( + CODE_GUIDELINE_PDF_FILE_REPO, CODE_SUMMARIES_FILE_REPO, CODE_SUMMARIES_PDF_FILE_REPO, PRDS_FILE_REPO, @@ -101,7 +103,7 @@ class Engineer(Role): m = json.loads(task_msg.content) return m.get("Task list") or m.get("Refined Task list") - async def _act_sp_with_cr(self, review=False, guideline="") -> Set[str]: + async def _act_sp_with_cr(self, review=False, guideline=Document()) -> Set[str]: changed_files = set() src_file_repo = CONFIG.git_repo.new_file_repository(CONFIG.src_workspace) for todo in self.code_todos: @@ -112,16 +114,16 @@ class Engineer(Role): 3. Do we need other codes (currently needed)? TODO: The goal is not to need it. After clear task decomposition, based on the design idea, you should be able to write a single file without needing other codes. If you can't, it means you need a clearer definition. This is the key to writing longer code. """ - coding_context = await todo.run(guideline=guideline) + coding_context = await todo.run(guideline=guideline.content) # Code review if review: action = WriteCodeReview(context=coding_context, llm=self.llm) self._init_action_system_message(action) - coding_context = await action.run(guideline=guideline) + coding_context = await action.run(guideline=guideline.content) dependencies = {coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path} - if guideline: - dependencies.add("code_guideline.json") + if guideline.content: + dependencies.add(guideline.root_relative_path) await src_file_repo.save( coding_context.filename, dependencies=dependencies, @@ -364,12 +366,11 @@ class Engineer(Role): code=old_codes, ) node = await WriteCodeGuideline().run(context=context) - guideline = node.instruct_content.model_dump_json() + guideline = node.instruct_content.model_dump() + await WriteCodeGuideline.save(guideline) + guideline = dict_to_markdown(guideline) - await CONFIG.git_repo.new_file_repository(CONFIG.git_repo.workdir).save( - filename="code_guideline.json", content=guideline - ) - return guideline + return Document(root_path=CODE_GUIDELINE_PDF_FILE_REPO, filename="code_guideline.md", content=guideline) @staticmethod async def get_old_codes() -> str: diff --git a/tests/data/incremental_dev_project/mock.py b/tests/data/incremental_dev_project/mock.py new file mode 100644 index 000000000..a7aa2bbe4 --- /dev/null +++ b/tests/data/incremental_dev_project/mock.py @@ -0,0 +1,466 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/17 +@Author : mannaandpoem +@File : mock.py +""" +NEW_REQUIREMENT_SAMPLE = """ +Adding graphical interface functionality to enhance the user experience in the number-guessing game. The existing number-guessing game currently relies on command-line input for numbers. The goal is to introduce a graphical interface to improve the game's usability and visual appeal +""" + +PRD_SAMPLE = """ +## Language + +en_us + +## Programming Language + +Python + +## Original Requirements + +Make a simple number guessing game + +## Product Goals + +- Ensure a user-friendly interface for the game +- Provide a challenging yet enjoyable game experience +- Design the game to be easily extendable for future features + +## User Stories + +- As a player, I want to guess numbers and receive feedback on whether my guess is too high or too low +- As a player, I want to be able to set the difficulty level by choosing the range of possible numbers +- As a player, I want to see my previous guesses to strategize my next guess +- As a player, I want to know how many attempts it took me to guess the number once I get it right + +## Competitive Analysis + +- Guess The Number Game A: Basic text interface, no difficulty levels +- Number Master B: Has difficulty levels, but cluttered interface +- Quick Guess C: Sleek design, but lacks performance tracking +- NumGuess D: Good performance tracking, but not mobile-friendly +- GuessIt E: Mobile-friendly, but too many ads +- Perfect Guess F: Offers hints, but the hints are not very helpful +- SmartGuesser G: Has a learning mode, but lacks a competitive edge + +## Competitive Quadrant Chart + +quadrantChart + title "User Engagement and Game Complexity" + x-axis "Low Complexity" --> "High Complexity" + y-axis "Low Engagement" --> "High Engagement" + quadrant-1 "Too Simple" + quadrant-2 "Niche Appeal" + quadrant-3 "Complex & Unengaging" + quadrant-4 "Sweet Spot" + "Guess The Number Game A": [0.2, 0.4] + "Number Master B": [0.5, 0.3] + "Quick Guess C": [0.6, 0.7] + "NumGuess D": [0.4, 0.6] + "GuessIt E": [0.7, 0.5] + "Perfect Guess F": [0.6, 0.4] + "SmartGuesser G": [0.8, 0.6] + "Our Target Product": [0.5, 0.8] + +## Requirement Analysis + +The game should be simple yet engaging, allowing players of different skill levels to enjoy it. It should provide immediate feedback and track the player's performance. The game should also be designed with a clean and intuitive interface, and it should be easy to add new features in the future. + +## Requirement Pool + +- ['P0', 'Implement the core game logic to randomly select a number and allow the user to guess it'] +- ['P0', 'Design a user interface that displays the game status and results clearly'] +- ['P1', 'Add difficulty levels by varying the range of possible numbers'] +- ['P1', 'Keep track of and display the number of attempts for each game session'] +- ['P2', "Store and show the history of the player's guesses during a game session"] + +## UI Design draft + +The UI will feature a clean and minimalist design with a number input field, submit button, and messages area to provide feedback. There will be options to select the difficulty level and a display showing the number of attempts and history of past guesses. + +## Anything UNCLEAR""" + +DESIGN_SAMPLE = """ +## Implementation approach + +We will create a Python-based number guessing game with a simple command-line interface. For the user interface, we will use the built-in 'input' and 'print' functions for interaction. The random library will be used for generating random numbers. We will structure the code to be modular and easily extendable, separating the game logic from the user interface. + +## File list + +- main.py +- game.py +- ui.py + +## Data structures and interfaces + + +classDiagram + class Game { + -int secret_number + -int min_range + -int max_range + -list attempts + +__init__(difficulty: str) + +start_game() + +check_guess(guess: int) str + +get_attempts() int + +get_history() list + } + class UI { + +start() + +display_message(message: str) + +get_user_input(prompt: str) str + +show_attempts(attempts: int) + +show_history(history: list) + +select_difficulty() str + } + class Main { + +main() + } + Main --> UI + UI --> Game + + +## Program call flow + + +sequenceDiagram + participant M as Main + participant UI as UI + participant G as Game + M->>UI: start() + UI->>UI: select_difficulty() + UI-->>G: __init__(difficulty) + G->>G: start_game() + loop Game Loop + UI->>UI: get_user_input("Enter your guess:") + UI-->>G: check_guess(guess) + G->>UI: display_message(feedback) + G->>UI: show_attempts(attempts) + G->>UI: show_history(history) + end + G->>UI: display_message("Correct! Game over.") + UI->>M: main() # Game session ends + + +## Anything UNCLEAR + +The requirement analysis suggests the need for a clean and intuitive interface. Since we are using a command-line interface, we need to ensure that the text-based UI is as user-friendly as possible. Further clarification on whether a graphical user interface (GUI) is expected in the future would be helpful for planning the extendability of the game.""" + +TASKS_SAMPLE = """ +## Required Python packages + +- random==2.2.1 + +## Required Other language third-party packages + +- No third-party dependencies required + +## Logic Analysis + +- ['game.py', 'Contains Game class with methods __init__, start_game, check_guess, get_attempts, get_history and uses random library for generating secret_number'] +- ['ui.py', 'Contains UI class with methods start, display_message, get_user_input, show_attempts, show_history, select_difficulty and interacts with Game class'] +- ['main.py', 'Contains Main class with method main that initializes UI class and starts the game loop'] + +## Task list + +- game.py +- ui.py +- main.py + +## Full API spec + + + +## Shared Knowledge + +`game.py` contains the core game logic and is used by `ui.py` to interact with the user. `main.py` serves as the entry point to start the game. + +## Anything UNCLEAR + +The requirement analysis suggests the need for a clean and intuitive interface. Since we are using a command-line interface, we need to ensure that the text-based UI is as user-friendly as possible. Further clarification on whether a graphical user interface (GUI) is expected in the future would be helpful for planning the extendability of the game.""" + +OLD_CODE_SAMPLE = """ +--- game.py +```## game.py + +import random + +class Game: + def __init__(self, difficulty: str = 'medium'): + self.min_range, self.max_range = self._set_difficulty(difficulty) + self.secret_number = random.randint(self.min_range, self.max_range) + self.attempts = [] + + def _set_difficulty(self, difficulty: str): + difficulties = { + 'easy': (1, 10), + 'medium': (1, 100), + 'hard': (1, 1000) + } + return difficulties.get(difficulty, (1, 100)) + + def start_game(self): + self.secret_number = random.randint(self.min_range, self.max_range) + self.attempts = [] + + def check_guess(self, guess: int) -> str: + self.attempts.append(guess) + if guess < self.secret_number: + return "It's higher." + elif guess > self.secret_number: + return "It's lower." + else: + return "Correct! Game over." + + def get_attempts(self) -> int: + return len(self.attempts) + + def get_history(self) -> list: + return self.attempts``` + +--- ui.py +```## ui.py + +from game import Game + +class UI: + def start(self): + difficulty = self.select_difficulty() + game = Game(difficulty) + game.start_game() + self.display_welcome_message(game) + + feedback = "" + while feedback != "Correct! Game over.": + guess = self.get_user_input("Enter your guess: ") + if self.is_valid_guess(guess): + feedback = game.check_guess(int(guess)) + self.display_message(feedback) + self.show_attempts(game.get_attempts()) + self.show_history(game.get_history()) + else: + self.display_message("Please enter a valid number.") + + def display_welcome_message(self, game): + print("Welcome to the Number Guessing Game!") + print(f"Guess the number between {game.min_range} and {game.max_range}.") + + def is_valid_guess(self, guess): + return guess.isdigit() + + def display_message(self, message: str): + print(message) + + def get_user_input(self, prompt: str) -> str: + return input(prompt) + + def show_attempts(self, attempts: int): + print(f"Number of attempts: {attempts}") + + def show_history(self, history: list): + print("Guess history:") + for guess in history: + print(guess) + + def select_difficulty(self) -> str: + while True: + difficulty = input("Select difficulty (easy, medium, hard): ").lower() + if difficulty in ['easy', 'medium', 'hard']: + return difficulty + else: + self.display_message("Invalid difficulty. Please choose 'easy', 'medium', or 'hard'.")``` + +--- main.py +```## main.py + +from ui import UI + +class Main: + def main(self): + user_interface = UI() + user_interface.start() + +if __name__ == "__main__": + main_instance = Main() + main_instance.main()``` +""" + +REFINED_PRD_JSON = { + "Language": "en_us", + "Programming Language": "Python", + "Refined Requirements": "Adding graphical interface functionality to enhance the user experience in the number-guessing game.", + "Project Name": "number_guessing_game", + "Refined Product Goals": [ + "Ensure a user-friendly interface for the game with the new graphical interface", + "Provide a challenging yet enjoyable game experience with visual enhancements", + "Design the game to be easily extendable for future features, including graphical elements", + ], + "Refined User Stories": [ + "As a player, I want to interact with a graphical interface to guess numbers and receive visual feedback on my guesses", + "As a player, I want to easily select the difficulty level through the graphical interface", + "As a player, I want to visually track my previous guesses and the number of attempts in the graphical interface", + "As a player, I want to be congratulated with a visually appealing message when I guess the number correctly", + ], + "Competitive Analysis": [ + "Guess The Number Game A: Basic text interface, no difficulty levels", + "Number Master B: Has difficulty levels, but cluttered interface", + "Quick Guess C: Sleek design, but lacks performance tracking", + "NumGuess D: Good performance tracking, but not mobile-friendly", + "GuessIt E: Mobile-friendly, but too many ads", + "Perfect Guess F: Offers hints, but the hints are not very helpful", + "SmartGuesser G: Has a learning mode, but lacks a competitive edge", + "Graphical Guess H: Graphical interface, but poor user experience due to complex design", + ], + "Competitive Quadrant Chart": 'quadrantChart\n title "User Engagement and Game Complexity with Graphical Interface"\n x-axis "Low Complexity" --> "High Complexity"\n y-axis "Low Engagement" --> "High Engagement"\n quadrant-1 "Too Simple"\n quadrant-2 "Niche Appeal"\n quadrant-3 "Complex & Unengaging"\n quadrant-4 "Sweet Spot"\n "Guess The Number Game A": [0.2, 0.4]\n "Number Master B": [0.5, 0.3]\n "Quick Guess C": [0.6, 0.7]\n "NumGuess D": [0.4, 0.6]\n "GuessIt E": [0.7, 0.5]\n "Perfect Guess F": [0.6, 0.4]\n "SmartGuesser G": [0.8, 0.6]\n "Graphical Guess H": [0.7, 0.3]\n "Our Target Product": [0.5, 0.9]', + "Refined Requirement Analysis": [ + "The game should maintain its simplicity while integrating a graphical interface for enhanced engagement.", + "Immediate visual feedback is crucial for user satisfaction in the graphical interface.", + "The interface must be intuitive, allowing for easy navigation and selection of game options.", + "The graphical design should be clean and not detract from the game's core guessing mechanic.", + ], + "Refined Requirement Pool": [ + ["P0", "Implement a graphical user interface (GUI) to replace the command-line interaction"], + [ + "P0", + "Design a user interface that displays the game status, results, and feedback clearly with graphical elements", + ], + ["P1", "Incorporate interactive elements for selecting difficulty levels"], + ["P1", "Visualize the history of the player's guesses and the number of attempts within the game session"], + ["P2", "Create animations for correct or incorrect guesses to enhance user feedback"], + ["P2", "Ensure the GUI is responsive and compatible with various screen sizes"], + ["P2", "Store and show the history of the player's guesses during a game session"], + ], + "UI Design draft": "The UI will feature a modern and minimalist design with a graphical number input field, a submit button with animations, and a dedicated area for visual feedback. It will include interactive elements to select the difficulty level and a visual display for the number of attempts and history of past guesses.", + "Anything UNCLEAR": "", +} + +REFINED_DESIGN_JSON = { + "Refined Implementation Approach": "To accommodate the new graphical user interface (GUI) requirements, we will leverage the Tkinter library, which is included with Python and supports the creation of a user-friendly GUI. The game logic will remain in Python, with Tkinter handling the rendering of the interface. We will ensure that the GUI is responsive and provides immediate visual feedback. The main game loop will be event-driven, responding to user inputs such as button clicks and difficulty selection.", + "Refined File list": ["main.py", "game.py", "ui.py", "gui.py"], + "Refined Data structures and interfaces": "\nclassDiagram\n class Game {\n -int secret_number\n -int min_range\n -int max_range\n -list attempts\n +__init__(difficulty: str)\n +start_game()\n +check_guess(guess: int) str\n +get_attempts() int\n +get_history() list\n }\n class UI {\n +start()\n +display_message(message: str)\n +get_user_input(prompt: str) str\n +show_attempts(attempts: int)\n +show_history(history: list)\n +select_difficulty() str\n }\n class GUI {\n +__init__()\n +setup_window()\n +bind_events()\n +update_feedback(message: str)\n +update_attempts(attempts: int)\n +update_history(history: list)\n +show_difficulty_selector()\n +animate_guess_result(correct: bool)\n }\n class Main {\n +main()\n }\n Main --> UI\n UI --> Game\n UI --> GUI\n GUI --> Game\n", + "Refined Program call flow": '\nsequenceDiagram\n participant M as Main\n participant UI as UI\n participant G as Game\n participant GU as GUI\n M->>UI: start()\n UI->>GU: setup_window()\n GU->>GU: bind_events()\n GU->>UI: select_difficulty()\n UI-->>G: __init__(difficulty)\n G->>G: start_game()\n loop Game Loop\n GU->>GU: show_difficulty_selector()\n GU->>UI: get_user_input("Enter your guess:")\n UI-->>G: check_guess(guess)\n G->>GU: update_feedback(feedback)\n G->>GU: update_attempts(attempts)\n G->>GU: update_history(history)\n GU->>GU: animate_guess_result(correct)\n end\n G->>GU: update_feedback("Correct! Game over.")\n GU->>M: main() # Game session ends\n', + "Anything UNCLEAR": "", +} + +REFINED_TASKS_JSON = { + "Required Python packages": ["random==2.2.1", "Tkinter==8.6"], + "Required Other language third-party packages": ["No third-party dependencies required"], + "Refined Logic Analysis": [ + [ + "game.py", + "Contains Game class with methods __init__, start_game, check_guess, get_attempts, get_history and uses random library for generating secret_number", + ], + [ + "ui.py", + "Contains UI class with methods start, display_message, get_user_input, show_attempts, show_history, select_difficulty and interacts with Game class", + ], + [ + "gui.py", + "Contains GUI class with methods __init__, setup_window, bind_events, update_feedback, update_attempts, update_history, show_difficulty_selector, animate_guess_result and interacts with Game class for GUI rendering", + ], + [ + "main.py", + "Contains Main class with method main that initializes UI class and starts the event-driven game loop", + ], + ], + "Refined Task list": ["game.py", "ui.py", "gui.py", "main.py"], + "Full API spec": "", + "Refined Shared Knowledge": "`game.py` contains the core game logic and is used by `ui.py` to interact with the user. `main.py` serves as the entry point to start the game. `gui.py` is introduced to handle the graphical user interface using Tkinter, which will interact with both `game.py` and `ui.py` for a responsive and user-friendly experience.", + "Anything UNCLEAR": "", +} + +GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE = { + "Guidelines and Incremental Change": '\n1. Guideline for gui.py: Develop the GUI using Tkinter to replace the command-line interface. Start by setting up the main window and event handling. Then, add widgets for displaying the game status, results, and feedback. Implement interactive elements for difficulty selection and visualize the guess history. Finally, create animations for guess feedback and ensure responsiveness across different screen sizes.\n```python\nclass GUI:\n- pass\n+ def __init__(self):\n+ self.setup_window()\n+\n+ def setup_window(self):\n+ # Initialize the main window using Tkinter\n+ pass\n+\n+ def bind_events(self):\n+ # Bind button clicks and other events\n+ pass\n+\n+ def update_feedback(self, message: str):\n+ # Update the feedback label with the given message\n+ pass\n+\n+ def update_attempts(self, attempts: int):\n+ # Update the attempts label with the number of attempts\n+ pass\n+\n+ def update_history(self, history: list):\n+ # Update the history view with the list of past guesses\n+ pass\n+\n+ def show_difficulty_selector(self):\n+ # Show buttons or a dropdown for difficulty selection\n+ pass\n+\n+ def animate_guess_result(self, correct: bool):\n+ # Trigger an animation for correct or incorrect guesses\n+ pass\n```\n\n2. Guideline for main.py: Modify the main.py to initialize the GUI and start the event-driven game loop. Ensure that the GUI is the primary interface for user interaction.\n```python\nclass Main:\n def main(self):\n- user_interface = UI()\n- user_interface.start()\n+ graphical_user_interface = GUI()\n+ graphical_user_interface.setup_window()\n+ graphical_user_interface.bind_events()\n+ # Start the Tkinter main loop\n+ pass\n\n if __name__ == "__main__":\n main_instance = Main()\n main_instance.main()\n```\n\n3. Guideline for ui.py: Refactor ui.py to work with the new GUI class. Remove command-line interactions and delegate display and input tasks to the GUI.\n```python\nclass UI:\n- def display_message(self, message: str):\n- print(message)\n+\n+ def display_message(self, message: str):\n+ # This method will now pass the message to the GUI to display\n+ pass\n\n- def get_user_input(self, prompt: str) -> str:\n- return input(prompt)\n+\n+ def get_user_input(self, prompt: str) -> str:\n+ # This method will now trigger the GUI to get user input\n+ pass\n\n- def show_attempts(self, attempts: int):\n- print(f"Number of attempts: {attempts}")\n+\n+ def show_attempts(self, attempts: int):\n+ # This method will now update the GUI with the number of attempts\n+ pass\n\n- def show_history(self, history: list):\n- print("Guess history:")\n- for guess in history:\n- print(guess)\n+\n+ def show_history(self, history: list):\n+ # This method will now update the GUI with the guess history\n+ pass\n```\n\n4. Guideline for game.py: Ensure game.py remains mostly unchanged as it contains the core game logic. However, make minor adjustments if necessary to integrate with the new GUI.\n```python\nclass Game:\n # No changes required for now\n```\n' +} + +REFINED_CODE_INPUT_SAMPLE = """ +-----Now, game.py to be rewritten +```## game.py + +import random + +class Game: + def __init__(self, difficulty: str = 'medium'): + self.min_range, self.max_range = self._set_difficulty(difficulty) + self.secret_number = random.randint(self.min_range, self.max_range) + self.attempts = [] + + def _set_difficulty(self, difficulty: str): + difficulties = { + 'easy': (1, 10), + 'medium': (1, 100), + 'hard': (1, 1000) + } + return difficulties.get(difficulty, (1, 100)) + + def start_game(self): + self.secret_number = random.randint(self.min_range, self.max_range) + self.attempts = [] + + def check_guess(self, guess: int) -> str: + self.attempts.append(guess) + if guess < self.secret_number: + return "It's higher." + elif guess > self.secret_number: + return "It's lower." + else: + return "Correct! Game over." + + def get_attempts(self) -> int: + return len(self.attempts) + + def get_history(self) -> list: + return self.attempts``` +""" + +REFINED_CODE_SAMPLE = """ +## game.py + +import random + +class Game: + def __init__(self, difficulty: str = 'medium'): + # Set the difficulty level with default value 'medium' + self.min_range, self.max_range = self._set_difficulty(difficulty) + # Initialize the secret number based on the difficulty + self.secret_number = random.randint(self.min_range, self.max_range) + # Initialize the list to keep track of attempts + self.attempts = [] + + def _set_difficulty(self, difficulty: str): + # Define the range of numbers for each difficulty level + difficulties = { + 'easy': (1, 10), + 'medium': (1, 100), + 'hard': (1, 1000) + } + # Return the corresponding range for the selected difficulty, default to 'medium' if not found + return difficulties.get(difficulty, (1, 100)) + + def start_game(self): + # Reset the secret number and attempts list for a new game + self.secret_number = random.randint(self.min_range, self.max_range) + self.attempts.clear() + + def check_guess(self, guess: int) -> str: + # Add the guess to the attempts list + self.attempts.append(guess) + # Provide feedback based on the guess + if guess < self.secret_number: + return "It's higher." + elif guess > self.secret_number: + return "It's lower." + else: + return "Correct! Game over." + + def get_attempts(self) -> int: + # Return the number of attempts made + return len(self.attempts) + + def get_history(self) -> list: + # Return the list of attempts made + return self.attempts +""" diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index c729c047c..2aa123224 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -7,89 +7,15 @@ """ import pytest +from metagpt.actions.action_node import ActionNode, dict_to_markdown +from metagpt.actions.design_api import NEW_REQ_TEMPLATE from metagpt.actions.design_api_an import REFINED_DESIGN_NODES from metagpt.llm import LLM - -CONTEXT = """ -### Legacy Content -{ - "Implementation approach": "We will use Python with the Pygame library to develop the core mechanics of the 2048 game. The game will feature a simple and intuitive user interface, score tracking with high score memory, and an undo move feature. We'll ensure the game has visually appealing graphics and animations while maintaining a minimalist design. The undo feature will allow a single move to be undone without affecting the score, to keep the implementation straightforward.", - "File list": [ - "main.py", - "game.py", - "ui.py", - "constants.py" - ], - "Data structures and interfaces": "classDiagram\n class Main {\n +pygame: PygameInstance\n +game: Game\n +run() void\n }\n class Game {\n -grid: list\n -current_score: int\n -high_score: int\n -last_move: list\n +move(direction: str) bool\n +undo() bool\n +check_game_over() bool\n +reset_game() void\n }\n class UI {\n -screen: PygameSurface\n -font: PygameFont\n +draw_grid(grid: list) void\n +display_score(current_score: int, high_score: int) void\n +show_game_over() void\n +show_undo_button() void\n }\n class Constants {\n +GRID_SIZE: int\n +WINDOW_WIDTH: int\n +WINDOW_HEIGHT: int\n +BACKGROUND_COLOR: tuple\n +TILE_COLORS: dict\n }\n Main --> Game\n Main --> UI\n Game --> Constants\n UI --> Constants", - "Program call flow": "sequenceDiagram\n participant M as Main\n participant G as Game\n participant U as UI\n M->>G: create instance\n M->>U: create instance\n loop game loop\n M->>U: draw_grid(G.grid)\n M->>U: display_score(G.current_score, G.high_score)\n M->>U: show_undo_button()\n M->>G: move(direction)\n alt if move is valid\n G-->>M: return true\n else if move is invalid\n G-->>M: return false\n end\n alt if undo is triggered\n M->>G: undo()\n G-->>M: return true\n else no undo\n G-->>M: return false\n end\n alt if game over\n M->>U: show_game_over()\n M->>G: reset_game()\n end\n end", - "Anything UNCLEAR": "The specifics of the scoring system and how the high score is stored and retrieved need to be clarified. Additionally, the exact graphical assets and animations for the game are not specified." -} - -### New Requirements -{ - "Language": "en_us", - "Programming Language": "Python", - "Refined Requirements": "Update the py2048_game to have a larger 8x8 grid and a new winning score target of 4096, while maintaining the core mechanics and user-friendly interface of the original 2048 game.", - "Project Name": "py2048_game", - "Refined Product Goals": [ - "Develop an enhanced version of the 2048 game with a larger grid and higher score target to provide a new challenge to players", - "Ensure the game remains visually appealing and maintains a consistent theme with the added complexity", - "Implement smooth and responsive game controls suitable for an 8x8 grid interface" - ], - "Refined User Stories": [ - "As a player, I want to experience a clear and simple interface on an 8x8 grid so that I can focus on the gameplay", - "As a player, I want to aim for a higher score target of 4096 to challenge my skills further", - "As a player, I want to see my current and high scores to track my progress on the new larger grid", - "As a player, I want the option to undo my last move to improve my strategy on the 8x8 grid", - "As a player, I want the game to perform smoothly despite the increased complexity of the larger grid" - ], - "Competitive Analysis": [ - "2048 Original: Classic gameplay with minimalistic design, but lacks modern features", - "2048 by Gabriele Cirulli: Open-source version with clean UI, but no additional features", - "2048 Hex: Unique hexagon board, providing a different challenge", - "2048 Multiplayer: Allows playing against others, but the interface is cluttered", - "2048 with AI: Includes AI challenge mode, but the AI is often too difficult for casual players", - "2048.io: Combines 2048 gameplay with .io style, though it can be overwhelming for new players", - "2048 Animated: Features animations, but has performance issues on some devices" - ], - "Competitive Quadrant Chart": "quadrantChart\n title \"2048 Game Market Positioning\"\n x-axis \"Basic Features\" --> \"Advanced Features\"\n y-axis \"Low User Engagement\" --> \"High User Engagement\"\n quadrant-1 \"Niche Innovators\"\n quadrant-2 \"Market Leaders\"\n quadrant-3 \"Emerging Contenders\"\n quadrant-4 \"Falling Behind\"\n \"2048 Original\": [0.2, 0.7]\n \"2048 by Gabriele Cirulli\": [0.3, 0.8]\n \"2048 Hex\": [0.5, 0.4]\n \"2048 Multiplayer\": [0.6, 0.6]\n \"2048 with AI\": [0.7, 0.5]\n \"2048.io\": [0.4, 0.3]\n \"2048 Animated\": [0.3, 0.2]\n \"Our Target Product\": [0.9, 0.9]", - "Incremental Requirement Analysis": [ - "Adjust the game logic to accommodate an 8x8 grid while ensuring performance remains optimal", - "Update the UI to fit the larger grid and include visual cues for the new score target", - "Enhance the scoring system to support the new target of 4096", - "Ensure the undo feature is adapted to work with the larger grid and increased game complexity", - "Test the game thoroughly to maintain a smooth and responsive experience on the new 8x8 grid" - ], - "Refined Requirement Pool": [ - [ - "P0", - "Expand the game grid to 8x8 and adjust the core mechanics accordingly" - ], - [ - "P0", - "Increase the score target to 4096 and update the scoring system" - ], - [ - "P1", - "Redesign the user interface to accommodate the larger grid size" - ], - [ - "P1", - "Ensure the undo move feature is compatible with the new grid and score target" - ], - [ - "P2", - "Optimize game performance for the increased complexity of an 8x8 grid" - ], - [ - "P2", - "Maintain a visually appealing and consistent theme with the updated game features" - ] - ], - "UI Design draft": "The UI will be updated to feature an 8x8 grid while maintaining a minimalist design. The main game screen will display the larger game grid, current score, high score, and an undo button. The color scheme and transitions will be adapted to ensure clarity and pleasant aesthetics despite the increased grid size.", - "Anything UNCLEAR": "The specifics of how the undo feature should work with the larger grid size and whether there should be any limitations on its use need to be clarified." -} -""" +from tests.data.incremental_dev_project.mock import ( + DESIGN_SAMPLE, + REFINED_DESIGN_JSON, + REFINED_PRD_JSON, +) @pytest.fixture() @@ -98,10 +24,16 @@ def llm(): @pytest.mark.asyncio -async def test_write_design_an(): - node = await REFINED_DESIGN_NODES.fill(CONTEXT, llm) - assert node.instruct_content - assert "Refined Implementation Approach" in node.instruct_content.model_dump_json() - assert "Refined File List" in node.instruct_content.model_dump_json() - assert "Refined Data Structures and Interfaces" in node.instruct_content.model_dump_json() - assert "Refined Program call flow" in node.instruct_content.model_dump_json() +async def test_write_design_an(mocker): + root = ActionNode.from_children( + "RefinedDesignAPI", [ActionNode(key="", expected_type=str, instruction="", example="")] + ) + root.instruct_content = REFINED_DESIGN_JSON + + mocker.patch("metagpt.actions.design_api_an.REFINED_DESIGN_NODES.fill", return_value=root) + prompt = NEW_REQ_TEMPLATE.format(old_design=DESIGN_SAMPLE, context=dict_to_markdown(REFINED_PRD_JSON)) + node = await REFINED_DESIGN_NODES.fill(prompt, llm) + assert "Refined Implementation Approach" in node.instruct_content + assert "Refined File list" in node.instruct_content + assert "Refined Data structures and interfaces" in node.instruct_content + assert "Refined Program call flow" in node.instruct_content diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index 68d73a3d9..0540ed6e1 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -7,129 +7,15 @@ """ import pytest +from metagpt.actions.action_node import ActionNode, dict_to_markdown +from metagpt.actions.project_management import NEW_REQ_TEMPLATE from metagpt.actions.project_management_an import REFINED_PM_NODES from metagpt.llm import LLM - -CONTEXT = """ -### Legacy Content -{ - "Required Python packages": [ - "pygame==2.0.1" - ], - "Required Other language third-party packages": [ - "No third-party dependencies required" - ], - "Logic Analysis": [ - [ - "constants.py", - "Contains all the constants like GRID_SIZE, WINDOW_WIDTH, WINDOW_HEIGHT, BACKGROUND_COLOR, TILE_COLORS" - ], - [ - "game.py", - "Contains Game class with methods for game logic such as move, undo, check_game_over, and reset_game" - ], - [ - "ui.py", - "Contains UI class responsible for drawing the grid, displaying scores, showing game over, and undo button" - ], - [ - "main.py", - "Contains Main class which initializes the game loop and orchestrates the interactions between Game and UI classes" - ] - ], - "Task list": [ - "constants.py", - "game.py", - "ui.py", - "main.py" - ], - "Full API spec": "", - "Shared Knowledge": "`constants.py` contains constants shared across `game.py` and `ui.py`. The Main class in `main.py` acts as the controller orchestrating the game flow and UI updates.", - "Anything UNCLEAR": "The specifics of the scoring system and how the high score is stored and retrieved need to be clarified. Additionally, the exact graphical assets and animations for the game are not specified." -} - -### New Requirements -{ - "Refined Implementation Approach": "We will refine our implementation approach to accommodate the new 8x8 grid and the increased winning score target of 4096. This will involve optimizing the game's core logic to handle the larger grid size efficiently and updating the scoring system. We will also enhance the UI to ensure it remains user-friendly and visually appealing with the new grid. The undo feature will be adapted to work seamlessly with the increased complexity of the game.", - "File list": [ - "main.py", - "game.py", - "ui.py", - "constants.py" - ], - "Refined Data Structures and Interfaces": "classDiagram - class Main { - +pygame: PygameInstance - +game: Game - +ui: UI - +run() void - } - class Game { - -grid: list - -current_score: int - -high_score: int - -last_move: list - -target_score: int - +__init__(grid_size: int, target_score: int) - +move(direction: str) bool - +undo() bool - +check_game_over() bool - +reset_game() void - +update_score(value: int) void - } - class UI { - -screen: PygameSurface - -font: PygameFont - +__init__(screen_size: tuple, grid_size: int) - +draw_grid(grid: list) void - +display_score(current_score: int, high_score: int) void - +show_game_over() void - +show_undo_button() void - +update_ui_for_larger_grid() void - } - class Constants { - +GRID_SIZE: int = 8 - +TARGET_SCORE: int = 4096 - +WINDOW_WIDTH: int - +WINDOW_HEIGHT: int - +BACKGROUND_COLOR: tuple - +TILE_COLORS: dict - } - Main --> Game - Main --> UI - Game --> Constants - UI --> Constants", - "Refined Program call flow": "sequenceDiagram - participant M as Main - participant G as Game - participant U as UI - M->>G: create instance(grid_size: Constants.GRID_SIZE, target_score: Constants.TARGET_SCORE) - M->>U: create instance(screen_size: (Constants.WINDOW_WIDTH, Constants.WINDOW_HEIGHT), grid_size: Constants.GRID_SIZE) - loop game loop - M->>U: draw_grid(G.grid) - M->>U: display_score(G.current_score, G.high_score) - M->>U: show_undo_button() - M->>G: move(direction) - alt if move is valid - G-->>M: return true - M->>G: update_score(value) - else if move is invalid - G-->>M: return false - end - alt if undo is triggered - M->>G: undo() - G-->>M: return true - else no undo - G-->>M: return false - end - alt if game over - M->>U: show_game_over() - M->>G: reset_game() - end - end", - "Anything UNCLEAR": "It remains unclear if the undo feature should have limitations on its use, such as a maximum number of undos per game or if it should be available without restriction. Further clarification on this aspect would be beneficial." -} -""" +from tests.data.incremental_dev_project.mock import ( + REFINED_DESIGN_JSON, + REFINED_TASKS_JSON, + TASKS_SAMPLE, +) @pytest.fixture() @@ -138,11 +24,17 @@ def llm(): @pytest.mark.asyncio -async def test_project_management_an(llm): - node = await REFINED_PM_NODES.fill(CONTEXT, llm) +async def test_project_management_an(mocker): + root = ActionNode.from_children( + "RefinedProjectManagement", [ActionNode(key="", expected_type=str, instruction="", example="")] + ) + root.instruct_content = REFINED_TASKS_JSON + + mocker.patch("metagpt.actions.project_management_an.REFINED_PM_NODES.fill", return_value=root) + + prompt = NEW_REQ_TEMPLATE.format(old_tasks=TASKS_SAMPLE, context=dict_to_markdown(REFINED_DESIGN_JSON)) + node = await REFINED_PM_NODES.fill(prompt, llm) assert node.instruct_content - assert "Required Python Packages" in node.instruct_content.model_dump_json() - assert "Required Other Language Packages" in node.instruct_content.model_dump_json() - assert "Refined Logic Analysis" in node.instruct_content.model_dump_json() - assert "Refined Task List" in node.instruct_content.model_dump_json() - assert "Refined Shared Knowledge" in node.instruct_content.model_dump_json() + assert "Refined Logic Analysis" in node.instruct_content + assert "Refined Task list" in node.instruct_content + assert "Refined Shared Knowledge" in node.instruct_content diff --git a/tests/metagpt/actions/test_write_code_guideline_an.py b/tests/metagpt/actions/test_write_code_guideline_an.py index f3fba26cc..998605c4b 100644 --- a/tests/metagpt/actions/test_write_code_guideline_an.py +++ b/tests/metagpt/actions/test_write_code_guideline_an.py @@ -7,300 +7,61 @@ """ import pytest +from metagpt.actions.action_node import ActionNode from metagpt.actions.write_code import WriteCode from metagpt.actions.write_code_guideline_an import ( CODE_GUIDELINE_CONTEXT, - REFINE_CODE_SCRIPT_EXAMPLE, REFINED_CODE_TEMPLATE, WriteCodeGuideline, ) - -REQUIREMENT_EXAMPLE = """Add subtraction, multiplication and division operations to the calculator. -The current calculator can only perform basic addition operations, and it is necessary to introduce subtraction, multiplication, division operation into the calculator -""" - -DESIGN_EXAMPLE = """ -{ - "Refined Implementation Approach": "To accommodate the new requirements, we will extend the existing Python-based calculator application. We will enhance the Tkinter-based UI to include buttons for subtraction, multiplication, and division, alongside the existing addition functionality. We will also implement input validation to handle edge cases such as division by zero. The architecture will be modular, with separate components for the UI, calculation logic, and error handling to maintain simplicity and facilitate future enhancements such as a history feature.", - "File list": [ - "main.py", - "calculator.py", - "interface.py", - "operations.py" - ], - "Refined Data Structures and Interfaces": "classDiagram\n class CalculatorApp {\n +main() None\n }\n class Calculator {\n -result float\n +add(number1: float, number2: float) float\n +subtract(number1: float, number2: float) float\n +multiply(number1: float, number2: float) float\n +divide(number1: float, number2: float) float\n +clear() None\n }\n class Interface {\n -calculator Calculator\n +start() None\n +display_result(result: float) None\n +get_input() float\n +show_error(message: str) None\n +update_operation(operation: str) None\n }\n class Operations {\n +perform_operation(operation: str, number1: float, number2: float) float\n }\n CalculatorApp --> Interface\n Interface --> Calculator\n Calculator --> Operations", - "Refined Program call flow": "sequenceDiagram\n participant CA as CalculatorApp\n participant I as Interface\n participant C as Calculator\n participant O as Operations\n CA->>I: start()\n I->>I: get_input()\n I->>I: update_operation(operation)\n loop For Each Operation\n I->>C: perform_operation(operation, number1, number2)\n C->>O: perform_operation(operation, number1, number2)\n O-->>C: return result\n C-->>I: return result\n I->>I: display_result(result)\n end\n I->>I: show_error(message)", - "Anything UNCLEAR": "The requirement for a history feature is mentioned but not prioritized. It is unclear whether this should be implemented now or in the future. Additionally, there is no specification on the limit to the size of the numbers or the number of operations that can be performed in sequence. These aspects will need clarification for complete implementation." -} -""" - -TASKS_EXAMPLE = """ -{ - "Required Python packages": [ - "tkinter" - ], - "Required Other language third-party packages": [ - "No third-party dependencies required" - ], - "Refined Logic Analysis": [ - [ - "main.py", - "Entry point of the application, creates an instance of the Interface class and starts the application." - ], - [ - "calculator.py", - "Contains the Calculator class with add, subtract, multiply, divide and clear methods for performing arithmetic operations." - ], - [ - "interface.py", - "Contains the Interface class responsible for the GUI, interacts with Calculator for the logic and displays results or errors." - ], - [ - "operations.py", - "Contains the Operations class with perform_operation method that delegates the arithmetic operation based on the operation argument." - ] - ], - "Refined Task list": [ - "operations.py", - "calculator.py", - "interface.py", - "main.py" - ], - "Full API spec": "", - "Refined Shared Knowledge": "`interface.py` will use the Calculator class from `calculator.py` to perform operations and display results. `main.py` will be the starting point that initializes the Interface. `calculator.py` will now also interact with `operations.py` to perform the arithmetic operations.", - "Anything UNCLEAR": "The requirement for a history feature is mentioned but not prioritized. It is unclear whether this should be implemented now or in the future. Additionally, there is no specification on the limit to the size of the numbers or the number of operations that can be performed in sequence. These aspects will need clarification for complete implementation." -} -""" - -CODE_GUIDELINE_SCRIPT_EXAMPLE = """ ------ calculator.py -```## calculator.py - -class Calculator: - def __init__(self): - self.result = 0.0 # Default value for the result - - def add(self, number1: float, number2: float) -> float: - ''' - Adds two numbers and returns the result. - - Args: - number1 (float): The first number to add. - number2 (float): The second number to add. - - Returns: - float: The sum of number1 and number2. - ''' - self.result = number1 + number2 - return self.result - - def clear(self) -> None: - ''' - Clears the result to its default value. - ''' - self.result = 0.0 -``` - ----- interface.py -```## interface.py -import tkinter as tk -from calculator import Calculator - -class Interface: - def __init__(self): - self.calculator = Calculator() - self.root = tk.Tk() - self.root.title("Calculator") - self.create_widgets() - - def create_widgets(self): - self.result_var = tk.StringVar() - self.result_display = tk.Entry(self.root, textvariable=self.result_var, state='readonly', justify='right', font=('Arial', 24)) - self.result_display.grid(row=0, column=0, columnspan=4, sticky='nsew') - - self.entry_number1 = tk.Entry(self.root, justify='right', font=('Arial', 18)) - self.entry_number1.grid(row=1, column=0, columnspan=2, sticky='nsew') - - self.entry_number2 = tk.Entry(self.root, justify='right', font=('Arial', 18)) - self.entry_number2.grid(row=1, column=2, columnspan=2, sticky='nsew') - - self.add_button = tk.Button(self.root, text='+', command=self.add, font=('Arial', 18)) - self.add_button.grid(row=2, column=0, sticky='nsew') - - self.clear_button = tk.Button(self.root, text='C', command=self.clear, font=('Arial', 18)) - self.clear_button.grid(row=2, column=1, sticky='nsew') - - self.quit_button = tk.Button(self.root, text='Quit', command=self.root.quit, font=('Arial', 18)) - self.quit_button.grid(row=2, column=2, columnspan=2, sticky='nsew') - - self.root.grid_rowconfigure(1, weight=1) - self.root.grid_columnconfigure(0, weight=1) - - def start(self): - self.root.mainloop() - - def display_result(self, result: float): - self.result_var.set(str(result)) - - def get_input(self): - try: - number1 = float(self.entry_number1.get()) - number2 = float(self.entry_number2.get()) - return number1, number2 - except ValueError: - self.show_error("Invalid input! Please enter valid numbers.") - return None, None - - def add(self): - number1, number2 = self.get_input() - if number1 is not None and number2 is not None: - result = self.calculator.add(number1, number2) - self.display_result(result) - - def clear(self): - self.entry_number1.delete(0, tk.END) - self.entry_number2.delete(0, tk.END) - self.result_var.set("") - - def show_error(self, message: str): - tk.messagebox.showerror("Error", message) - -# This code is meant to be used as a module and not as a standalone script. -# The Interface class will be instantiated and started by the main.py file. -``` - ----- main.py -```## main.py -from interface import Interface - - -class CalculatorApp: - @staticmethod - def main(): - interface = Interface() - interface.start() - - -if __name__ == "__main__": - CalculatorApp.main() -```""" - -GUIDELINES_AND_INCREMENTAL_CHANGE_EXAMPLE = """ -{ - "Guidelines and Incremental Change": " -1. Guideline for calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Additionally, implement robust error handling for the division operation to mitigate potential issues related to division by zero. -```python -class Calculator: - self.result = number1 + number2 - return self.result - -- def sub(self, number1, number2) -> float: -+ def subtract(self, number1: float, number2: float) -> float: -+ ''' -+ Subtracts the second number from the first and returns the result. -+ -+ Args: -+ number1 (float): The number to be subtracted from. -+ number2 (float): The number to subtract. -+ -+ Returns: -+ float: The difference of number1 and number2. -+ ''' -+ self.result = number1 - number2 -+ return self.result -+ - def multiply(self, number1: float, number2: float) -> float: -- pass -+ ''' -+ Multiplies two numbers and returns the result. -+ -+ Args: -+ number1 (float): The first number to multiply. -+ number2 (float): The second number to multiply. -+ -+ Returns: -+ float: The product of number1 and number2. -+ ''' -+ self.result = number1 * number2 -+ return self.result -+ - def divide(self, number1: float, number2: float) -> float: -- pass -+ ''' -+ ValueError: If the second number is zero. -+ ''' -+ if number2 == 0: -+ raise ValueError('Cannot divide by zero') -+ self.result = number1 / number2 -+ return self.result -+ -- def reset_result(self): -+ def clear(self): -+ if self.result != 0.0: -+ print("Result is not zero, clearing...") -+ else: -+ print("Result is already zero, no need to clear.") -+ - self.result = 0.0 -``` - -2. Guideline for main.py: Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Then, ensure seamless integration with the overall application architecture and maintain consistency with coding standards. -```python -def add_numbers(): - result = calculator.add_numbers(num1, num2) - return jsonify({'result': result}), 200 - --# TODO: Implement subtraction, multiplication, and division operations -+@app.route('/subtract_numbers', methods=['POST']) -+def subtract_numbers(): -+ data = request.get_json() -+ num1 = data.get('num1', 0) -+ num2 = data.get('num2', 0) -+ result = calculator.subtract_numbers(num1, num2) -+ return jsonify({'result': result}), 200 -+ -+@app.route('/multiply_numbers', methods=['POST']) -+def multiply_numbers(): -+ data = request.get_json() -+ num1 = data.get('num1', 0) -+ num2 = data.get('num2', 0) -+ try: -+ result = calculator.divide_numbers(num1, num2) -+ except ValueError as e: -+ return jsonify({'error': str(e)}), 400 -+ return jsonify({'result': result}), 200 -+ - if __name__ == '__main__': - app.run() -```" -} -""" +from tests.data.incremental_dev_project.mock import ( + DESIGN_SAMPLE, + GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE, + NEW_REQUIREMENT_SAMPLE, + OLD_CODE_SAMPLE, + REFINED_CODE_INPUT_SAMPLE, + REFINED_CODE_SAMPLE, + REFINED_DESIGN_JSON, + REFINED_PRD_JSON, + REFINED_TASKS_JSON, + TASKS_SAMPLE, +) @pytest.mark.asyncio -async def test_write_code_guideline_an(): +async def test_write_code_guideline_an(mocker): + root = ActionNode.from_children( + "WriteCodeGuideline", [ActionNode(key="", expected_type=str, instruction="", example="")] + ) + root.instruct_content = GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE + mocker.patch("metagpt.actions.write_code_guideline_an.WriteCodeGuideline.run", return_value=root) + write_code_guideline = WriteCodeGuideline() context = CODE_GUIDELINE_CONTEXT.format( - requirement=REQUIREMENT_EXAMPLE, design=DESIGN_EXAMPLE, tasks=TASKS_EXAMPLE, code=CODE_GUIDELINE_SCRIPT_EXAMPLE + user_requirement=NEW_REQUIREMENT_SAMPLE, + product_requirement_pools=REFINED_PRD_JSON.get("Refined Requirement Pool", ""), + design=REFINED_DESIGN_JSON, + tasks=REFINED_TASKS_JSON, + code=OLD_CODE_SAMPLE, ) node = await write_code_guideline.run(context=context) - assert node.instruct_content - assert "Incremental Change" in node.instruct_content.model_dump_json() + assert "Guidelines and Incremental Change" in node.instruct_content @pytest.mark.asyncio -async def test_refine_code(): +async def test_refine_code(mocker): + mocker.patch("metagpt.actions.write_code.WriteCode.write_code", return_value=REFINED_CODE_SAMPLE) prompt = REFINED_CODE_TEMPLATE.format( - requirement=REQUIREMENT_EXAMPLE, - guideline=GUIDELINES_AND_INCREMENTAL_CHANGE_EXAMPLE, - design=DESIGN_EXAMPLE, - tasks=TASKS_EXAMPLE, - code=REFINE_CODE_SCRIPT_EXAMPLE, + user_requirement=NEW_REQUIREMENT_SAMPLE, + guideline=GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE, + design=DESIGN_SAMPLE, + tasks=TASKS_SAMPLE, + code=REFINED_CODE_INPUT_SAMPLE, logs="", feedback="", - filename="interface.py", + filename="game.py", summary_log="", ) code = await WriteCode().write_code(prompt=prompt) assert code - assert "def create_widgets" in code + assert "def" in code diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index 693f4924d..e7f288c68 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -7,73 +7,14 @@ """ import pytest -from metagpt.actions.write_prd_an import REFINE_PRD_NODE +from metagpt.actions.action_node import ActionNode +from metagpt.actions.write_prd_an import REFINE_PRD_NODE, REFINE_PRD_TEMPLATE from metagpt.llm import LLM - -CONTEXT = """ -### New Project Name -py2048_game - -### New Requirements -Changed score target for 2048 game from 2048 to 4096. -Please change the game's score target from 2048 to 4096, and change the interface size from 4*4 to 8*8. - -### Legacy Content -{ - "Language": "en_us", - "Programming Language": "Python", - "Original Requirements": "make a simple 2048 game based on pygame", - "Project Name": "pygame_2048", - "Product Goals": [ - "Develop a user-friendly and intuitive 2048 game", - "Ensure the game is visually appealing and maintains a consistent theme", - "Implement smooth and responsive game controls" - ], - "User Stories": [ - "As a player, I want to experience a clear and simple interface so that I can focus on the gameplay", - "As a player, I want to see my current and high scores to track my progress", - "As a player, I want the option to undo my last move to improve my strategy" - ], - "Competitive Analysis": [ - "2048 Original: Classic gameplay with minimalistic design, but lacks modern features", - "2048 by Gabriele Cirulli: Open-source version with clean UI, but no additional features", - "2048 Hex: Unique hexagon board, providing a different challenge", - "2048 Multiplayer: Allows playing against others, but the interface is cluttered", - "2048 with AI: Includes AI challenge mode, but the AI is often too difficult for casual players", - "2048.io: Combines 2048 gameplay with .io style, though it can be overwhelming for new players", - "2048 Animated: Features animations, but has performance issues on some devices" - ], - "Competitive Quadrant Chart": "quadrantChart\n title \"2048 Game Market Positioning\"\n x-axis \"Basic Features\" --> \"Advanced Features\"\n y-axis \"Low User Engagement\" --> \"High User Engagement\"\n quadrant-1 \"Niche Innovators\"\n quadrant-2 \"Market Leaders\"\n quadrant-3 \"Emerging Contenders\"\n quadrant-4 \"Falling Behind\"\n \"2048 Original\": [0.2, 0.7]\n \"2048 by Gabriele Cirulli\": [0.3, 0.8]\n \"2048 Hex\": [0.5, 0.4]\n \"2048 Multiplayer\": [0.6, 0.6]\n \"2048 with AI\": [0.7, 0.5]\n \"2048.io\": [0.4, 0.3]\n \"2048 Animated\": [0.3, 0.2]\n \"Our Target Product\": [0.8, 0.9]", - "Requirement Analysis": "The game should be simple yet engaging, with a focus on smooth performance and an intuitive user interface. High scores and undo functionality are important to users for a competitive and strategic gameplay experience. Aesthetic appeal and a consistent theme will also contribute to the game's success.", - "Requirement Pool": [ - [ - "P0", - "Develop core 2048 game mechanics using pygame" - ], - [ - "P0", - "Design a clean and intuitive user interface" - ], - [ - "P1", - "Implement score tracking with high score memory" - ], - [ - "P1", - "Add undo move feature for enhanced gameplay strategy" - ], - [ - "P2", - "Create visually appealing graphics and animations" - ] - ], - "UI Design draft": "The UI will feature a minimalist design with a focus on ease of use. The main game screen will display the game grid, current score, high score, and an undo button. The color scheme will be consistent and pleasant to the eye, with smooth transitions for tile movements.", - "Anything UNCLEAR": "The specifics of the undo feature need to be clarified, such as how many moves can be undone and whether it affects the scoring." -} - -### Search Information -- -""" +from tests.data.incremental_dev_project.mock import ( + NEW_REQUIREMENT_SAMPLE, + PRD_SAMPLE, + REFINED_PRD_JSON, +) @pytest.fixture() @@ -82,11 +23,19 @@ def llm(): @pytest.mark.asyncio -async def test_write_prd_an(llm): - node = await REFINE_PRD_NODE.fill(CONTEXT, llm) - assert node.instruct_content - assert "Refined Requirements" in node.instruct_content.model_dump_json() - assert "Refined Product Goals" in node.instruct_content.model_dump_json() - assert "Refined User Stories" in node.instruct_content.model_dump_json() - assert "Refined Requirement Analysis" in node.instruct_content.model_dump_json() - assert "Refined Requirement Pool" in node.instruct_content.model_dump_json() +async def test_write_prd_an(mocker): + root = ActionNode.from_children("RefinePRD", [ActionNode(key="", expected_type=str, instruction="", example="")]) + root.instruct_content = REFINED_PRD_JSON + + mocker.patch("metagpt.actions.write_prd_an.REFINE_PRD_NODE.fill", return_value=root) + prompt = REFINE_PRD_TEMPLATE.format( + requirements=NEW_REQUIREMENT_SAMPLE, + old_prd=PRD_SAMPLE, + project_name="", + ) + node = await REFINE_PRD_NODE.fill(prompt, llm) + assert "Refined Requirements" in node.instruct_content + assert "Refined Product Goals" in node.instruct_content + assert "Refined User Stories" in node.instruct_content + assert "Refined Requirement Analysis" in node.instruct_content + assert "Refined Requirement Pool" in node.instruct_content From d5ac56f8631274fef8c329df7e0fd8d102874742 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Wed, 17 Jan 2024 18:11:42 +0800 Subject: [PATCH 381/637] feat: remove all unnecessary CONTEXT references feat: replace CONTEXT with local context --- metagpt/actions/run_code.py | 4 +- metagpt/config2.py | 5 +- metagpt/learn/skill_loader.py | 6 +-- metagpt/provider/fireworks_api.py | 2 +- metagpt/roles/assistant.py | 7 +-- metagpt/roles/role.py | 2 +- metagpt/utils/git_repository.py | 5 +- tests/conftest.py | 18 ++++---- tests/metagpt/actions/test_debug_error.py | 15 +++--- tests/metagpt/actions/test_design_api.py | 9 ++-- .../metagpt/actions/test_design_api_review.py | 4 +- tests/metagpt/actions/test_fix_bug.py | 4 +- .../actions/test_generate_questions.py | 8 ++-- tests/metagpt/actions/test_invoice_ocr.py | 4 +- .../metagpt/actions/test_prepare_documents.py | 15 +++--- .../metagpt/actions/test_prepare_interview.py | 4 +- .../actions/test_project_management.py | 13 ++---- .../actions/test_rebuild_class_view.py | 12 ++--- .../actions/test_rebuild_sequence_view.py | 20 ++++---- tests/metagpt/actions/test_research.py | 20 ++++---- tests/metagpt/actions/test_run_code.py | 10 ++-- tests/metagpt/actions/test_skill_action.py | 10 ++-- tests/metagpt/actions/test_summarize_code.py | 46 +++++++------------ tests/metagpt/actions/test_talk_action.py | 11 ++--- tests/metagpt/actions/test_write_code.py | 34 ++++++-------- .../metagpt/actions/test_write_code_review.py | 19 ++++---- tests/metagpt/actions/test_write_docstring.py | 4 +- tests/metagpt/actions/test_write_prd.py | 11 ++--- .../metagpt/actions/test_write_prd_review.py | 4 +- tests/metagpt/actions/test_write_review.py | 8 ++-- .../actions/test_write_teaching_plan.py | 6 +-- tests/metagpt/actions/test_write_test.py | 10 ++-- tests/metagpt/actions/test_write_tutorial.py | 8 ++-- tests/metagpt/learn/test_skill_loader.py | 7 ++- tests/metagpt/roles/test_architect.py | 7 ++- tests/metagpt/roles/test_assistant.py | 11 ++--- tests/metagpt/roles/test_engineer.py | 33 +++++-------- .../roles/test_invoice_ocr_assistant.py | 6 ++- tests/metagpt/roles/test_product_manager.py | 44 ++++++++++++++++-- tests/metagpt/roles/test_project_manager.py | 4 +- tests/metagpt/roles/test_qa_engineer.py | 11 ++--- tests/metagpt/roles/test_researcher.py | 8 ++-- tests/metagpt/roles/test_role.py | 4 +- .../metagpt/roles/test_tutorial_assistant.py | 4 +- .../serialize_deserialize/test_action.py | 11 ++--- .../serialize_deserialize/test_architect.py | 10 ++-- .../serialize_deserialize/test_environment.py | 24 +++++----- .../serialize_deserialize/test_memory.py | 4 +- .../test_prepare_interview.py | 6 +-- .../test_product_manager.py | 6 +-- .../test_project_manager.py | 6 +-- .../serialize_deserialize/test_reasearcher.py | 6 +-- .../serialize_deserialize/test_role.py | 12 ++--- .../serialize_deserialize/test_team.py | 20 ++++---- .../test_tutorial_assistant.py | 2 +- .../serialize_deserialize/test_write_code.py | 15 +++--- .../test_write_code_review.py | 9 ++-- .../test_write_design.py | 12 ++--- .../test_write_docstring.py | 6 +-- .../serialize_deserialize/test_write_prd.py | 6 +-- .../test_write_review.py | 10 ++-- .../test_write_tutorial.py | 12 ++--- tests/metagpt/test_context_mixin.py | 8 +++- .../tools/test_metagpt_oas3_api_svc.py | 6 +-- tests/metagpt/tools/test_openapi_v3_hello.py | 6 +-- tests/metagpt/utils/test_mermaid.py | 5 +- 66 files changed, 350 insertions(+), 349 deletions(-) diff --git a/metagpt/actions/run_code.py b/metagpt/actions/run_code.py index 072ee8f22..7b84a79bb 100644 --- a/metagpt/actions/run_code.py +++ b/metagpt/actions/run_code.py @@ -47,7 +47,7 @@ WRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION. You should fill in necessary instruction, status, send to, and finally return all content between the --- segment line. """ -CONTEXT = """ +TEMPLATE_CONTEXT = """ ## Development Code File Name {code_file_name} ## Development Code @@ -130,7 +130,7 @@ class RunCode(Action): logger.info(f"{outs=}") logger.info(f"{errs=}") - context = CONTEXT.format( + context = TEMPLATE_CONTEXT.format( code=self.i_context.code, code_file_name=self.i_context.code_filename, test_code=self.i_context.test_code, diff --git a/metagpt/config2.py b/metagpt/config2.py index 1d58b9d63..92dd98bad 100644 --- a/metagpt/config2.py +++ b/metagpt/config2.py @@ -84,7 +84,10 @@ class Config(CLIParams, YamlModel): @classmethod def from_home(cls, path): """Load config from ~/.metagpt/config.yaml""" - return Config.from_yaml_file(CONFIG_ROOT / path) + pathname = CONFIG_ROOT / path + if not pathname.exists(): + return None + return Config.from_yaml_file(pathname) @classmethod def default(cls): diff --git a/metagpt/learn/skill_loader.py b/metagpt/learn/skill_loader.py index b60fa9093..ddcd7ccba 100644 --- a/metagpt/learn/skill_loader.py +++ b/metagpt/learn/skill_loader.py @@ -13,7 +13,7 @@ import aiofiles import yaml from pydantic import BaseModel, Field -from metagpt.context import CONTEXT +from metagpt.context import CONTEXT, Context class Example(BaseModel): @@ -73,14 +73,14 @@ class SkillsDeclaration(BaseModel): skill_data = yaml.safe_load(data) return SkillsDeclaration(**skill_data) - def get_skill_list(self, entity_name: str = "Assistant") -> Dict: + def get_skill_list(self, entity_name: str = "Assistant", context: Context = CONTEXT) -> Dict: """Return the skill name based on the skill description.""" entity = self.entities.get(entity_name) if not entity: return {} # List of skills that the agent chooses to activate. - agent_skills = CONTEXT.kwargs.agent_skills + agent_skills = context.kwargs.agent_skills if not agent_skills: return {} diff --git a/metagpt/provider/fireworks_api.py b/metagpt/provider/fireworks_api.py index f9ff7e655..d56453a85 100644 --- a/metagpt/provider/fireworks_api.py +++ b/metagpt/provider/fireworks_api.py @@ -84,7 +84,7 @@ class FireworksLLM(OpenAILLM): def _update_costs(self, usage: CompletionUsage): if self.config.calc_usage and usage: try: - # use FireworksCostManager not CONTEXT.cost_manager + # use FireworksCostManager not context.cost_manager self.cost_manager.update_cost(usage.prompt_tokens, usage.completion_tokens, self.model) except Exception as e: logger.error(f"updating costs failed!, exp: {e}") diff --git a/metagpt/roles/assistant.py b/metagpt/roles/assistant.py index 1c5315eee..2e9ec9bf7 100644 --- a/metagpt/roles/assistant.py +++ b/metagpt/roles/assistant.py @@ -48,7 +48,8 @@ class Assistant(Role): def __init__(self, **kwargs): super().__init__(**kwargs) - self.constraints = self.constraints.format(language=kwargs.get("language") or CONTEXT.kwargs.language) + language = kwargs.get("language") or self.context.kwargs.language or CONTEXT.kwargs.language + self.constraints = self.constraints.format(language=language) async def think(self) -> bool: """Everything will be done part by part.""" @@ -56,11 +57,11 @@ class Assistant(Role): if not last_talk: return False if not self.skills: - skill_path = Path(CONTEXT.kwargs.SKILL_PATH) if CONTEXT.kwargs.SKILL_PATH else None + skill_path = Path(self.context.kwargs.SKILL_PATH) if self.context.kwargs.SKILL_PATH else None self.skills = await SkillsDeclaration.load(skill_yaml_file_name=skill_path) prompt = "" - skills = self.skills.get_skill_list() + skills = self.skills.get_skill_list(context=self.context) for desc, name in skills.items(): prompt += f"If the text explicitly want you to {desc}, return `[SKILL]: {name}` brief and clear. For instance: [SKILL]: {name}\n" prompt += 'Otherwise, return `[TALK]: {talk}` brief and clear. For instance: if {talk} is "xxxx" return [TALK]: xxxx\n\n' diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 47a4f45a7..3a790005c 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -261,7 +261,7 @@ class Role(SerializationMixin, ContextMixin, BaseModel): self._reset() for action in actions: if not isinstance(action, Action): - i = action() + i = action(context=self.context) else: if self.is_human and not isinstance(action.llm, HumanProvider): logger.warning( diff --git a/metagpt/utils/git_repository.py b/metagpt/utils/git_repository.py index 61e5f3251..16f675175 100644 --- a/metagpt/utils/git_repository.py +++ b/metagpt/utils/git_repository.py @@ -201,7 +201,10 @@ class GitRepository: new_path = self.workdir.parent / new_dir_name if new_path.exists(): logger.info(f"Delete directory {str(new_path)}") - shutil.rmtree(new_path) + try: + shutil.rmtree(new_path) + except Exception as e: + logger.warning(f"rm {str(new_path)} error: {e}") if new_path.exists(): # Recheck for windows os logger.warning(f"Failed to delete directory {str(new_path)}") return diff --git a/tests/conftest.py b/tests/conftest.py index 42b460357..9552166d2 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -17,10 +17,11 @@ from typing import Callable import pytest from metagpt.const import DEFAULT_WORKSPACE_ROOT, TEST_DATA_PATH -from metagpt.context import CONTEXT +from metagpt.context import Context as MetagptContext from metagpt.llm import LLM from metagpt.logs import logger from metagpt.utils.git_repository import GitRepository +from metagpt.utils.project_repo import ProjectRepo from tests.mock.mock_aiohttp import MockAioResponse from tests.mock.mock_curl_cffi import MockCurlCffiResponse from tests.mock.mock_httplib2 import MockHttplib2Response @@ -141,19 +142,20 @@ def loguru_caplog(caplog): yield caplog -# init & dispose git repo -@pytest.fixture(scope="function", autouse=True) -def setup_and_teardown_git_repo(request): - CONTEXT.git_repo = GitRepository(local_path=DEFAULT_WORKSPACE_ROOT / f"unittest/{uuid.uuid4().hex}") - CONTEXT.config.git_reinit = True +@pytest.fixture(scope="function") +def context(request): + ctx = MetagptContext() + ctx.git_repo = GitRepository(local_path=DEFAULT_WORKSPACE_ROOT / f"unittest/{uuid.uuid4().hex}") + ctx.repo = ProjectRepo(ctx.git_repo) # Destroy git repo at the end of the test session. def fin(): - if CONTEXT.git_repo: - CONTEXT.git_repo.delete_repository() + if ctx.git_repo: + ctx.git_repo.delete_repository() # Register the function for destroying the environment. request.addfinalizer(fin) + return ctx @pytest.fixture(scope="session", autouse=True) diff --git a/tests/metagpt/actions/test_debug_error.py b/tests/metagpt/actions/test_debug_error.py index e093eb83f..c88818bbd 100644 --- a/tests/metagpt/actions/test_debug_error.py +++ b/tests/metagpt/actions/test_debug_error.py @@ -11,9 +11,7 @@ import uuid import pytest from metagpt.actions.debug_error import DebugError -from metagpt.context import CONTEXT from metagpt.schema import RunCodeContext, RunCodeResult -from metagpt.utils.project_repo import ProjectRepo CODE_CONTENT = ''' from typing import List @@ -116,9 +114,8 @@ if __name__ == '__main__': @pytest.mark.asyncio -async def test_debug_error(): - CONTEXT.src_workspace = CONTEXT.git_repo.workdir / uuid.uuid4().hex - project_repo = ProjectRepo(CONTEXT.git_repo) +async def test_debug_error(context): + context.src_workspace = context.git_repo.workdir / uuid.uuid4().hex ctx = RunCodeContext( code_filename="player.py", test_filename="test_player.py", @@ -126,8 +123,8 @@ async def test_debug_error(): output_filename="output.log", ) - await project_repo.with_src_path(CONTEXT.src_workspace).srcs.save(filename=ctx.code_filename, content=CODE_CONTENT) - await project_repo.tests.save(filename=ctx.test_filename, content=TEST_CONTENT) + await context.repo.with_src_path(context.src_workspace).srcs.save(filename=ctx.code_filename, content=CODE_CONTENT) + await context.repo.tests.save(filename=ctx.test_filename, content=TEST_CONTENT) output_data = RunCodeResult( stdout=";", stderr="", @@ -141,8 +138,8 @@ async def test_debug_error(): "----------------------------------------------------------------------\n" "Ran 5 tests in 0.007s\n\nFAILED (failures=1)\n;\n", ) - await project_repo.test_outputs.save(filename=ctx.output_filename, content=output_data.model_dump_json()) - debug_error = DebugError(i_context=ctx) + await context.repo.test_outputs.save(filename=ctx.output_filename, content=output_data.model_dump_json()) + debug_error = DebugError(i_context=ctx, context=context) rsp = await debug_error.run() diff --git a/tests/metagpt/actions/test_design_api.py b/tests/metagpt/actions/test_design_api.py index fc231e578..7d3efa7ff 100644 --- a/tests/metagpt/actions/test_design_api.py +++ b/tests/metagpt/actions/test_design_api.py @@ -9,20 +9,17 @@ import pytest from metagpt.actions.design_api import WriteDesign -from metagpt.context import CONTEXT from metagpt.logs import logger from metagpt.schema import Message -from metagpt.utils.project_repo import ProjectRepo @pytest.mark.asyncio -async def test_design_api(): +async def test_design_api(context): inputs = ["我们需要一个音乐播放器,它应该有播放、暂停、上一曲、下一曲等功能。"] # PRD_SAMPLE - project_repo = ProjectRepo(CONTEXT.git_repo) for prd in inputs: - await project_repo.docs.prd.save(filename="new_prd.txt", content=prd) + await context.repo.docs.prd.save(filename="new_prd.txt", content=prd) - design_api = WriteDesign() + design_api = WriteDesign(context=context) result = await design_api.run(Message(content=prd, instruct_content=None)) logger.info(result) diff --git a/tests/metagpt/actions/test_design_api_review.py b/tests/metagpt/actions/test_design_api_review.py index cfc29056f..a648dba3f 100644 --- a/tests/metagpt/actions/test_design_api_review.py +++ b/tests/metagpt/actions/test_design_api_review.py @@ -11,7 +11,7 @@ from metagpt.actions.design_api_review import DesignReview @pytest.mark.asyncio -async def test_design_api_review(): +async def test_design_api_review(context): prd = "我们需要一个音乐播放器,它应该有播放、暂停、上一曲、下一曲等功能。" api_design = """ 数据结构: @@ -26,7 +26,7 @@ API列表: """ _ = "API设计看起来非常合理,满足了PRD中的所有需求。" - design_api_review = DesignReview() + design_api_review = DesignReview(context=context) result = await design_api_review.run(prd, api_design) diff --git a/tests/metagpt/actions/test_fix_bug.py b/tests/metagpt/actions/test_fix_bug.py index b2dc8d0f4..cbd9d0b57 100644 --- a/tests/metagpt/actions/test_fix_bug.py +++ b/tests/metagpt/actions/test_fix_bug.py @@ -12,6 +12,6 @@ from metagpt.actions.fix_bug import FixBug @pytest.mark.asyncio -async def test_fix_bug(): - fix_bug = FixBug() +async def test_fix_bug(context): + fix_bug = FixBug(context=context) assert fix_bug.name == "FixBug" diff --git a/tests/metagpt/actions/test_generate_questions.py b/tests/metagpt/actions/test_generate_questions.py index b7c9d3984..6adb2e617 100644 --- a/tests/metagpt/actions/test_generate_questions.py +++ b/tests/metagpt/actions/test_generate_questions.py @@ -10,7 +10,7 @@ import pytest from metagpt.actions.generate_questions import GenerateQuestions from metagpt.logs import logger -context = """ +msg = """ ## topic 如何做一个生日蛋糕 @@ -20,9 +20,9 @@ context = """ @pytest.mark.asyncio -async def test_generate_questions(): - action = GenerateQuestions() - rsp = await action.run(context) +async def test_generate_questions(context): + action = GenerateQuestions(context=context) + rsp = await action.run(msg) logger.info(f"{rsp.content=}") assert "Questions" in rsp.content diff --git a/tests/metagpt/actions/test_invoice_ocr.py b/tests/metagpt/actions/test_invoice_ocr.py index b4560f61b..4df0cf4f8 100644 --- a/tests/metagpt/actions/test_invoice_ocr.py +++ b/tests/metagpt/actions/test_invoice_ocr.py @@ -23,9 +23,9 @@ from metagpt.const import TEST_DATA_PATH Path("invoices/invoice-4.zip"), ], ) -async def test_invoice_ocr(invoice_path: Path): +async def test_invoice_ocr(invoice_path: Path, context): invoice_path = TEST_DATA_PATH / invoice_path - resp = await InvoiceOCR().run(file_path=Path(invoice_path)) + resp = await InvoiceOCR(context=context).run(file_path=Path(invoice_path)) assert isinstance(resp, list) diff --git a/tests/metagpt/actions/test_prepare_documents.py b/tests/metagpt/actions/test_prepare_documents.py index a72019c5c..7ad0dee4e 100644 --- a/tests/metagpt/actions/test_prepare_documents.py +++ b/tests/metagpt/actions/test_prepare_documents.py @@ -10,21 +10,18 @@ import pytest from metagpt.actions.prepare_documents import PrepareDocuments from metagpt.const import REQUIREMENT_FILENAME -from metagpt.context import CONTEXT +from metagpt.context import Context from metagpt.schema import Message -from metagpt.utils.project_repo import ProjectRepo @pytest.mark.asyncio async def test_prepare_documents(): msg = Message(content="New user requirements balabala...") + context = Context() - if CONTEXT.git_repo: - CONTEXT.git_repo.delete_repository() - CONTEXT.git_repo = None - - await PrepareDocuments(context=CONTEXT).run(with_messages=[msg]) - assert CONTEXT.git_repo - doc = await ProjectRepo(CONTEXT.git_repo).docs.get(filename=REQUIREMENT_FILENAME) + await PrepareDocuments(context=context).run(with_messages=[msg]) + assert context.git_repo + assert context.repo + doc = await context.repo.docs.get(filename=REQUIREMENT_FILENAME) assert doc assert doc.content == msg.content diff --git a/tests/metagpt/actions/test_prepare_interview.py b/tests/metagpt/actions/test_prepare_interview.py index cd0c850ed..111f24d5f 100644 --- a/tests/metagpt/actions/test_prepare_interview.py +++ b/tests/metagpt/actions/test_prepare_interview.py @@ -12,8 +12,8 @@ from metagpt.logs import logger @pytest.mark.asyncio -async def test_prepare_interview(): - action = PrepareInterview() +async def test_prepare_interview(context): + action = PrepareInterview(context=context) rsp = await action.run("I just graduated and hope to find a job as a Python engineer") logger.info(f"{rsp.content=}") diff --git a/tests/metagpt/actions/test_project_management.py b/tests/metagpt/actions/test_project_management.py index 9fd3b1721..f3bb405ca 100644 --- a/tests/metagpt/actions/test_project_management.py +++ b/tests/metagpt/actions/test_project_management.py @@ -9,21 +9,18 @@ import pytest from metagpt.actions.project_management import WriteTasks -from metagpt.context import CONTEXT from metagpt.logs import logger from metagpt.schema import Message -from metagpt.utils.project_repo import ProjectRepo from tests.metagpt.actions.mock_json import DESIGN, PRD @pytest.mark.asyncio -async def test_design_api(): - project_repo = ProjectRepo(CONTEXT.git_repo) - await project_repo.docs.prd.save("1.txt", content=str(PRD)) - await project_repo.docs.system_design.save("1.txt", content=str(DESIGN)) - logger.info(CONTEXT.git_repo) +async def test_design_api(context): + await context.repo.docs.prd.save("1.txt", content=str(PRD)) + await context.repo.docs.system_design.save("1.txt", content=str(DESIGN)) + logger.info(context.git_repo) - action = WriteTasks() + action = WriteTasks(context=context) result = await action.run(Message(content="", instruct_content=None)) logger.info(result) diff --git a/tests/metagpt/actions/test_rebuild_class_view.py b/tests/metagpt/actions/test_rebuild_class_view.py index 94295fd55..04b7d91fc 100644 --- a/tests/metagpt/actions/test_rebuild_class_view.py +++ b/tests/metagpt/actions/test_rebuild_class_view.py @@ -11,19 +11,19 @@ from pathlib import Path import pytest from metagpt.actions.rebuild_class_view import RebuildClassView -from metagpt.const import GRAPH_REPO_FILE_REPO -from metagpt.context import CONTEXT from metagpt.llm import LLM @pytest.mark.asyncio -async def test_rebuild(): +async def test_rebuild(context): action = RebuildClassView( - name="RedBean", i_context=str(Path(__file__).parent.parent.parent.parent / "metagpt"), llm=LLM() + name="RedBean", + i_context=str(Path(__file__).parent.parent.parent.parent / "metagpt"), + llm=LLM(), + context=context, ) await action.run() - graph_file_repo = CONTEXT.git_repo.new_file_repository(relative_path=GRAPH_REPO_FILE_REPO) - assert graph_file_repo.changed_files + assert context.repo.docs.graph_repo.changed_files @pytest.mark.parametrize( diff --git a/tests/metagpt/actions/test_rebuild_sequence_view.py b/tests/metagpt/actions/test_rebuild_sequence_view.py index 717aee964..49d444f2f 100644 --- a/tests/metagpt/actions/test_rebuild_sequence_view.py +++ b/tests/metagpt/actions/test_rebuild_sequence_view.py @@ -11,28 +11,28 @@ import pytest from metagpt.actions.rebuild_sequence_view import RebuildSequenceView from metagpt.const import GRAPH_REPO_FILE_REPO -from metagpt.context import CONTEXT from metagpt.llm import LLM from metagpt.utils.common import aread from metagpt.utils.git_repository import ChangeType -from metagpt.utils.project_repo import ProjectRepo @pytest.mark.asyncio -async def test_rebuild(): +async def test_rebuild(context): # Mock data = await aread(filename=Path(__file__).parent / "../../data/graph_db/networkx.json") - graph_db_filename = Path(CONTEXT.git_repo.workdir.name).with_suffix(".json") - project_repo = ProjectRepo(CONTEXT.git_repo) - await project_repo.docs.graph_repo.save(filename=str(graph_db_filename), content=data) - CONTEXT.git_repo.add_change({f"{GRAPH_REPO_FILE_REPO}/{graph_db_filename}": ChangeType.UNTRACTED}) - CONTEXT.git_repo.commit("commit1") + graph_db_filename = Path(context.repo.workdir.name).with_suffix(".json") + await context.repo.docs.graph_repo.save(filename=str(graph_db_filename), content=data) + context.git_repo.add_change({f"{GRAPH_REPO_FILE_REPO}/{graph_db_filename}": ChangeType.UNTRACTED}) + context.git_repo.commit("commit1") action = RebuildSequenceView( - name="RedBean", i_context=str(Path(__file__).parent.parent.parent.parent / "metagpt"), llm=LLM() + name="RedBean", + i_context=str(Path(__file__).parent.parent.parent.parent / "metagpt"), + llm=LLM(), + context=context, ) await action.run() - assert project_repo.docs.graph_repo.changed_files + assert context.repo.docs.graph_repo.changed_files @pytest.mark.parametrize( diff --git a/tests/metagpt/actions/test_research.py b/tests/metagpt/actions/test_research.py index 8c5ed0c7c..372a1e876 100644 --- a/tests/metagpt/actions/test_research.py +++ b/tests/metagpt/actions/test_research.py @@ -14,7 +14,7 @@ from metagpt.tools.search_engine import SearchEngine @pytest.mark.asyncio -async def test_collect_links(mocker, search_engine_mocker): +async def test_collect_links(mocker, search_engine_mocker, context): async def mock_llm_ask(self, prompt: str, system_msgs): if "Please provide up to 2 necessary keywords" in prompt: return '["metagpt", "llm"]' @@ -28,7 +28,7 @@ async def test_collect_links(mocker, search_engine_mocker): return "[1,2]" mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", mock_llm_ask) - resp = await research.CollectLinks(search_engine=SearchEngine(SearchEngineType.DUCK_DUCK_GO)).run( + resp = await research.CollectLinks(search_engine=SearchEngine(SearchEngineType.DUCK_DUCK_GO), context=context).run( "The application of MetaGPT" ) for i in ["MetaGPT use cases", "The roadmap of MetaGPT", "The function of MetaGPT", "What llm MetaGPT support"]: @@ -36,7 +36,7 @@ async def test_collect_links(mocker, search_engine_mocker): @pytest.mark.asyncio -async def test_collect_links_with_rank_func(mocker, search_engine_mocker): +async def test_collect_links_with_rank_func(mocker, search_engine_mocker, context): rank_before = [] rank_after = [] url_per_query = 4 @@ -50,7 +50,7 @@ async def test_collect_links_with_rank_func(mocker, search_engine_mocker): mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", mock_collect_links_llm_ask) resp = await research.CollectLinks( - search_engine=SearchEngine(SearchEngineType.DUCK_DUCK_GO), rank_func=rank_func + search_engine=SearchEngine(SearchEngineType.DUCK_DUCK_GO), rank_func=rank_func, context=context ).run("The application of MetaGPT") for x, y, z in zip(rank_before, rank_after, resp.values()): assert x[::-1] == y @@ -58,7 +58,7 @@ async def test_collect_links_with_rank_func(mocker, search_engine_mocker): @pytest.mark.asyncio -async def test_web_browse_and_summarize(mocker): +async def test_web_browse_and_summarize(mocker, context): async def mock_llm_ask(*args, **kwargs): return "metagpt" @@ -66,20 +66,20 @@ async def test_web_browse_and_summarize(mocker): url = "https://github.com/geekan/MetaGPT" url2 = "https://github.com/trending" query = "What's new in metagpt" - resp = await research.WebBrowseAndSummarize().run(url, query=query) + resp = await research.WebBrowseAndSummarize(context=context).run(url, query=query) assert len(resp) == 1 assert url in resp assert resp[url] == "metagpt" - resp = await research.WebBrowseAndSummarize().run(url, url2, query=query) + resp = await research.WebBrowseAndSummarize(context=context).run(url, url2, query=query) assert len(resp) == 2 async def mock_llm_ask(*args, **kwargs): return "Not relevant." mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", mock_llm_ask) - resp = await research.WebBrowseAndSummarize().run(url, query=query) + resp = await research.WebBrowseAndSummarize(context=context).run(url, query=query) assert len(resp) == 1 assert url in resp @@ -87,7 +87,7 @@ async def test_web_browse_and_summarize(mocker): @pytest.mark.asyncio -async def test_conduct_research(mocker): +async def test_conduct_research(mocker, context): data = None async def mock_llm_ask(*args, **kwargs): @@ -101,7 +101,7 @@ async def test_conduct_research(mocker): "outputs user stories / competitive analysis / requirements / data structures / APIs / documents, etc." ) - resp = await research.ConductResearch().run("The application of MetaGPT", content) + resp = await research.ConductResearch(context=context).run("The application of MetaGPT", content) assert resp == data diff --git a/tests/metagpt/actions/test_run_code.py b/tests/metagpt/actions/test_run_code.py index 76397734d..afd308da7 100644 --- a/tests/metagpt/actions/test_run_code.py +++ b/tests/metagpt/actions/test_run_code.py @@ -24,19 +24,19 @@ async def test_run_text(): @pytest.mark.asyncio -async def test_run_script(): +async def test_run_script(context): # Successful command - out, err = await RunCode().run_script(".", command=["echo", "Hello World"]) + out, err = await RunCode(context=context).run_script(".", command=["echo", "Hello World"]) assert out.strip() == "Hello World" assert err == "" # Unsuccessful command - out, err = await RunCode().run_script(".", command=["python", "-c", "print(1/0)"]) + out, err = await RunCode(context=context).run_script(".", command=["python", "-c", "print(1/0)"]) assert "ZeroDivisionError" in err @pytest.mark.asyncio -async def test_run(): +async def test_run(context): inputs = [ (RunCodeContext(mode="text", code_filename="a.txt", code="print('Hello, World')"), "PASS"), ( @@ -61,5 +61,5 @@ async def test_run(): ), ] for ctx, result in inputs: - rsp = await RunCode(i_context=ctx).run() + rsp = await RunCode(i_context=ctx, context=context).run() assert result in rsp.summary diff --git a/tests/metagpt/actions/test_skill_action.py b/tests/metagpt/actions/test_skill_action.py index 69cd8129d..2ebe79b30 100644 --- a/tests/metagpt/actions/test_skill_action.py +++ b/tests/metagpt/actions/test_skill_action.py @@ -47,18 +47,18 @@ class TestSkillAction: assert args.get("size_type") == "512x512" @pytest.mark.asyncio - async def test_parser_action(self, mocker): + async def test_parser_action(self, mocker, context): # mock mocker.patch("metagpt.learn.text_to_image", return_value="https://mock.com/xxx") - parser_action = ArgumentsParingAction(skill=self.skill, ask="Draw an apple") + parser_action = ArgumentsParingAction(skill=self.skill, ask="Draw an apple", context=context) rsp = await parser_action.run() assert rsp assert parser_action.args assert parser_action.args.get("text") == "Draw an apple" assert parser_action.args.get("size_type") == "512x512" - action = SkillAction(skill=self.skill, args=parser_action.args) + action = SkillAction(skill=self.skill, args=parser_action.args, context=context) rsp = await action.run() assert rsp assert "image/png;base64," in rsp.content or "http" in rsp.content @@ -81,8 +81,8 @@ class TestSkillAction: await SkillAction.find_and_call_function("dummy_call", {"a": 1}) @pytest.mark.asyncio - async def test_skill_action_error(self): - action = SkillAction(skill=self.skill, args={}) + async def test_skill_action_error(self, context): + action = SkillAction(skill=self.skill, args={}, context=context) rsp = await action.run() assert "Error" in rsp.content diff --git a/tests/metagpt/actions/test_summarize_code.py b/tests/metagpt/actions/test_summarize_code.py index e73192406..a404047c1 100644 --- a/tests/metagpt/actions/test_summarize_code.py +++ b/tests/metagpt/actions/test_summarize_code.py @@ -6,18 +6,14 @@ @File : test_summarize_code.py @Modifiled By: mashenquan, 2023-12-6. Unit test for summarize_code.py """ -import shutil import uuid from pathlib import Path import pytest from metagpt.actions.summarize_code import SummarizeCode -from metagpt.context import Context from metagpt.logs import logger from metagpt.schema import CodeSummarizeContext -from metagpt.utils.git_repository import GitRepository -from metagpt.utils.project_repo import ProjectRepo DESIGN_CONTENT = """ {"Implementation approach": "To develop this snake game, we will use the Python language and choose the Pygame library. Pygame is an open-source Python module collection specifically designed for writing video games. It provides functionalities such as displaying images and playing sounds, making it suitable for creating intuitive and responsive user interfaces. We will ensure efficient game logic to prevent any delays during gameplay. The scoring system will be simple, with the snake gaining points for each food it eats. We will use Pygame's event handling system to implement pause and resume functionality, as well as high-score tracking. The difficulty will increase by speeding up the snake's movement. In the initial version, we will focus on single-player mode and consider adding multiplayer mode and customizable skins in future updates. Based on the new requirement, we will also add a moving obstacle that appears randomly. If the snake eats this obstacle, the game will end. If the snake does not eat the obstacle, it will disappear after 5 seconds. For this, we need to add mechanisms for obstacle generation, movement, and disappearance in the game logic.", "Project_name": "snake_game", "File list": ["main.py", "game.py", "snake.py", "food.py", "obstacle.py", "scoreboard.py", "constants.py", "assets/styles.css", "assets/index.html"], "Data structures and interfaces": "```mermaid\n classDiagram\n class Game{\n +int score\n +int speed\n +bool game_over\n +bool paused\n +Snake snake\n +Food food\n +Obstacle obstacle\n +Scoreboard scoreboard\n +start_game() void\n +pause_game() void\n +resume_game() void\n +end_game() void\n +increase_difficulty() void\n +update() void\n +render() void\n Game()\n }\n class Snake{\n +list body_parts\n +str direction\n +bool grow\n +move() void\n +grow() void\n +check_collision() bool\n Snake()\n }\n class Food{\n +tuple position\n +spawn() void\n Food()\n }\n class Obstacle{\n +tuple position\n +int lifetime\n +bool active\n +spawn() void\n +move() void\n +check_collision() bool\n +disappear() void\n Obstacle()\n }\n class Scoreboard{\n +int high_score\n +update_score(int) void\n +reset_score() void\n +load_high_score() void\n +save_high_score() void\n Scoreboard()\n }\n class Constants{\n }\n Game \"1\" -- \"1\" Snake: has\n Game \"1\" -- \"1\" Food: has\n Game \"1\" -- \"1\" Obstacle: has\n Game \"1\" -- \"1\" Scoreboard: has\n ```", "Program call flow": "```sequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n participant O as Obstacle\n participant SB as Scoreboard\n M->>G: start_game()\n loop game loop\n G->>S: move()\n G->>S: check_collision()\n G->>F: spawn()\n G->>O: spawn()\n G->>O: move()\n G->>O: check_collision()\n G->>O: disappear()\n G->>SB: update_score(score)\n G->>G: update()\n G->>G: render()\n alt if paused\n M->>G: pause_game()\n M->>G: resume_game()\n end\n alt if game_over\n G->>M: end_game()\n end\n end\n```", "Anything UNCLEAR": "There is no need for further clarification as the requirements are already clear."} @@ -181,35 +177,27 @@ class Snake: @pytest.mark.asyncio -async def test_summarize_code(): +async def test_summarize_code(context): git_dir = Path(__file__).parent / f"unittest/{uuid.uuid4().hex}" git_dir.mkdir(parents=True, exist_ok=True) - try: - context = Context() - context.git_repo = GitRepository(local_path=git_dir) - context.src_workspace = context.git_repo.workdir / "src" - project_repo = ProjectRepo(context.git_repo) - await project_repo.docs.system_design.save(filename="1.json", content=DESIGN_CONTENT) - await project_repo.docs.task.save(filename="1.json", content=TASK_CONTENT) - await project_repo.with_src_path(context.src_workspace).srcs.save(filename="food.py", content=FOOD_PY) - assert project_repo.srcs.workdir == context.src_workspace - await project_repo.srcs.save(filename="game.py", content=GAME_PY) - await project_repo.srcs.save(filename="main.py", content=MAIN_PY) - await project_repo.srcs.save(filename="snake.py", content=SNAKE_PY) + context.src_workspace = context.git_repo.workdir / "src" + await context.repo.docs.system_design.save(filename="1.json", content=DESIGN_CONTENT) + await context.repo.docs.task.save(filename="1.json", content=TASK_CONTENT) + await context.repo.with_src_path(context.src_workspace).srcs.save(filename="food.py", content=FOOD_PY) + assert context.repo.srcs.workdir == context.src_workspace + await context.repo.srcs.save(filename="game.py", content=GAME_PY) + await context.repo.srcs.save(filename="main.py", content=MAIN_PY) + await context.repo.srcs.save(filename="snake.py", content=SNAKE_PY) - all_files = project_repo.srcs.all_files - summarization_context = CodeSummarizeContext( - design_filename="1.json", task_filename="1.json", codes_filenames=all_files - ) - action = SummarizeCode(context=context, i_context=summarization_context) - rsp = await action.run() - assert rsp - logger.info(rsp) - except Exception as e: - assert not e - finally: - shutil.rmtree(git_dir) + all_files = context.repo.srcs.all_files + summarization_context = CodeSummarizeContext( + design_filename="1.json", task_filename="1.json", codes_filenames=all_files + ) + action = SummarizeCode(context=context, i_context=summarization_context) + rsp = await action.run() + assert rsp + logger.info(rsp) if __name__ == "__main__": diff --git a/tests/metagpt/actions/test_talk_action.py b/tests/metagpt/actions/test_talk_action.py index b722d7c40..206abfbae 100644 --- a/tests/metagpt/actions/test_talk_action.py +++ b/tests/metagpt/actions/test_talk_action.py @@ -9,13 +9,12 @@ import pytest from metagpt.actions.talk_action import TalkAction -from metagpt.context import CONTEXT from metagpt.schema import Message @pytest.mark.asyncio @pytest.mark.parametrize( - ("agent_description", "language", "context", "knowledge", "history_summary"), + ("agent_description", "language", "talk_context", "knowledge", "history_summary"), [ ( "mathematician", @@ -33,12 +32,12 @@ from metagpt.schema import Message ), ], ) -async def test_prompt(agent_description, language, context, knowledge, history_summary): +async def test_prompt(agent_description, language, talk_context, knowledge, history_summary, context): # Prerequisites - CONTEXT.kwargs.agent_description = agent_description - CONTEXT.kwargs.language = language + context.kwargs.agent_description = agent_description + context.kwargs.language = language - action = TalkAction(i_context=context, knowledge=knowledge, history_summary=history_summary) + action = TalkAction(i_context=talk_context, knowledge=knowledge, history_summary=history_summary, context=context) assert "{" not in action.prompt assert "{" not in action.prompt_gpt4 diff --git a/tests/metagpt/actions/test_write_code.py b/tests/metagpt/actions/test_write_code.py index 96d982c69..ee05e0f7d 100644 --- a/tests/metagpt/actions/test_write_code.py +++ b/tests/metagpt/actions/test_write_code.py @@ -12,25 +12,22 @@ from pathlib import Path import pytest from metagpt.actions.write_code import WriteCode -from metagpt.context import CONTEXT -from metagpt.llm import LLM from metagpt.logs import logger from metagpt.schema import CodingContext, Document from metagpt.utils.common import aread -from metagpt.utils.project_repo import ProjectRepo from tests.metagpt.actions.mock_markdown import TASKS_2, WRITE_CODE_PROMPT_SAMPLE @pytest.mark.asyncio -async def test_write_code(): +async def test_write_code(context): # Prerequisites - CONTEXT.src_workspace = CONTEXT.git_repo.workdir / "writecode" + context.src_workspace = context.git_repo.workdir / "writecode" coding_ctx = CodingContext( filename="task_filename.py", design_doc=Document(content="设计一个名为'add'的函数,该函数接受两个整数作为输入,并返回它们的和。") ) doc = Document(content=coding_ctx.model_dump_json()) - write_code = WriteCode(i_context=doc) + write_code = WriteCode(i_context=doc, context=context) code = await write_code.run() logger.info(code.model_dump_json()) @@ -41,45 +38,44 @@ async def test_write_code(): @pytest.mark.asyncio -async def test_write_code_directly(): +async def test_write_code_directly(context): prompt = WRITE_CODE_PROMPT_SAMPLE + "\n" + TASKS_2[0] - llm = LLM() + llm = context.llm_with_cost_manager_from_llm_config(context.config.llm) rsp = await llm.aask(prompt) logger.info(rsp) @pytest.mark.asyncio -async def test_write_code_deps(): +async def test_write_code_deps(context): # Prerequisites - CONTEXT.src_workspace = CONTEXT.git_repo.workdir / "snake1/snake1" + context.src_workspace = context.git_repo.workdir / "snake1/snake1" demo_path = Path(__file__).parent / "../../data/demo_project" - project_repo = ProjectRepo(CONTEXT.git_repo) - await project_repo.test_outputs.save( + await context.repo.test_outputs.save( filename="test_game.py.json", content=await aread(str(demo_path / "test_game.py.json")) ) - await project_repo.docs.code_summary.save( + await context.repo.docs.code_summary.save( filename="20231221155954.json", content=await aread(str(demo_path / "code_summaries.json")), ) - await project_repo.docs.system_design.save( + await context.repo.docs.system_design.save( filename="20231221155954.json", content=await aread(str(demo_path / "system_design.json")), ) - await project_repo.docs.task.save( + await context.repo.docs.task.save( filename="20231221155954.json", content=await aread(str(demo_path / "tasks.json")) ) - await project_repo.with_src_path(CONTEXT.src_workspace).srcs.save( + await context.repo.with_src_path(context.src_workspace).srcs.save( filename="main.py", content='if __name__ == "__main__":\nmain()' ) ccontext = CodingContext( filename="game.py", - design_doc=await project_repo.docs.system_design.get(filename="20231221155954.json"), - task_doc=await project_repo.docs.task.get(filename="20231221155954.json"), + design_doc=await context.repo.docs.system_design.get(filename="20231221155954.json"), + task_doc=await context.repo.docs.task.get(filename="20231221155954.json"), code_doc=Document(filename="game.py", content="", root_path="snake1"), ) coding_doc = Document(root_path="snake1", filename="game.py", content=ccontext.json()) - action = WriteCode(i_context=coding_doc) + action = WriteCode(i_context=coding_doc, context=context) rsp = await action.run() assert rsp assert rsp.code_doc.content diff --git a/tests/metagpt/actions/test_write_code_review.py b/tests/metagpt/actions/test_write_code_review.py index 951929b76..a08dd07bc 100644 --- a/tests/metagpt/actions/test_write_code_review.py +++ b/tests/metagpt/actions/test_write_code_review.py @@ -12,28 +12,25 @@ from metagpt.schema import CodingContext, Document @pytest.mark.asyncio -async def test_write_code_review(capfd): +async def test_write_code_review(capfd, context): + context.src_workspace = context.repo.workdir / "srcs" code = """ def add(a, b): return a + """ - context = CodingContext( + coding_context = CodingContext( filename="math.py", design_doc=Document(content="编写一个从a加b的函数,返回a+b"), code_doc=Document(content=code) ) - context = await WriteCodeReview(i_context=context).run() + await WriteCodeReview(i_context=coding_context, context=context).run() # 我们不能精确地预测生成的代码评审,但我们可以检查返回的是否为字符串 - assert isinstance(context.code_doc.content, str) - assert len(context.code_doc.content) > 0 + assert isinstance(coding_context.code_doc.content, str) + assert len(coding_context.code_doc.content) > 0 captured = capfd.readouterr() print(f"输出内容: {captured.out}") -# @pytest.mark.asyncio -# async def test_write_code_review_directly(): -# code = SEARCH_CODE_SAMPLE -# write_code_review = WriteCodeReview("write_code_review") -# review = await write_code_review.run(code) -# logger.info(review) +if __name__ == "__main__": + pytest.main([__file__, "-s"]) diff --git a/tests/metagpt/actions/test_write_docstring.py b/tests/metagpt/actions/test_write_docstring.py index a0fc46ebd..ebb7e8cb1 100644 --- a/tests/metagpt/actions/test_write_docstring.py +++ b/tests/metagpt/actions/test_write_docstring.py @@ -27,8 +27,8 @@ class Person: ], ids=["google", "numpy", "sphinx"], ) -async def test_write_docstring(style: str, part: str): - ret = await WriteDocstring().run(code, style=style) +async def test_write_docstring(style: str, part: str, context): + ret = await WriteDocstring(context=context).run(code, style=style) assert part in ret diff --git a/tests/metagpt/actions/test_write_prd.py b/tests/metagpt/actions/test_write_prd.py index d854cd8d2..31d20018e 100644 --- a/tests/metagpt/actions/test_write_prd.py +++ b/tests/metagpt/actions/test_write_prd.py @@ -10,21 +10,18 @@ import pytest from metagpt.actions import UserRequirement, WritePRD from metagpt.const import REQUIREMENT_FILENAME -from metagpt.context import CONTEXT from metagpt.logs import logger from metagpt.roles.product_manager import ProductManager from metagpt.roles.role import RoleReactMode from metagpt.schema import Message from metagpt.utils.common import any_to_str -from metagpt.utils.project_repo import ProjectRepo @pytest.mark.asyncio -async def test_write_prd(new_filename): - product_manager = ProductManager() +async def test_write_prd(new_filename, context): + product_manager = ProductManager(context=context) requirements = "开发一个基于大语言模型与私有知识库的搜索引擎,希望可以基于大语言模型进行搜索总结" - project_repo = ProjectRepo(CONTEXT.git_repo) - await project_repo.docs.save(filename=REQUIREMENT_FILENAME, content=requirements) + await context.repo.docs.save(filename=REQUIREMENT_FILENAME, content=requirements) product_manager.rc.react_mode = RoleReactMode.BY_ORDER prd = await product_manager.run(Message(content=requirements, cause_by=UserRequirement)) assert prd.cause_by == any_to_str(WritePRD) @@ -34,7 +31,7 @@ async def test_write_prd(new_filename): # Assert the prd is not None or empty assert prd is not None assert prd.content != "" - assert ProjectRepo(product_manager.context.git_repo).docs.prd.changed_files + assert product_manager.context.repo.docs.prd.changed_files if __name__ == "__main__": diff --git a/tests/metagpt/actions/test_write_prd_review.py b/tests/metagpt/actions/test_write_prd_review.py index 9b3f0a285..8e1601b2e 100644 --- a/tests/metagpt/actions/test_write_prd_review.py +++ b/tests/metagpt/actions/test_write_prd_review.py @@ -11,7 +11,7 @@ from metagpt.actions.write_prd_review import WritePRDReview @pytest.mark.asyncio -async def test_write_prd_review(): +async def test_write_prd_review(context): prd = """ Introduction: This is a new feature for our product. Goals: The goal is to improve user engagement. @@ -23,7 +23,7 @@ async def test_write_prd_review(): Timeline: The feature should be ready for testing in 1.5 months. """ - write_prd_review = WritePRDReview(name="write_prd_review") + write_prd_review = WritePRDReview(name="write_prd_review", context=context) prd_review = await write_prd_review.run(prd) diff --git a/tests/metagpt/actions/test_write_review.py b/tests/metagpt/actions/test_write_review.py index 2d188b720..0274a3532 100644 --- a/tests/metagpt/actions/test_write_review.py +++ b/tests/metagpt/actions/test_write_review.py @@ -9,7 +9,7 @@ import pytest from metagpt.actions.write_review import WriteReview -CONTEXT = """ +TEMPLATE_CONTEXT = """ { "Language": "zh_cn", "Programming Language": "Python", @@ -46,8 +46,8 @@ CONTEXT = """ @pytest.mark.asyncio -async def test_write_review(): - write_review = WriteReview() - review = await write_review.run(CONTEXT) +async def test_write_review(context): + write_review = WriteReview(context=context) + review = await write_review.run(TEMPLATE_CONTEXT) assert review.instruct_content assert review.get("LGTM") in ["LGTM", "LBTM"] diff --git a/tests/metagpt/actions/test_write_teaching_plan.py b/tests/metagpt/actions/test_write_teaching_plan.py index 3d556ab92..bb68d4286 100644 --- a/tests/metagpt/actions/test_write_teaching_plan.py +++ b/tests/metagpt/actions/test_write_teaching_plan.py @@ -13,11 +13,11 @@ from metagpt.actions.write_teaching_plan import WriteTeachingPlanPart @pytest.mark.asyncio @pytest.mark.parametrize( - ("topic", "context"), + ("topic", "content"), [("Title", "Lesson 1: Learn to draw an apple."), ("Teaching Content", "Lesson 1: Learn to draw an apple.")], ) -async def test_write_teaching_plan_part(topic, context): - action = WriteTeachingPlanPart(topic=topic, i_context=context) +async def test_write_teaching_plan_part(topic, content, context): + action = WriteTeachingPlanPart(topic=topic, i_context=content, context=context) rsp = await action.run() assert rsp diff --git a/tests/metagpt/actions/test_write_test.py b/tests/metagpt/actions/test_write_test.py index e09038414..9469dd312 100644 --- a/tests/metagpt/actions/test_write_test.py +++ b/tests/metagpt/actions/test_write_test.py @@ -13,7 +13,7 @@ from metagpt.schema import Document, TestingContext @pytest.mark.asyncio -async def test_write_test(): +async def test_write_test(context): code = """ import random from typing import Tuple @@ -25,8 +25,8 @@ async def test_write_test(): def generate(self, max_y: int, max_x: int): self.position = (random.randint(1, max_y - 1), random.randint(1, max_x - 1)) """ - context = TestingContext(filename="food.py", code_doc=Document(filename="food.py", content=code)) - write_test = WriteTest(i_context=context) + testing_context = TestingContext(filename="food.py", code_doc=Document(filename="food.py", content=code)) + write_test = WriteTest(i_context=testing_context, context=context) context = await write_test.run() logger.info(context.model_dump_json()) @@ -39,12 +39,12 @@ async def test_write_test(): @pytest.mark.asyncio -async def test_write_code_invalid_code(mocker): +async def test_write_code_invalid_code(mocker, context): # Mock the _aask method to return an invalid code string mocker.patch.object(WriteTest, "_aask", return_value="Invalid Code String") # Create an instance of WriteTest - write_test = WriteTest() + write_test = WriteTest(context=context) # Call the write_code method code = await write_test.write_code("Some prompt:") diff --git a/tests/metagpt/actions/test_write_tutorial.py b/tests/metagpt/actions/test_write_tutorial.py index 27a323b44..a83da1a1c 100644 --- a/tests/metagpt/actions/test_write_tutorial.py +++ b/tests/metagpt/actions/test_write_tutorial.py @@ -14,8 +14,8 @@ from metagpt.actions.write_tutorial import WriteContent, WriteDirectory @pytest.mark.asyncio @pytest.mark.parametrize(("language", "topic"), [("English", "Write a tutorial about Python")]) -async def test_write_directory(language: str, topic: str): - ret = await WriteDirectory(language=language).run(topic=topic) +async def test_write_directory(language: str, topic: str, context): + ret = await WriteDirectory(language=language, context=context).run(topic=topic) assert isinstance(ret, dict) assert "title" in ret assert "directory" in ret @@ -29,8 +29,8 @@ async def test_write_directory(language: str, topic: str): ("language", "topic", "directory"), [("English", "Write a tutorial about Python", {"Introduction": ["What is Python?", "Why learn Python?"]})], ) -async def test_write_content(language: str, topic: str, directory: Dict): - ret = await WriteContent(language=language, directory=directory).run(topic=topic) +async def test_write_content(language: str, topic: str, directory: Dict, context): + ret = await WriteContent(language=language, directory=directory, context=context).run(topic=topic) assert isinstance(ret, str) assert list(directory.keys())[0] in ret for value in list(directory.values())[0]: diff --git a/tests/metagpt/learn/test_skill_loader.py b/tests/metagpt/learn/test_skill_loader.py index 45697160b..f1952c275 100644 --- a/tests/metagpt/learn/test_skill_loader.py +++ b/tests/metagpt/learn/test_skill_loader.py @@ -10,13 +10,12 @@ from pathlib import Path import pytest -from metagpt.context import CONTEXT from metagpt.learn.skill_loader import SkillsDeclaration @pytest.mark.asyncio -async def test_suite(): - CONTEXT.kwargs.agent_skills = [ +async def test_suite(context): + context.kwargs.agent_skills = [ {"id": 1, "name": "text_to_speech", "type": "builtin", "config": {}, "enabled": True}, {"id": 2, "name": "text_to_image", "type": "builtin", "config": {}, "enabled": True}, {"id": 3, "name": "ai_call", "type": "builtin", "config": {}, "enabled": True}, @@ -27,7 +26,7 @@ async def test_suite(): ] pathname = Path(__file__).parent / "../../../docs/.well-known/skills.yaml" loader = await SkillsDeclaration.load(skill_yaml_file_name=pathname) - skills = loader.get_skill_list() + skills = loader.get_skill_list(context=context) assert skills assert len(skills) >= 3 for desc, name in skills.items(): diff --git a/tests/metagpt/roles/test_architect.py b/tests/metagpt/roles/test_architect.py index f9d6606ac..b02242ed2 100644 --- a/tests/metagpt/roles/test_architect.py +++ b/tests/metagpt/roles/test_architect.py @@ -13,7 +13,6 @@ import pytest from metagpt.actions import WriteDesign, WritePRD from metagpt.const import PRDS_FILE_REPO -from metagpt.context import CONTEXT from metagpt.logs import logger from metagpt.roles import Architect from metagpt.schema import Message @@ -22,12 +21,12 @@ from tests.metagpt.roles.mock import MockMessages @pytest.mark.asyncio -async def test_architect(): +async def test_architect(context): # Prerequisites filename = uuid.uuid4().hex + ".json" - await awrite(CONTEXT.git_repo.workdir / PRDS_FILE_REPO / filename, data=MockMessages.prd.content) + await awrite(context.repo.workdir / PRDS_FILE_REPO / filename, data=MockMessages.prd.content) - role = Architect() + role = Architect(context=context) rsp = await role.run(with_message=Message(content="", cause_by=WritePRD)) logger.info(rsp) assert len(rsp.content) > 0 diff --git a/tests/metagpt/roles/test_assistant.py b/tests/metagpt/roles/test_assistant.py index b9740a112..bd0efea35 100644 --- a/tests/metagpt/roles/test_assistant.py +++ b/tests/metagpt/roles/test_assistant.py @@ -12,7 +12,6 @@ from pydantic import BaseModel from metagpt.actions.skill_action import SkillAction from metagpt.actions.talk_action import TalkAction -from metagpt.context import CONTEXT from metagpt.memory.brain_memory import BrainMemory from metagpt.roles.assistant import Assistant from metagpt.schema import Message @@ -20,11 +19,11 @@ from metagpt.utils.common import any_to_str @pytest.mark.asyncio -async def test_run(mocker): +async def test_run(mocker, context): # mock mocker.patch("metagpt.learn.text_to_image", return_value="http://mock.com/1.png") - CONTEXT.kwargs.language = "Chinese" + context.kwargs.language = "Chinese" class Input(BaseModel): memory: BrainMemory @@ -80,7 +79,7 @@ async def test_run(mocker): for i in inputs: seed = Input(**i) - role = Assistant(language="Chinese") + role = Assistant(language="Chinese", context=context) role.context.kwargs.language = seed.language role.context.kwargs.agent_description = seed.agent_description role.context.kwargs.agent_skills = agent_skills @@ -115,8 +114,8 @@ async def test_run(mocker): ], ) @pytest.mark.asyncio -async def test_memory(memory): - role = Assistant() +async def test_memory(memory, context): + role = Assistant(context=context) role.context.kwargs.agent_skills = [] role.load_memory(memory) diff --git a/tests/metagpt/roles/test_engineer.py b/tests/metagpt/roles/test_engineer.py index 17b94828c..675e21aa0 100644 --- a/tests/metagpt/roles/test_engineer.py +++ b/tests/metagpt/roles/test_engineer.py @@ -8,44 +8,37 @@ distribution feature for message handling. """ import json -import uuid from pathlib import Path import pytest from metagpt.actions import WriteCode, WriteTasks -from metagpt.const import ( - DEFAULT_WORKSPACE_ROOT, - REQUIREMENT_FILENAME, - SYSTEM_DESIGN_FILE_REPO, - TASK_FILE_REPO, -) -from metagpt.context import CONTEXT, Context +from metagpt.const import REQUIREMENT_FILENAME, SYSTEM_DESIGN_FILE_REPO, TASK_FILE_REPO from metagpt.logs import logger from metagpt.roles.engineer import Engineer from metagpt.schema import CodingContext, Message from metagpt.utils.common import CodeParser, any_to_name, any_to_str, aread, awrite -from metagpt.utils.git_repository import ChangeType, GitRepository +from metagpt.utils.git_repository import ChangeType from metagpt.utils.project_repo import ProjectRepo from tests.metagpt.roles.mock import STRS_FOR_PARSING, TASKS, MockMessages @pytest.mark.asyncio -async def test_engineer(): +async def test_engineer(context): # Prerequisites rqno = "20231221155954.json" - project_repo = ProjectRepo(CONTEXT.git_repo) + project_repo = ProjectRepo(context.git_repo) await project_repo.save(REQUIREMENT_FILENAME, content=MockMessages.req.content) await project_repo.docs.prd.save(rqno, content=MockMessages.prd.content) await project_repo.docs.system_design.save(rqno, content=MockMessages.system_design.content) await project_repo.docs.task.save(rqno, content=MockMessages.json_tasks.content) - engineer = Engineer() + engineer = Engineer(context=context) rsp = await engineer.run(Message(content="", cause_by=WriteTasks)) logger.info(rsp) assert rsp.cause_by == any_to_str(WriteCode) - assert project_repo.with_src_path(CONTEXT.src_workspace).srcs.changed_files + assert project_repo.with_src_path(context.src_workspace).srcs.changed_files def test_parse_str(): @@ -112,10 +105,8 @@ def test_todo(): @pytest.mark.asyncio -async def test_new_coding_context(): +async def test_new_coding_context(context): # Prerequisites - context = Context() - context.git_repo = GitRepository(local_path=DEFAULT_WORKSPACE_ROOT / f"unittest/{uuid.uuid4().hex}") demo_path = Path(__file__).parent / "../../data/demo_project" deps = json.loads(await aread(demo_path / "dependencies.json")) dependency = await context.git_repo.get_dependency() @@ -123,11 +114,11 @@ async def test_new_coding_context(): await dependency.update(k, set(v)) data = await aread(demo_path / "system_design.json") rqno = "20231221155954.json" - await awrite(context.git_repo.workdir / SYSTEM_DESIGN_FILE_REPO / rqno, data) + await awrite(context.repo.workdir / SYSTEM_DESIGN_FILE_REPO / rqno, data) data = await aread(demo_path / "tasks.json") - await awrite(context.git_repo.workdir / TASK_FILE_REPO / rqno, data) + await awrite(context.repo.workdir / TASK_FILE_REPO / rqno, data) - context.src_workspace = Path(context.git_repo.workdir) / "game_2048" + context.src_workspace = Path(context.repo.workdir) / "game_2048" try: filename = "game.py" @@ -149,9 +140,7 @@ async def test_new_coding_context(): context.git_repo.add_change({f"{TASK_FILE_REPO}/{rqno}": ChangeType.UNTRACTED}) context.git_repo.commit("mock env") - await ProjectRepo(context.git_repo).with_src_path(context.src_workspace).srcs.save( - filename=filename, content="content" - ) + await context.repo.with_src_path(context.src_workspace).srcs.save(filename=filename, content="content") role = Engineer(context=context) assert not role.code_todos await role._new_code_actions() diff --git a/tests/metagpt/roles/test_invoice_ocr_assistant.py b/tests/metagpt/roles/test_invoice_ocr_assistant.py index e3a9259da..bedcd6712 100644 --- a/tests/metagpt/roles/test_invoice_ocr_assistant.py +++ b/tests/metagpt/roles/test_invoice_ocr_assistant.py @@ -41,9 +41,11 @@ from metagpt.schema import Message ), ], ) -async def test_invoice_ocr_assistant(query: str, invoice_path: Path, invoice_table_path: Path, expected_result: dict): +async def test_invoice_ocr_assistant( + query: str, invoice_path: Path, invoice_table_path: Path, expected_result: dict, context +): invoice_path = TEST_DATA_PATH / invoice_path - role = InvoiceOCRAssistant() + role = InvoiceOCRAssistant(context=context) await role.run(Message(content=query, instruct_content=InvoicePath(file_path=invoice_path))) invoice_table_path = DATA_PATH / invoice_table_path df = pd.read_excel(invoice_table_path) diff --git a/tests/metagpt/roles/test_product_manager.py b/tests/metagpt/roles/test_product_manager.py index 1083e81b0..59b5aa81a 100644 --- a/tests/metagpt/roles/test_product_manager.py +++ b/tests/metagpt/roles/test_product_manager.py @@ -5,17 +5,51 @@ @Author : alexanderwu @File : test_product_manager.py """ +import json + import pytest +from metagpt.actions import WritePRD +from metagpt.actions.prepare_documents import PrepareDocuments +from metagpt.const import REQUIREMENT_FILENAME +from metagpt.context import Context from metagpt.logs import logger from metagpt.roles import ProductManager +from metagpt.utils.common import any_to_str from tests.metagpt.roles.mock import MockMessages @pytest.mark.asyncio async def test_product_manager(new_filename): - product_manager = ProductManager() - rsp = await product_manager.run(MockMessages.req) - logger.info(rsp) - assert len(rsp.content) > 0 - assert rsp.content == MockMessages.req.content + context = Context() + try: + assert context.git_repo is None + assert context.repo is None + product_manager = ProductManager(context=context) + # prepare documents + rsp = await product_manager.run(MockMessages.req) + assert context.git_repo + assert context.repo + assert rsp.cause_by == any_to_str(PrepareDocuments) + assert REQUIREMENT_FILENAME in context.repo.docs.changed_files + + # write prd + rsp = await product_manager.run(rsp) + assert rsp.cause_by == any_to_str(WritePRD) + logger.info(rsp) + assert len(rsp.content) > 0 + doc = list(rsp.instruct_content.docs.values())[0] + m = json.loads(doc.content) + assert m["Original Requirements"] == MockMessages.req.content + + # nothing to do + rsp = await product_manager.run(rsp) + assert rsp is None + except Exception as e: + assert not e + finally: + context.git_repo.delete_repository() + + +if __name__ == "__main__": + pytest.main([__file__, "-s"]) diff --git a/tests/metagpt/roles/test_project_manager.py b/tests/metagpt/roles/test_project_manager.py index 9207623bc..9b016927e 100644 --- a/tests/metagpt/roles/test_project_manager.py +++ b/tests/metagpt/roles/test_project_manager.py @@ -13,7 +13,7 @@ from tests.metagpt.roles.mock import MockMessages @pytest.mark.asyncio -async def test_project_manager(): - project_manager = ProjectManager() +async def test_project_manager(context): + project_manager = ProjectManager(context=context) rsp = await project_manager.run(MockMessages.system_design) logger.info(rsp) diff --git a/tests/metagpt/roles/test_qa_engineer.py b/tests/metagpt/roles/test_qa_engineer.py index c51642e6a..b89e7d5eb 100644 --- a/tests/metagpt/roles/test_qa_engineer.py +++ b/tests/metagpt/roles/test_qa_engineer.py @@ -13,20 +13,19 @@ from pydantic import Field from metagpt.actions import DebugError, RunCode, WriteTest from metagpt.actions.summarize_code import SummarizeCode -from metagpt.context import CONTEXT from metagpt.environment import Environment from metagpt.roles import QaEngineer from metagpt.schema import Message from metagpt.utils.common import any_to_str, aread, awrite -async def test_qa(): +async def test_qa(context): # Prerequisites demo_path = Path(__file__).parent / "../../data/demo_project" - CONTEXT.src_workspace = Path(CONTEXT.git_repo.workdir) / "qa/game_2048" + context.src_workspace = Path(context.repo.workdir) / "qa/game_2048" data = await aread(filename=demo_path / "game.py", encoding="utf-8") - await awrite(filename=CONTEXT.src_workspace / "game.py", data=data, encoding="utf-8") - await awrite(filename=Path(CONTEXT.git_repo.workdir) / "requirements.txt", data="") + await awrite(filename=context.src_workspace / "game.py", data=data, encoding="utf-8") + await awrite(filename=Path(context.repo.workdir) / "requirements.txt", data="") class MockEnv(Environment): msgs: List[Message] = Field(default_factory=list) @@ -37,7 +36,7 @@ async def test_qa(): env = MockEnv() - role = QaEngineer() + role = QaEngineer(context=context) role.set_env(env) await role.run(with_message=Message(content="", cause_by=SummarizeCode)) assert env.msgs diff --git a/tests/metagpt/roles/test_researcher.py b/tests/metagpt/roles/test_researcher.py index 7d0ec450d..af81777ac 100644 --- a/tests/metagpt/roles/test_researcher.py +++ b/tests/metagpt/roles/test_researcher.py @@ -28,12 +28,12 @@ async def mock_llm_ask(self, prompt: str, system_msgs): @pytest.mark.asyncio -async def test_researcher(mocker, search_engine_mocker): +async def test_researcher(mocker, search_engine_mocker, context): with TemporaryDirectory() as dirname: topic = "dataiku vs. datarobot" mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", mock_llm_ask) researcher.RESEARCH_PATH = Path(dirname) - role = researcher.Researcher() + role = researcher.Researcher(context=context) for i in role.actions: if isinstance(i, CollectLinks): i.search_engine = SearchEngine(SearchEngineType.DUCK_DUCK_GO) @@ -41,7 +41,7 @@ async def test_researcher(mocker, search_engine_mocker): assert (researcher.RESEARCH_PATH / f"{topic}.md").read_text().startswith("# Research Report") -def test_write_report(mocker): +def test_write_report(mocker, context): with TemporaryDirectory() as dirname: for i, topic in enumerate( [ @@ -53,7 +53,7 @@ def test_write_report(mocker): ): researcher.RESEARCH_PATH = Path(dirname) content = "# Research Report" - researcher.Researcher().write_report(topic, content) + researcher.Researcher(context=context).write_report(topic, content) assert (researcher.RESEARCH_PATH / f"{i+1}. metagpt.md").read_text().startswith("# Research Report") diff --git a/tests/metagpt/roles/test_role.py b/tests/metagpt/roles/test_role.py index 809f5c735..8b11e2d4a 100644 --- a/tests/metagpt/roles/test_role.py +++ b/tests/metagpt/roles/test_role.py @@ -13,8 +13,8 @@ def test_role_desc(): assert role.desc == "Best Seller" -def test_role_human(): - role = Role(is_human=True) +def test_role_human(context): + role = Role(is_human=True, context=context) assert isinstance(role.llm, HumanProvider) diff --git a/tests/metagpt/roles/test_tutorial_assistant.py b/tests/metagpt/roles/test_tutorial_assistant.py index 0e6c1efb9..c12c2b26e 100644 --- a/tests/metagpt/roles/test_tutorial_assistant.py +++ b/tests/metagpt/roles/test_tutorial_assistant.py @@ -15,8 +15,8 @@ from metagpt.roles.tutorial_assistant import TutorialAssistant @pytest.mark.asyncio @pytest.mark.parametrize(("language", "topic"), [("Chinese", "Write a tutorial about pip")]) -async def test_tutorial_assistant(language: str, topic: str): - role = TutorialAssistant(language=language) +async def test_tutorial_assistant(language: str, topic: str, context): + role = TutorialAssistant(language=language, context=context) msg = await role.run(topic) assert TUTORIAL_PATH.exists() filename = msg.content diff --git a/tests/metagpt/serialize_deserialize/test_action.py b/tests/metagpt/serialize_deserialize/test_action.py index f66900241..d234a160f 100644 --- a/tests/metagpt/serialize_deserialize/test_action.py +++ b/tests/metagpt/serialize_deserialize/test_action.py @@ -5,23 +5,22 @@ import pytest from metagpt.actions import Action -from metagpt.llm import LLM @pytest.mark.asyncio -async def test_action_serdeser(): - action = Action() +async def test_action_serdeser(context): + action = Action(context=context) ser_action_dict = action.model_dump() assert "name" in ser_action_dict assert "llm" not in ser_action_dict # not export assert "__module_class_name" in ser_action_dict - action = Action(name="test") + action = Action(name="test", context=context) ser_action_dict = action.model_dump() assert "test" in ser_action_dict["name"] - new_action = Action(**ser_action_dict) + new_action = Action(**ser_action_dict, context=context) assert new_action.name == "test" - assert isinstance(new_action.llm, type(LLM())) + assert isinstance(new_action.llm, type(context.llm())) assert len(await new_action._aask("who are you")) > 0 diff --git a/tests/metagpt/serialize_deserialize/test_architect.py b/tests/metagpt/serialize_deserialize/test_architect.py index a6823197a..e3c2703fa 100644 --- a/tests/metagpt/serialize_deserialize/test_architect.py +++ b/tests/metagpt/serialize_deserialize/test_architect.py @@ -9,16 +9,20 @@ from metagpt.roles.architect import Architect @pytest.mark.asyncio -async def test_architect_serdeser(): - role = Architect() +async def test_architect_serdeser(context): + role = Architect(context=context) ser_role_dict = role.model_dump(by_alias=True) assert "name" in ser_role_dict assert "states" in ser_role_dict assert "actions" in ser_role_dict - new_role = Architect(**ser_role_dict) + new_role = Architect(**ser_role_dict, context=context) assert new_role.name == "Bob" assert len(new_role.actions) == 1 assert len(new_role.rc.watch) == 1 assert isinstance(new_role.actions[0], Action) await new_role.actions[0].run(with_messages="write a cli snake game") + + +if __name__ == "__main__": + pytest.main([__file__, "-s"]) diff --git a/tests/metagpt/serialize_deserialize/test_environment.py b/tests/metagpt/serialize_deserialize/test_environment.py index 3e2a3abba..4e6ea93b5 100644 --- a/tests/metagpt/serialize_deserialize/test_environment.py +++ b/tests/metagpt/serialize_deserialize/test_environment.py @@ -18,20 +18,20 @@ from tests.metagpt.serialize_deserialize.test_serdeser_base import ( ) -def test_env_serdeser(): - env = Environment() +def test_env_serdeser(context): + env = Environment(context=context) env.publish_message(message=Message(content="test env serialize")) ser_env_dict = env.model_dump() assert "roles" in ser_env_dict assert len(ser_env_dict["roles"]) == 0 - new_env = Environment(**ser_env_dict) + new_env = Environment(**ser_env_dict, context=context) assert len(new_env.roles) == 0 assert len(new_env.history) == 25 -def test_environment_serdeser(): +def test_environment_serdeser(context): out_mapping = {"field1": (list[str], ...)} out_data = {"field1": ["field1 value1", "field1 value2"]} ic_obj = ActionNode.create_model_class("prd", out_mapping) @@ -40,7 +40,7 @@ def test_environment_serdeser(): content="prd", instruct_content=ic_obj(**out_data), role="product manager", cause_by=any_to_str(UserRequirement) ) - environment = Environment() + environment = Environment(context=context) role_c = RoleC() environment.add_role(role_c) environment.publish_message(message) @@ -48,7 +48,7 @@ def test_environment_serdeser(): ser_data = environment.model_dump() assert ser_data["roles"]["Role C"]["name"] == "RoleC" - new_env: Environment = Environment(**ser_data) + new_env: Environment = Environment(**ser_data, context=context) assert len(new_env.roles) == 1 assert list(new_env.roles.values())[0].states == list(environment.roles.values())[0].states @@ -57,22 +57,22 @@ def test_environment_serdeser(): assert type(list(new_env.roles.values())[0].actions[1]) == ActionRaise -def test_environment_serdeser_v2(): - environment = Environment() +def test_environment_serdeser_v2(context): + environment = Environment(context=context) pm = ProjectManager() environment.add_role(pm) ser_data = environment.model_dump() - new_env: Environment = Environment(**ser_data) + new_env: Environment = Environment(**ser_data, context=context) role = new_env.get_role(pm.profile) assert isinstance(role, ProjectManager) assert isinstance(role.actions[0], WriteTasks) assert isinstance(list(new_env.roles.values())[0].actions[0], WriteTasks) -def test_environment_serdeser_save(): - environment = Environment() +def test_environment_serdeser_save(context): + environment = Environment(context=context) role_c = RoleC() stg_path = serdeser_path.joinpath("team", "environment") @@ -82,6 +82,6 @@ def test_environment_serdeser_save(): write_json_file(env_path, environment.model_dump()) env_dict = read_json_file(env_path) - new_env: Environment = Environment(**env_dict) + new_env: Environment = Environment(**env_dict, context=context) assert len(new_env.roles) == 1 assert type(list(new_env.roles.values())[0].actions[0]) == ActionOK diff --git a/tests/metagpt/serialize_deserialize/test_memory.py b/tests/metagpt/serialize_deserialize/test_memory.py index fdaea7861..560ae2c51 100644 --- a/tests/metagpt/serialize_deserialize/test_memory.py +++ b/tests/metagpt/serialize_deserialize/test_memory.py @@ -13,7 +13,7 @@ from metagpt.utils.common import any_to_str, read_json_file, write_json_file from tests.metagpt.serialize_deserialize.test_serdeser_base import serdeser_path -def test_memory_serdeser(): +def test_memory_serdeser(context): msg1 = Message(role="Boss", content="write a snake game", cause_by=UserRequirement) out_mapping = {"field2": (list[str], ...)} @@ -39,7 +39,7 @@ def test_memory_serdeser(): assert memory.count() == 2 -def test_memory_serdeser_save(): +def test_memory_serdeser_save(context): msg1 = Message(role="User", content="write a 2048 game", cause_by=UserRequirement) out_mapping = {"field1": (list[str], ...)} diff --git a/tests/metagpt/serialize_deserialize/test_prepare_interview.py b/tests/metagpt/serialize_deserialize/test_prepare_interview.py index 3b57aa27e..a3e3edafc 100644 --- a/tests/metagpt/serialize_deserialize/test_prepare_interview.py +++ b/tests/metagpt/serialize_deserialize/test_prepare_interview.py @@ -8,12 +8,12 @@ from metagpt.actions.prepare_interview import PrepareInterview @pytest.mark.asyncio -async def test_action_serdeser(): - action = PrepareInterview() +async def test_action_serdeser(context): + action = PrepareInterview(context=context) serialized_data = action.model_dump() assert serialized_data["name"] == "PrepareInterview" - new_action = PrepareInterview(**serialized_data) + new_action = PrepareInterview(**serialized_data, context=context) assert new_action.name == "PrepareInterview" assert type(await new_action.run("python developer")) == ActionNode diff --git a/tests/metagpt/serialize_deserialize/test_product_manager.py b/tests/metagpt/serialize_deserialize/test_product_manager.py index 1a056f9d4..2338b406d 100644 --- a/tests/metagpt/serialize_deserialize/test_product_manager.py +++ b/tests/metagpt/serialize_deserialize/test_product_manager.py @@ -10,10 +10,10 @@ from metagpt.schema import Message @pytest.mark.asyncio -async def test_product_manager_serdeser(new_filename): - role = ProductManager() +async def test_product_manager_serdeser(new_filename, context): + role = ProductManager(context=context) ser_role_dict = role.model_dump(by_alias=True) - new_role = ProductManager(**ser_role_dict) + new_role = ProductManager(**ser_role_dict, context=context) assert new_role.name == "Alice" assert len(new_role.actions) == 2 diff --git a/tests/metagpt/serialize_deserialize/test_project_manager.py b/tests/metagpt/serialize_deserialize/test_project_manager.py index f2c5af853..fb998ae31 100644 --- a/tests/metagpt/serialize_deserialize/test_project_manager.py +++ b/tests/metagpt/serialize_deserialize/test_project_manager.py @@ -10,14 +10,14 @@ from metagpt.roles.project_manager import ProjectManager @pytest.mark.asyncio -async def test_project_manager_serdeser(): - role = ProjectManager() +async def test_project_manager_serdeser(context): + role = ProjectManager(context=context) ser_role_dict = role.model_dump(by_alias=True) assert "name" in ser_role_dict assert "states" in ser_role_dict assert "actions" in ser_role_dict - new_role = ProjectManager(**ser_role_dict) + new_role = ProjectManager(**ser_role_dict, context=context) assert new_role.name == "Eve" assert len(new_role.actions) == 1 assert isinstance(new_role.actions[0], Action) diff --git a/tests/metagpt/serialize_deserialize/test_reasearcher.py b/tests/metagpt/serialize_deserialize/test_reasearcher.py index a2d1fa513..67c52e692 100644 --- a/tests/metagpt/serialize_deserialize/test_reasearcher.py +++ b/tests/metagpt/serialize_deserialize/test_reasearcher.py @@ -8,13 +8,13 @@ from metagpt.roles.researcher import Researcher @pytest.mark.asyncio -async def test_tutorial_assistant_serdeser(): - role = Researcher() +async def test_tutorial_assistant_serdeser(context): + role = Researcher(context=context) ser_role_dict = role.model_dump() assert "name" in ser_role_dict assert "language" in ser_role_dict - new_role = Researcher(**ser_role_dict) + new_role = Researcher(**ser_role_dict, context=context) assert new_role.language == "en-us" assert len(new_role.actions) == 3 assert isinstance(new_role.actions[0], CollectLinks) diff --git a/tests/metagpt/serialize_deserialize/test_role.py b/tests/metagpt/serialize_deserialize/test_role.py index bbfe350b7..aaf7c1935 100644 --- a/tests/metagpt/serialize_deserialize/test_role.py +++ b/tests/metagpt/serialize_deserialize/test_role.py @@ -26,7 +26,7 @@ from tests.metagpt.serialize_deserialize.test_serdeser_base import ( ) -def test_roles(): +def test_roles(context): role_a = RoleA() assert len(role_a.rc.watch) == 1 role_b = RoleB() @@ -37,7 +37,7 @@ def test_roles(): assert len(role_d.actions) == 1 -def test_role_subclasses(): +def test_role_subclasses(context): """test subclasses of role with same fields in ser&deser""" class RoleSubClasses(BaseModel): @@ -51,7 +51,7 @@ def test_role_subclasses(): assert isinstance(new_role_subcls.roles[1], RoleB) -def test_role_serialize(): +def test_role_serialize(context): role = Role() ser_role_dict = role.model_dump() assert "name" in ser_role_dict @@ -59,7 +59,7 @@ def test_role_serialize(): assert "actions" in ser_role_dict -def test_engineer_serdeser(): +def test_engineer_serdeser(context): role = Engineer() ser_role_dict = role.model_dump() assert "name" in ser_role_dict @@ -73,7 +73,7 @@ def test_engineer_serdeser(): assert isinstance(new_role.actions[0], WriteCode) -def test_role_serdeser_save(): +def test_role_serdeser_save(context): shutil.rmtree(serdeser_path.joinpath("team"), ignore_errors=True) pm = ProductManager() @@ -89,7 +89,7 @@ def test_role_serdeser_save(): @pytest.mark.asyncio -async def test_role_serdeser_interrupt(): +async def test_role_serdeser_interrupt(context): role_c = RoleC() shutil.rmtree(serdeser_path.joinpath("team"), ignore_errors=True) diff --git a/tests/metagpt/serialize_deserialize/test_team.py b/tests/metagpt/serialize_deserialize/test_team.py index 57c8a8508..d45c8cf21 100644 --- a/tests/metagpt/serialize_deserialize/test_team.py +++ b/tests/metagpt/serialize_deserialize/test_team.py @@ -21,8 +21,8 @@ from tests.metagpt.serialize_deserialize.test_serdeser_base import ( ) -def test_team_deserialize(): - company = Team() +def test_team_deserialize(context): + company = Team(context=context) pm = ProductManager() arch = Architect() @@ -52,10 +52,10 @@ def mock_team_serialize(self, stg_path: Path = serdeser_path.joinpath("team")): write_json_file(team_info_path, self.model_dump()) -def test_team_serdeser_save(mocker): +def test_team_serdeser_save(mocker, context): mocker.patch("metagpt.team.Team.serialize", mock_team_serialize) - company = Team() + company = Team(context=context) company.hire([RoleC()]) stg_path = serdeser_path.joinpath("team") @@ -69,14 +69,14 @@ def test_team_serdeser_save(mocker): @pytest.mark.asyncio -async def test_team_recover(mocker): +async def test_team_recover(mocker, context): mocker.patch("metagpt.team.Team.serialize", mock_team_serialize) idea = "write a snake game" stg_path = serdeser_path.joinpath("team") shutil.rmtree(stg_path, ignore_errors=True) - company = Team() + company = Team(context=context) role_c = RoleC() company.hire([role_c]) company.run_project(idea) @@ -95,14 +95,14 @@ async def test_team_recover(mocker): @pytest.mark.asyncio -async def test_team_recover_save(mocker): +async def test_team_recover_save(mocker, context): mocker.patch("metagpt.team.Team.serialize", mock_team_serialize) idea = "write a 2048 web game" stg_path = serdeser_path.joinpath("team") shutil.rmtree(stg_path, ignore_errors=True) - company = Team() + company = Team(context=context) role_c = RoleC() company.hire([role_c]) company.run_project(idea) @@ -121,7 +121,7 @@ async def test_team_recover_save(mocker): @pytest.mark.asyncio -async def test_team_recover_multi_roles_save(mocker): +async def test_team_recover_multi_roles_save(mocker, context): mocker.patch("metagpt.team.Team.serialize", mock_team_serialize) idea = "write a snake game" @@ -131,7 +131,7 @@ async def test_team_recover_multi_roles_save(mocker): role_a = RoleA() role_b = RoleB() - company = Team() + company = Team(context=context) company.hire([role_a, role_b]) company.run_project(idea) await company.run(n_round=4) diff --git a/tests/metagpt/serialize_deserialize/test_tutorial_assistant.py b/tests/metagpt/serialize_deserialize/test_tutorial_assistant.py index cb8feec19..ab5db4c57 100644 --- a/tests/metagpt/serialize_deserialize/test_tutorial_assistant.py +++ b/tests/metagpt/serialize_deserialize/test_tutorial_assistant.py @@ -7,7 +7,7 @@ from metagpt.roles.tutorial_assistant import TutorialAssistant @pytest.mark.asyncio -async def test_tutorial_assistant_serdeser(): +async def test_tutorial_assistant_serdeser(context): role = TutorialAssistant() ser_role_dict = role.model_dump() assert "name" in ser_role_dict diff --git a/tests/metagpt/serialize_deserialize/test_write_code.py b/tests/metagpt/serialize_deserialize/test_write_code.py index 132f343bc..2f3c08f9b 100644 --- a/tests/metagpt/serialize_deserialize/test_write_code.py +++ b/tests/metagpt/serialize_deserialize/test_write_code.py @@ -9,22 +9,23 @@ from metagpt.actions import WriteCode from metagpt.schema import CodingContext, Document -def test_write_design_serdeser(): - action = WriteCode() +def test_write_design_serdeser(context): + action = WriteCode(context=context) ser_action_dict = action.model_dump() assert ser_action_dict["name"] == "WriteCode" assert "llm" not in ser_action_dict # not export @pytest.mark.asyncio -async def test_write_code_serdeser(): - context = CodingContext( +async def test_write_code_serdeser(context): + context.src_workspace = context.repo.workdir / "srcs" + coding_context = CodingContext( filename="test_code.py", design_doc=Document(content="write add function to calculate two numbers") ) - doc = Document(content=context.model_dump_json()) - action = WriteCode(i_context=doc) + doc = Document(content=coding_context.model_dump_json()) + action = WriteCode(i_context=doc, context=context) serialized_data = action.model_dump() - new_action = WriteCode(**serialized_data) + new_action = WriteCode(**serialized_data, context=context) assert new_action.name == "WriteCode" await action.run() diff --git a/tests/metagpt/serialize_deserialize/test_write_code_review.py b/tests/metagpt/serialize_deserialize/test_write_code_review.py index 70a4f2077..32a017a97 100644 --- a/tests/metagpt/serialize_deserialize/test_write_code_review.py +++ b/tests/metagpt/serialize_deserialize/test_write_code_review.py @@ -9,22 +9,23 @@ from metagpt.schema import CodingContext, Document @pytest.mark.asyncio -async def test_write_code_review_serdeser(): +async def test_write_code_review_serdeser(context): + context.src_workspace = context.repo.workdir / "srcs" code_content = """ def div(a: int, b: int = 0): return a / b """ - context = CodingContext( + coding_context = CodingContext( filename="test_op.py", design_doc=Document(content="divide two numbers"), code_doc=Document(content=code_content), ) - action = WriteCodeReview(i_context=context) + action = WriteCodeReview(i_context=coding_context) serialized_data = action.model_dump() assert serialized_data["name"] == "WriteCodeReview" - new_action = WriteCodeReview(**serialized_data) + new_action = WriteCodeReview(**serialized_data, context=context) assert new_action.name == "WriteCodeReview" await new_action.run() diff --git a/tests/metagpt/serialize_deserialize/test_write_design.py b/tests/metagpt/serialize_deserialize/test_write_design.py index 37d505914..6519d8cdc 100644 --- a/tests/metagpt/serialize_deserialize/test_write_design.py +++ b/tests/metagpt/serialize_deserialize/test_write_design.py @@ -8,24 +8,24 @@ from metagpt.actions import WriteDesign, WriteTasks @pytest.mark.asyncio -async def test_write_design_serialize(): - action = WriteDesign() +async def test_write_design_serialize(context): + action = WriteDesign(context=context) ser_action_dict = action.model_dump() assert "name" in ser_action_dict assert "llm" not in ser_action_dict # not export - new_action = WriteDesign(**ser_action_dict) + new_action = WriteDesign(**ser_action_dict, context=context) assert new_action.name == "WriteDesign" await new_action.run(with_messages="write a cli snake game") @pytest.mark.asyncio -async def test_write_task_serialize(): - action = WriteTasks() +async def test_write_task_serialize(context): + action = WriteTasks(context=context) ser_action_dict = action.model_dump() assert "name" in ser_action_dict assert "llm" not in ser_action_dict # not export - new_action = WriteTasks(**ser_action_dict) + new_action = WriteTasks(**ser_action_dict, context=context) assert new_action.name == "WriteTasks" await new_action.run(with_messages="write a cli snake game") diff --git a/tests/metagpt/serialize_deserialize/test_write_docstring.py b/tests/metagpt/serialize_deserialize/test_write_docstring.py index fb927f089..363bed05e 100644 --- a/tests/metagpt/serialize_deserialize/test_write_docstring.py +++ b/tests/metagpt/serialize_deserialize/test_write_docstring.py @@ -29,14 +29,14 @@ class Person: ], ids=["google", "numpy", "sphinx"], ) -async def test_action_serdeser(style: str, part: str): - action = WriteDocstring() +async def test_action_serdeser(style: str, part: str, context): + action = WriteDocstring(context=context) serialized_data = action.model_dump() assert "name" in serialized_data assert serialized_data["desc"] == "Write docstring for code." - new_action = WriteDocstring(**serialized_data) + new_action = WriteDocstring(**serialized_data, context=context) assert new_action.name == "WriteDocstring" assert new_action.desc == "Write docstring for code." diff --git a/tests/metagpt/serialize_deserialize/test_write_prd.py b/tests/metagpt/serialize_deserialize/test_write_prd.py index afc483e9a..e4951efb7 100644 --- a/tests/metagpt/serialize_deserialize/test_write_prd.py +++ b/tests/metagpt/serialize_deserialize/test_write_prd.py @@ -10,13 +10,13 @@ from metagpt.schema import Message @pytest.mark.asyncio -async def test_action_serdeser(new_filename): - action = WritePRD() +async def test_action_serdeser(new_filename, context): + action = WritePRD(context=context) ser_action_dict = action.model_dump() assert "name" in ser_action_dict assert "llm" not in ser_action_dict # not export - new_action = WritePRD(**ser_action_dict) + new_action = WritePRD(**ser_action_dict, context=context) assert new_action.name == "WritePRD" with pytest.raises(FileNotFoundError): await new_action.run(with_messages=Message(content="write a cli snake game")) diff --git a/tests/metagpt/serialize_deserialize/test_write_review.py b/tests/metagpt/serialize_deserialize/test_write_review.py index 17e212276..de2fd9d7a 100644 --- a/tests/metagpt/serialize_deserialize/test_write_review.py +++ b/tests/metagpt/serialize_deserialize/test_write_review.py @@ -5,7 +5,7 @@ import pytest from metagpt.actions.action_node import ActionNode from metagpt.actions.write_review import WriteReview -CONTEXT = """ +TEMPLATE_CONTEXT = """ { "Language": "zh_cn", "Programming Language": "Python", @@ -42,13 +42,13 @@ CONTEXT = """ @pytest.mark.asyncio -async def test_action_serdeser(): - action = WriteReview() +async def test_action_serdeser(context): + action = WriteReview(context=context) serialized_data = action.model_dump() assert serialized_data["name"] == "WriteReview" - new_action = WriteReview(**serialized_data) - review = await new_action.run(CONTEXT) + new_action = WriteReview(**serialized_data, context=context) + review = await new_action.run(TEMPLATE_CONTEXT) assert new_action.name == "WriteReview" assert type(review) == ActionNode diff --git a/tests/metagpt/serialize_deserialize/test_write_tutorial.py b/tests/metagpt/serialize_deserialize/test_write_tutorial.py index 4eeef7e0d..d41b7b341 100644 --- a/tests/metagpt/serialize_deserialize/test_write_tutorial.py +++ b/tests/metagpt/serialize_deserialize/test_write_tutorial.py @@ -9,13 +9,13 @@ from metagpt.actions.write_tutorial import WriteContent, WriteDirectory @pytest.mark.asyncio @pytest.mark.parametrize(("language", "topic"), [("English", "Write a tutorial about Python")]) -async def test_write_directory_serdeser(language: str, topic: str): - action = WriteDirectory() +async def test_write_directory_serdeser(language: str, topic: str, context): + action = WriteDirectory(context=context) serialized_data = action.model_dump() assert serialized_data["name"] == "WriteDirectory" assert serialized_data["language"] == "Chinese" - new_action = WriteDirectory(**serialized_data) + new_action = WriteDirectory(**serialized_data, context=context) ret = await new_action.run(topic=topic) assert isinstance(ret, dict) assert "title" in ret @@ -30,12 +30,12 @@ async def test_write_directory_serdeser(language: str, topic: str): ("language", "topic", "directory"), [("English", "Write a tutorial about Python", {"Introduction": ["What is Python?", "Why learn Python?"]})], ) -async def test_write_content_serdeser(language: str, topic: str, directory: Dict): - action = WriteContent(language=language, directory=directory) +async def test_write_content_serdeser(language: str, topic: str, directory: Dict, context): + action = WriteContent(language=language, directory=directory, context=context) serialized_data = action.model_dump() assert serialized_data["name"] == "WriteContent" - new_action = WriteContent(**serialized_data) + new_action = WriteContent(**serialized_data, context=context) ret = await new_action.run(topic=topic) assert isinstance(ret, str) assert list(directory.keys())[0] in ret diff --git a/tests/metagpt/test_context_mixin.py b/tests/metagpt/test_context_mixin.py index a8a096d69..1ef0e4832 100644 --- a/tests/metagpt/test_context_mixin.py +++ b/tests/metagpt/test_context_mixin.py @@ -5,11 +5,14 @@ @Author : alexanderwu @File : test_context_mixin.py """ +from pathlib import Path + import pytest from pydantic import BaseModel from metagpt.actions import Action from metagpt.config2 import Config +from metagpt.const import CONFIG_ROOT from metagpt.context_mixin import ContextMixin from metagpt.environment import Environment from metagpt.roles import Role @@ -101,7 +104,10 @@ def test_config_mixin_4_multi_inheritance_override_config(): @pytest.mark.asyncio async def test_config_priority(): """If action's config is set, then its llm will be set, otherwise, it will use the role's llm""" + home_dir = Path.home() / CONFIG_ROOT gpt4t = Config.from_home("gpt-4-1106-preview.yaml") + if not home_dir.exists(): + assert gpt4t is None gpt35 = Config.default() gpt4 = Config.default() gpt4.llm.model = "gpt-4-0613" @@ -120,7 +126,7 @@ async def test_config_priority(): env = Environment(desc="US election live broadcast") Team(investment=10.0, env=env, roles=[A, B, C]) - assert a1.llm.model == "gpt-4-1106-preview" + assert a1.llm.model == "gpt-4-1106-preview" if Path(home_dir / "gpt-4-1106-preview.yaml").exists() else "gpt-4-0613" assert a2.llm.model == "gpt-4-0613" assert a3.llm.model == "gpt-3.5-turbo-1106" diff --git a/tests/metagpt/tools/test_metagpt_oas3_api_svc.py b/tests/metagpt/tools/test_metagpt_oas3_api_svc.py index 3cf5e515b..5be139106 100644 --- a/tests/metagpt/tools/test_metagpt_oas3_api_svc.py +++ b/tests/metagpt/tools/test_metagpt_oas3_api_svc.py @@ -12,14 +12,12 @@ from pathlib import Path import pytest import requests -from metagpt.context import CONTEXT - @pytest.mark.asyncio -async def test_oas2_svc(): +async def test_oas2_svc(context): workdir = Path(__file__).parent.parent.parent.parent script_pathname = workdir / "metagpt/tools/metagpt_oas3_api_svc.py" - env = CONTEXT.new_environ() + env = context.new_environ() env["PYTHONPATH"] = str(workdir) + ":" + env.get("PYTHONPATH", "") process = subprocess.Popen(["python", str(script_pathname)], cwd=str(workdir), env=env) await asyncio.sleep(5) diff --git a/tests/metagpt/tools/test_openapi_v3_hello.py b/tests/metagpt/tools/test_openapi_v3_hello.py index daa5d21c6..f49b8412a 100644 --- a/tests/metagpt/tools/test_openapi_v3_hello.py +++ b/tests/metagpt/tools/test_openapi_v3_hello.py @@ -12,14 +12,12 @@ from pathlib import Path import pytest import requests -from metagpt.context import CONTEXT - @pytest.mark.asyncio -async def test_hello(): +async def test_hello(context): workdir = Path(__file__).parent.parent.parent.parent script_pathname = workdir / "metagpt/tools/openapi_v3_hello.py" - env = CONTEXT.new_environ() + env = context.new_environ() env["PYTHONPATH"] = str(workdir) + ":" + env.get("PYTHONPATH", "") process = subprocess.Popen(["python", str(script_pathname)], cwd=workdir, env=env) await asyncio.sleep(5) diff --git a/tests/metagpt/utils/test_mermaid.py b/tests/metagpt/utils/test_mermaid.py index 367223332..7e9129314 100644 --- a/tests/metagpt/utils/test_mermaid.py +++ b/tests/metagpt/utils/test_mermaid.py @@ -8,20 +8,19 @@ import pytest -from metagpt.context import CONTEXT from metagpt.utils.common import check_cmd_exists from metagpt.utils.mermaid import MMC1, mermaid_to_file @pytest.mark.asyncio @pytest.mark.parametrize("engine", ["nodejs", "ink"]) # TODO: playwright and pyppeteer -async def test_mermaid(engine): +async def test_mermaid(engine, context): # nodejs prerequisites: npm install -g @mermaid-js/mermaid-cli # ink prerequisites: connected to internet # playwright prerequisites: playwright install --with-deps chromium assert check_cmd_exists("npm") == 0 - save_to = CONTEXT.git_repo.workdir / f"{engine}/1" + save_to = context.git_repo.workdir / f"{engine}/1" await mermaid_to_file(engine, MMC1, save_to) # ink does not support pdf From db086e47e764cc8303c4b2fd64a0834f6108202a Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 18 Jan 2024 09:42:09 +0800 Subject: [PATCH 382/637] modify: get value of ActionNode by key of ActionNode --- metagpt/actions/design_api.py | 13 ++++++++++--- metagpt/actions/project_management.py | 8 ++++++-- metagpt/actions/write_code.py | 3 ++- metagpt/actions/write_prd.py | 3 ++- 4 files changed, 20 insertions(+), 7 deletions(-) diff --git a/metagpt/actions/design_api.py b/metagpt/actions/design_api.py index 073eb20ae..933c59b74 100644 --- a/metagpt/actions/design_api.py +++ b/metagpt/actions/design_api.py @@ -14,7 +14,14 @@ from pathlib import Path from typing import Optional from metagpt.actions import Action, ActionOutput -from metagpt.actions.design_api_an import DESIGN_API_NODE, REFINED_DESIGN_NODES +from metagpt.actions.design_api_an import ( + DATA_STRUCTURES_AND_INTERFACES, + DESIGN_API_NODE, + PROGRAM_CALL_FLOW, + REFINED_DATA_STRUCTURES_AND_INTERFACES, + REFINED_DESIGN_NODES, + REFINED_PROGRAM_CALL_FLOW, +) from metagpt.config import CONFIG from metagpt.const import ( DATA_API_DESIGN_FILE_REPO, @@ -109,7 +116,7 @@ class WriteDesign(Action): @staticmethod async def _save_data_api_design(design_doc): m = json.loads(design_doc.content) - data_api_design = m.get("Data structures and interfaces") or m.get("Refined Data structures and interfaces") + data_api_design = m.get(DATA_STRUCTURES_AND_INTERFACES.key) or m.get(REFINED_DATA_STRUCTURES_AND_INTERFACES.key) if not data_api_design: return pathname = CONFIG.git_repo.workdir / DATA_API_DESIGN_FILE_REPO / Path(design_doc.filename).with_suffix("") @@ -119,7 +126,7 @@ class WriteDesign(Action): @staticmethod async def _save_seq_flow(design_doc): m = json.loads(design_doc.content) - seq_flow = m.get("Program call flow") or m.get("Refined Program call flow") + seq_flow = m.get(PROGRAM_CALL_FLOW.key) or m.get(REFINED_PROGRAM_CALL_FLOW.key) if not seq_flow: return pathname = CONFIG.git_repo.workdir / Path(SEQ_FLOW_FILE_REPO) / Path(design_doc.filename).with_suffix("") diff --git a/metagpt/actions/project_management.py b/metagpt/actions/project_management.py index 3c379d96f..6e298e500 100644 --- a/metagpt/actions/project_management.py +++ b/metagpt/actions/project_management.py @@ -15,7 +15,11 @@ from typing import Optional from metagpt.actions import ActionOutput from metagpt.actions.action import Action -from metagpt.actions.project_management_an import PM_NODE, REFINED_PM_NODES +from metagpt.actions.project_management_an import ( + PM_NODE, + REFINED_PM_NODES, + REQUIRED_PYTHON_PACKAGES, +) from metagpt.config import CONFIG from metagpt.const import ( PACKAGE_REQUIREMENTS_FILENAME, @@ -100,7 +104,7 @@ class WriteTasks(Action): @staticmethod async def _update_requirements(doc): m = json.loads(doc.content) - packages = set(m.get("Required Python third-party packages", set())) + packages = set(m.get(REQUIRED_PYTHON_PACKAGES.key, set())) file_repo = CONFIG.git_repo.new_file_repository() requirement_doc = await file_repo.get(filename=PACKAGE_REQUIREMENTS_FILENAME) if not requirement_doc: diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index ce0e2fe3b..30ebf09f8 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -21,6 +21,7 @@ from pydantic import Field from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions.action import Action +from metagpt.actions.project_management_an import REFINED_TASK_LIST, TASK_LIST from metagpt.actions.write_code_guideline_an import REFINED_CODE_TEMPLATE from metagpt.config import CONFIG from metagpt.const import ( @@ -180,7 +181,7 @@ class WriteCode(Action): if not task_doc.content: task_doc.content = FileRepository.get_file(filename=task_doc.filename, relative_path=TASK_FILE_REPO) m = json.loads(task_doc.content) - code_filenames = m.get("Task list", []) if mode == "normal" else m.get("Refined Task list", []) + code_filenames = m.get(TASK_LIST.key, []) if mode == "normal" else m.get(REFINED_TASK_LIST.key, []) codes = [] src_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.src_workspace) diff --git a/metagpt/actions/write_prd.py b/metagpt/actions/write_prd.py index 84a9fa13d..e12c1e1ec 100644 --- a/metagpt/actions/write_prd.py +++ b/metagpt/actions/write_prd.py @@ -21,6 +21,7 @@ from metagpt.actions import Action, ActionOutput from metagpt.actions.action_node import ActionNode from metagpt.actions.fix_bug import FixBug from metagpt.actions.write_prd_an import ( + COMPETITIVE_QUADRANT_CHART, PROJECT_NAME, REFINE_PRD_NODE, REFINE_PRD_TEMPLATE, @@ -166,7 +167,7 @@ class WritePRD(Action): @staticmethod async def _save_competitive_analysis(prd_doc): m = json.loads(prd_doc.content) - quadrant_chart = m.get("Competitive Quadrant Chart") + quadrant_chart = m.get(COMPETITIVE_QUADRANT_CHART.key) if not quadrant_chart: return pathname = ( From fe64b23a0ec2f38f8b47368a6ad603079989520b Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 18 Jan 2024 10:01:33 +0800 Subject: [PATCH 383/637] update test file of ActionNode --- tests/metagpt/actions/test_design_api_an.py | 19 +++++++++++------ .../actions/test_project_management_an.py | 17 +++++++++------ .../actions/test_write_code_guideline_an.py | 11 ++++++++-- tests/metagpt/actions/test_write_prd_an.py | 21 ++++++++++++------- 4 files changed, 47 insertions(+), 21 deletions(-) diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index 2aa123224..39de2a595 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -6,6 +6,7 @@ @File : test_design_api_an.py """ import pytest +from openai._models import BaseModel from metagpt.actions.action_node import ActionNode, dict_to_markdown from metagpt.actions.design_api import NEW_REQ_TEMPLATE @@ -23,17 +24,23 @@ def llm(): return LLM() +def mock_refined_design_json(): + return REFINED_DESIGN_JSON + + @pytest.mark.asyncio async def test_write_design_an(mocker): root = ActionNode.from_children( "RefinedDesignAPI", [ActionNode(key="", expected_type=str, instruction="", example="")] ) - root.instruct_content = REFINED_DESIGN_JSON - + root.instruct_content = BaseModel() + root.instruct_content.model_dump = mock_refined_design_json mocker.patch("metagpt.actions.design_api_an.REFINED_DESIGN_NODES.fill", return_value=root) + prompt = NEW_REQ_TEMPLATE.format(old_design=DESIGN_SAMPLE, context=dict_to_markdown(REFINED_PRD_JSON)) node = await REFINED_DESIGN_NODES.fill(prompt, llm) - assert "Refined Implementation Approach" in node.instruct_content - assert "Refined File list" in node.instruct_content - assert "Refined Data structures and interfaces" in node.instruct_content - assert "Refined Program call flow" in node.instruct_content + + assert "Refined Implementation Approach" in node.instruct_content.model_dump() + assert "Refined File list" in node.instruct_content.model_dump() + assert "Refined Data structures and interfaces" in node.instruct_content.model_dump() + assert "Refined Program call flow" in node.instruct_content.model_dump() diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index 0540ed6e1..50dc47067 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -6,6 +6,7 @@ @File : test_project_management_an.py """ import pytest +from openai._models import BaseModel from metagpt.actions.action_node import ActionNode, dict_to_markdown from metagpt.actions.project_management import NEW_REQ_TEMPLATE @@ -23,18 +24,22 @@ def llm(): return LLM() +def mock_refined_tasks_json(): + return REFINED_TASKS_JSON + + @pytest.mark.asyncio async def test_project_management_an(mocker): root = ActionNode.from_children( "RefinedProjectManagement", [ActionNode(key="", expected_type=str, instruction="", example="")] ) - root.instruct_content = REFINED_TASKS_JSON - + root.instruct_content = BaseModel() + root.instruct_content.model_dump = mock_refined_tasks_json mocker.patch("metagpt.actions.project_management_an.REFINED_PM_NODES.fill", return_value=root) prompt = NEW_REQ_TEMPLATE.format(old_tasks=TASKS_SAMPLE, context=dict_to_markdown(REFINED_DESIGN_JSON)) node = await REFINED_PM_NODES.fill(prompt, llm) - assert node.instruct_content - assert "Refined Logic Analysis" in node.instruct_content - assert "Refined Task list" in node.instruct_content - assert "Refined Shared Knowledge" in node.instruct_content + + assert "Refined Logic Analysis" in node.instruct_content.model_dump() + assert "Refined Task list" in node.instruct_content.model_dump() + assert "Refined Shared Knowledge" in node.instruct_content.model_dump() diff --git a/tests/metagpt/actions/test_write_code_guideline_an.py b/tests/metagpt/actions/test_write_code_guideline_an.py index 998605c4b..5a4e19d57 100644 --- a/tests/metagpt/actions/test_write_code_guideline_an.py +++ b/tests/metagpt/actions/test_write_code_guideline_an.py @@ -6,6 +6,7 @@ @File : test_write_code_guideline_an.py """ import pytest +from openai._models import BaseModel from metagpt.actions.action_node import ActionNode from metagpt.actions.write_code import WriteCode @@ -28,12 +29,17 @@ from tests.data.incremental_dev_project.mock import ( ) +def mock_guidelines_and_incremental_change(): + return GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE + + @pytest.mark.asyncio async def test_write_code_guideline_an(mocker): root = ActionNode.from_children( "WriteCodeGuideline", [ActionNode(key="", expected_type=str, instruction="", example="")] ) - root.instruct_content = GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE + root.instruct_content = BaseModel() + root.instruct_content.model_dump = mock_guidelines_and_incremental_change mocker.patch("metagpt.actions.write_code_guideline_an.WriteCodeGuideline.run", return_value=root) write_code_guideline = WriteCodeGuideline() @@ -45,7 +51,8 @@ async def test_write_code_guideline_an(mocker): code=OLD_CODE_SAMPLE, ) node = await write_code_guideline.run(context=context) - assert "Guidelines and Incremental Change" in node.instruct_content + + assert "Guidelines and Incremental Change" in node.instruct_content.model_dump() @pytest.mark.asyncio diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index e7f288c68..1fdaa75c2 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -6,6 +6,7 @@ @File : test_write_prd_an.py """ import pytest +from openai._models import BaseModel from metagpt.actions.action_node import ActionNode from metagpt.actions.write_prd_an import REFINE_PRD_NODE, REFINE_PRD_TEMPLATE @@ -22,20 +23,26 @@ def llm(): return LLM() +def mock_refined_prd_json(): + return REFINED_PRD_JSON + + @pytest.mark.asyncio async def test_write_prd_an(mocker): root = ActionNode.from_children("RefinePRD", [ActionNode(key="", expected_type=str, instruction="", example="")]) - root.instruct_content = REFINED_PRD_JSON - + root.instruct_content = BaseModel() + root.instruct_content.model_dump = mock_refined_prd_json mocker.patch("metagpt.actions.write_prd_an.REFINE_PRD_NODE.fill", return_value=root) + prompt = REFINE_PRD_TEMPLATE.format( requirements=NEW_REQUIREMENT_SAMPLE, old_prd=PRD_SAMPLE, project_name="", ) node = await REFINE_PRD_NODE.fill(prompt, llm) - assert "Refined Requirements" in node.instruct_content - assert "Refined Product Goals" in node.instruct_content - assert "Refined User Stories" in node.instruct_content - assert "Refined Requirement Analysis" in node.instruct_content - assert "Refined Requirement Pool" in node.instruct_content + + assert "Refined Requirements" in node.instruct_content.model_dump() + assert "Refined Product Goals" in node.instruct_content.model_dump() + assert "Refined User Stories" in node.instruct_content.model_dump() + assert "Refined Requirement Analysis" in node.instruct_content.model_dump() + assert "Refined Requirement Pool" in node.instruct_content.model_dump() From 2bc88cd71b45a6bcdc6e9c287f05e722f69b44f0 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 18 Jan 2024 10:15:36 +0800 Subject: [PATCH 384/637] update function of save code_guideline file --- metagpt/actions/write_code_guideline_an.py | 15 +++++---------- metagpt/roles/engineer.py | 7 +++---- 2 files changed, 8 insertions(+), 14 deletions(-) diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_guideline_an.py index ff59ee7d6..28af592c0 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_guideline_an.py @@ -187,20 +187,15 @@ class WriteCodeGuideline(Action): return await WRITE_CODE_GUIDELINE_NODE.fill(context=context, llm=self.llm, schema="json") @staticmethod - async def save(guideline): - await WriteCodeGuideline.save_json(guideline) - await WriteCodeGuideline.save_md(guideline) - - @staticmethod - async def save_json(guideline): - filename = "code_guideline.json" + async def save_json(guideline, filename="code_guideline.json"): await CONFIG.git_repo.new_file_repository(CODE_GUIDELINE_FILE_REPO).save( filename=filename, content=str(guideline) ) @staticmethod - async def save_md(guideline): - filename = "code_guideline.md" + async def save_md(guideline, filename="code_guideline.md"): + guideline_md = dict_to_markdown(guideline) await CONFIG.git_repo.new_file_repository(CODE_GUIDELINE_PDF_FILE_REPO).save( - filename=filename, content=dict_to_markdown(guideline) + filename=filename, content=guideline_md ) + return guideline_md diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index bb0a2e857..d767b1ebf 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -26,7 +26,6 @@ from pathlib import Path from typing import Set from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks -from metagpt.actions.action_node import dict_to_markdown from metagpt.actions.fix_bug import FixBug from metagpt.actions.summarize_code import SummarizeCode from metagpt.actions.write_code_guideline_an import ( @@ -367,10 +366,10 @@ class Engineer(Role): ) node = await WriteCodeGuideline().run(context=context) guideline = node.instruct_content.model_dump() - await WriteCodeGuideline.save(guideline) - guideline = dict_to_markdown(guideline) + await WriteCodeGuideline.save_json(guideline) + guideline_md = await WriteCodeGuideline.save_md(guideline) - return Document(root_path=CODE_GUIDELINE_PDF_FILE_REPO, filename="code_guideline.md", content=guideline) + return Document(root_path=CODE_GUIDELINE_PDF_FILE_REPO, filename="code_guideline.md", content=guideline_md) @staticmethod async def get_old_codes() -> str: From 526a37d950b531ed056dae3167a13d2d6b22c6e5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Thu, 18 Jan 2024 10:17:00 +0800 Subject: [PATCH 385/637] fixbug: unit test --- tests/metagpt/roles/test_engineer.py | 12 +++++------- 1 file changed, 5 insertions(+), 7 deletions(-) diff --git a/tests/metagpt/roles/test_engineer.py b/tests/metagpt/roles/test_engineer.py index 675e21aa0..383d28096 100644 --- a/tests/metagpt/roles/test_engineer.py +++ b/tests/metagpt/roles/test_engineer.py @@ -19,7 +19,6 @@ from metagpt.roles.engineer import Engineer from metagpt.schema import CodingContext, Message from metagpt.utils.common import CodeParser, any_to_name, any_to_str, aread, awrite from metagpt.utils.git_repository import ChangeType -from metagpt.utils.project_repo import ProjectRepo from tests.metagpt.roles.mock import STRS_FOR_PARSING, TASKS, MockMessages @@ -27,18 +26,17 @@ from tests.metagpt.roles.mock import STRS_FOR_PARSING, TASKS, MockMessages async def test_engineer(context): # Prerequisites rqno = "20231221155954.json" - project_repo = ProjectRepo(context.git_repo) - await project_repo.save(REQUIREMENT_FILENAME, content=MockMessages.req.content) - await project_repo.docs.prd.save(rqno, content=MockMessages.prd.content) - await project_repo.docs.system_design.save(rqno, content=MockMessages.system_design.content) - await project_repo.docs.task.save(rqno, content=MockMessages.json_tasks.content) + await context.repo.save(REQUIREMENT_FILENAME, content=MockMessages.req.content) + await context.repo.docs.prd.save(rqno, content=MockMessages.prd.content) + await context.repo.docs.system_design.save(rqno, content=MockMessages.system_design.content) + await context.repo.docs.task.save(rqno, content=MockMessages.json_tasks.content) engineer = Engineer(context=context) rsp = await engineer.run(Message(content="", cause_by=WriteTasks)) logger.info(rsp) assert rsp.cause_by == any_to_str(WriteCode) - assert project_repo.with_src_path(context.src_workspace).srcs.changed_files + assert context.repo.with_src_path(context.src_workspace).srcs.changed_files def test_parse_str(): From 10129c6ecf431b9f40b1dd2a349eb2cc0de5c024 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 18 Jan 2024 12:07:31 +0800 Subject: [PATCH 386/637] update scrape_web. --- metagpt/tools/__init__.py | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index 95872940f..222edf312 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -16,7 +16,7 @@ from metagpt.prompts.tool_type import ( FEATURE_ENGINEERING_PROMPT, MODEL_TRAIN_PROMPT, MODEL_EVALUATE_PROMPT, - VISION_PROMPT + VISION_PROMPT, ) @@ -81,7 +81,8 @@ TOOL_TYPE_MAPPINGS = { name="scrape_web", module="metagpt.tools.functions.libs.scrape_web.scrape_web", desc="Scrape data from web page.", - usage_prompt=""), + usage_prompt="", + ), "vision": ToolType( name="vision", module=str(TOOL_LIBS_PATH / "vision"), From f3612f8123941abfa5e8aae221ba5c1ab69512a0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 18 Jan 2024 12:10:37 +0800 Subject: [PATCH 387/637] add only_code arg for WriteCodeByGenerate. --- metagpt/roles/ml_engineer_simple.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/ml_engineer_simple.py b/metagpt/roles/ml_engineer_simple.py index 3f10af8d0..9ff1c9880 100644 --- a/metagpt/roles/ml_engineer_simple.py +++ b/metagpt/roles/ml_engineer_simple.py @@ -75,7 +75,7 @@ class MLEngineerSimple(Role): context = self.get_useful_memories() print(f"memories数量:{len(context)}") # print("===\n" +str(context) + "\n===") - code = await WriteCodeByGenerate().run(context=context, temperature=0.0) + code = await WriteCodeByGenerate().run(context=context, temperature=0.0, only_code=True) cause_by = WriteCodeByGenerate self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) From 002bc56c0e68a79b7e5311b6799c946d6dd633bf Mon Sep 17 00:00:00 2001 From: zhanglei Date: Thu, 18 Jan 2024 12:37:28 +0800 Subject: [PATCH 388/637] add: openai speech to text --- metagpt/provider/openai_api.py | 4 ++++ tests/metagpt/provider/test_openai.py | 9 +++++++++ 2 files changed, 13 insertions(+) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 3a9aca870..d6944eae6 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -239,3 +239,7 @@ class OpenAILLM(BaseLLM): async def atext_to_speech(self, **kwargs): """text to speech""" return await self.aclient.audio.speech.create(**kwargs) + + async def aspeech_to_text(self, **kwargs): + """speech to text""" + return await self.aclient.audio.transcriptions.create(**kwargs) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index 2d52ad10e..7a0dbe5c4 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -1,5 +1,6 @@ import pytest +from metagpt.const import TEST_DATA_PATH from metagpt.llm import LLM from metagpt.logs import logger from metagpt.provider import OpenAILLM @@ -53,6 +54,14 @@ async def test_text_to_speech(): assert 200 == resp.response.status_code +@pytest.mark.asyncio +async def test_speech_to_text(): + llm = LLM() + audio_file = open(f"{TEST_DATA_PATH}/audio/hello.mp3", "rb") + resp = await llm.aspeech_to_text(file=audio_file, model="whisper-1") + assert "你好" == resp.text + + class TestOpenAI: def test_make_client_kwargs_without_proxy(self): instance = OpenAILLM(mock_llm_config) From 99511fd264ec9354cb411c0762bd75f8850d7c74 Mon Sep 17 00:00:00 2001 From: zhanglei Date: Thu, 18 Jan 2024 12:41:53 +0800 Subject: [PATCH 389/637] update:OpenAI text to speech unittest --- tests/metagpt/provider/test_openai.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index 7a0dbe5c4..bc7f92f33 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -49,7 +49,7 @@ async def test_text_to_speech(): resp = await llm.atext_to_speech( model="tts-1", voice="alloy", - input="人生说起来长,但知道一个岁月回头看,许多事件仅是仓促的。一段一段拼凑一起,合成了人生。苦难当头时,当下不免觉得是折磨;回头看,也不够是一段短短的人生旅程。", + input="人生说起来长,但直到一个岁月回头看,许多事件仅是仓促的。一段一段拼凑一起,合成了人生。苦难当头时,当下不免觉得是折磨;回头看,也不够是一段短短的人生旅程。", ) assert 200 == resp.response.status_code From 89f92ffb87033b26596e81e3e5bd21f82bbcddb8 Mon Sep 17 00:00:00 2001 From: Sirui Hong <34952977+stellaHSR@users.noreply.github.com> Date: Thu, 18 Jan 2024 14:14:26 +0800 Subject: [PATCH 390/637] Update README.md --- README.md | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 026920700..1b05f35c5 100644 --- a/README.md +++ b/README.md @@ -34,7 +34,7 @@ # MetaGPT: The Multi-Agent Framework

Software Company Multi-Role Schematic (Gradually Implementing)

## News -🚀 Jan 16: Our paper: [MetaGPT: Meta Programming for A Multi-Agent Collaborative Framework](https://arxiv.org/abs/2308.00352) has been accepted by ICLR 2024 for oral presentation! More details are [here](https://openreview.net/forum?id=VtmBAGCN7o). +🚀 Jan 16: Our paper: [MetaGPT: Meta Programming for A Multi-Agent Collaborative Framework](https://arxiv.org/abs/2308.00352) has been accepted by ICLR 2024 for **oral presentation (top 1.2%)**! More details are [here](https://openreview.net/forum?id=VtmBAGCN7o). 🚀 Jan 03: Here comes [v0.6.0](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! In this version, we added serialization and deserialization of important objects and enabled breakpoint recovery. We upgraded OpenAI package to v1.6.0 and supported Gemini, ZhipuAI, Ollama, OpenLLM, etc. Moreover, we provided extremely simple examples where you need only 7 lines to implement a general election [debate](https://github.com/geekan/MetaGPT/blob/main/examples/debate_simple.py). Check out more details [here](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! @@ -42,6 +42,8 @@ ## News 🚀 Dec 15: [v0.5.0](https://github.com/geekan/MetaGPT/releases/tag/v0.5.0) is released! We introduced **incremental development**, facilitating agents to build up larger projects on top of their previous efforts or existing codebase. We also launched a whole collection of important features, including **multilingual support** (experimental), multiple **programming languages support** (experimental), **incremental development** (experimental), CLI support, pip support, enhanced code review, documentation mechanism, and optimized messaging mechanism! +🔥 Nov 8: MetaGPT is selected into [Open100: Top 100 Open Source achievements](https://www.benchcouncil.org/evaluation/opencs/annual.html). + ## Install ### Pip installation From 6fb48664cb56941f0466dd17b8afc89a62712aa4 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 18 Jan 2024 15:18:29 +0800 Subject: [PATCH 391/637] replace REQUIRED_PYTHON_PACKAGES.key to "Required Python packages" --- metagpt/actions/project_management.py | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/metagpt/actions/project_management.py b/metagpt/actions/project_management.py index 6e298e500..d6b351d18 100644 --- a/metagpt/actions/project_management.py +++ b/metagpt/actions/project_management.py @@ -15,11 +15,7 @@ from typing import Optional from metagpt.actions import ActionOutput from metagpt.actions.action import Action -from metagpt.actions.project_management_an import ( - PM_NODE, - REFINED_PM_NODES, - REQUIRED_PYTHON_PACKAGES, -) +from metagpt.actions.project_management_an import PM_NODE, REFINED_PM_NODES from metagpt.config import CONFIG from metagpt.const import ( PACKAGE_REQUIREMENTS_FILENAME, @@ -104,7 +100,7 @@ class WriteTasks(Action): @staticmethod async def _update_requirements(doc): m = json.loads(doc.content) - packages = set(m.get(REQUIRED_PYTHON_PACKAGES.key, set())) + packages = set(m.get("Required Python packages", set())) file_repo = CONFIG.git_repo.new_file_repository() requirement_doc = await file_repo.get(filename=PACKAGE_REQUIREMENTS_FILENAME) if not requirement_doc: From de0db068c201bc31e55addaa7ab5d01e2b7c4204 Mon Sep 17 00:00:00 2001 From: Sirui Hong <34952977+stellaHSR@users.noreply.github.com> Date: Thu, 18 Jan 2024 20:07:45 +0800 Subject: [PATCH 392/637] Update README.md update trending history --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 1b05f35c5..3ede740f7 100644 --- a/README.md +++ b/README.md @@ -44,6 +44,8 @@ ## News 🔥 Nov 8: MetaGPT is selected into [Open100: Top 100 Open Source achievements](https://www.benchcouncil.org/evaluation/opencs/annual.html). +🔥 Sep 1: MetaGPT clinched the top spot for the **17th time** in GitHub's Trending Monthly for August 2023. + ## Install ### Pip installation From d78db8994c6cb6c05f40c30891987358dcafd242 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 18 Jan 2024 20:57:43 +0800 Subject: [PATCH 393/637] delete arg only_code. --- metagpt/actions/write_analysis_code.py | 10 ++-------- 1 file changed, 2 insertions(+), 8 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index bceb100b1..9104fdf82 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -85,17 +85,11 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): plan: Plan = None, system_msg: str = None, **kwargs, - ) -> str: + ) -> dict: # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) prompt = self.process_msg(context, system_msg) - is_only_code = kwargs.pop("only_code", False) - code_content = await self.llm.aask_code(prompt, **kwargs) - if is_only_code: - return code_content["code"] - else: - return code_content - + return code_content class WriteCodeWithTools(BaseWriteAnalysisCode): From 46cd219e817eae2abf6d5a8b552bebf531672526 Mon Sep 17 00:00:00 2001 From: yzlin Date: Sat, 13 Jan 2024 01:28:49 +0800 Subject: [PATCH 394/637] add tool registry --- metagpt/actions/write_analysis_code.py | 58 +- metagpt/actions/write_plan.py | 7 +- metagpt/prompts/ml_engineer.py | 2 +- metagpt/tools/__init__.py | 11 - .../tools/functions/libs/data_preprocess.py | 13 + .../functions/libs/feature_engineering.py | 17 +- .../data_preprocess/FillMissingValue.yml | 61 ++ .../schemas/data_preprocess/LabelEncode.yml | 48 ++ .../schemas/data_preprocess/MaxAbsScale.yml | 48 ++ .../schemas/data_preprocess/MinMaxScale.yml | 48 ++ .../schemas/data_preprocess/OneHotEncode.yml | 48 ++ .../schemas/data_preprocess/StandardScale.yml | 48 ++ .../schemas/feature_engineering/CatCount.yml | 48 ++ .../schemas/feature_engineering/CatCross.yml | 52 ++ .../feature_engineering/GeneralSelection.yml | 48 ++ .../schemas/feature_engineering/GroupStat.yml | 58 ++ .../KFoldTargetMeanEncoder.yml | 60 ++ .../PolynomialExpansion.yml | 548 ++++++++++++++++++ .../schemas/feature_engineering/SplitBins.yml | 56 ++ .../feature_engineering/TargetMeanEncoder.yml | 52 ++ .../TreeBasedSelection.yml | 56 ++ .../VarianceBasedSelection.yml | 52 ++ metagpt/tools/tool_registry.py | 128 ++++ metagpt/tools/tool_schema.py | 31 + metagpt/tools/tool_types.py | 43 ++ 25 files changed, 1582 insertions(+), 59 deletions(-) create mode 100644 metagpt/tools/functions/schemas/data_preprocess/FillMissingValue.yml create mode 100644 metagpt/tools/functions/schemas/data_preprocess/LabelEncode.yml create mode 100644 metagpt/tools/functions/schemas/data_preprocess/MaxAbsScale.yml create mode 100644 metagpt/tools/functions/schemas/data_preprocess/MinMaxScale.yml create mode 100644 metagpt/tools/functions/schemas/data_preprocess/OneHotEncode.yml create mode 100644 metagpt/tools/functions/schemas/data_preprocess/StandardScale.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/CatCount.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/CatCross.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/GeneralSelection.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/GroupStat.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/KFoldTargetMeanEncoder.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/PolynomialExpansion.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/SplitBins.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/TargetMeanEncoder.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/TreeBasedSelection.yml create mode 100644 metagpt/tools/functions/schemas/feature_engineering/VarianceBasedSelection.yml create mode 100644 metagpt/tools/tool_registry.py create mode 100644 metagpt/tools/tool_schema.py create mode 100644 metagpt/tools/tool_types.py diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 9104fdf82..f4ae1e572 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -8,11 +8,9 @@ import re from pathlib import Path from typing import Dict, List, Tuple, Union -import yaml from tenacity import retry, stop_after_attempt, wait_fixed from metagpt.actions import Action -from metagpt.const import TOOL_SCHEMA_PATH from metagpt.llm import LLM from metagpt.logs import logger from metagpt.prompts.ml_engineer import ( @@ -24,12 +22,9 @@ from metagpt.prompts.ml_engineer import ( TOOL_USAGE_PROMPT, ) from metagpt.schema import Message, Plan -from metagpt.tools import TOOL_TYPE_MAPPINGS +from metagpt.tools.tool_registry import TOOL_REGISTRY from metagpt.utils.common import create_func_config, remove_comments -TOOL_TYPE_MODULE = {k: v.module for k, v in TOOL_TYPE_MAPPINGS.items()} -TOOL_TYPE_USAGE_PROMPT = {k: v.usage_prompt for k, v in TOOL_TYPE_MAPPINGS.items()} - class BaseWriteAnalysisCode(Action): DEFAULT_SYSTEM_MSG: str = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt @@ -95,49 +90,27 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - schema_path: Union[Path, str] = TOOL_SCHEMA_PATH available_tools: dict = {} def __init__(self, **kwargs): super().__init__(**kwargs) - self._load_tools(self.schema_path) - def _load_tools(self, schema_path, schema_module=None): - """Load tools from yaml file""" - if isinstance(schema_path, dict): - schema_module = schema_module or "udf" - self.available_tools.update({schema_module: schema_path}) - else: - if isinstance(schema_path, list): - yml_files = schema_path - elif isinstance(schema_path, Path) and schema_path.is_file(): - yml_files = [schema_path] - else: - yml_files = schema_path.glob("*.yml") - - for yml_file in yml_files: - module = yml_file.stem - with open(yml_file, "r", encoding="utf-8") as f: - self.available_tools[module] = yaml.safe_load(f) - - def _parse_recommend_tools(self, module: str, recommend_tools: list) -> dict: + def _parse_recommend_tools(self, recommend_tools: list) -> dict: """ Parses and validates a list of recommended tools, and retrieves their schema from registry. Args: - module (str): The module name for querying tools in the registry. recommend_tools (list): A list of recommended tools. Returns: dict: A dict of valid tool schemas. """ valid_tools = [] - available_tools = self.available_tools[module].keys() - for tool in recommend_tools: - if tool in available_tools: - valid_tools.append(tool) + for tool_name in recommend_tools: + if TOOL_REGISTRY.has_tool(tool_name): + valid_tools.append(TOOL_REGISTRY.get_tool(tool_name)) - tool_catalog = {tool: self.available_tools[module][tool] for tool in valid_tools} + tool_catalog = {tool.name: tool.schema for tool in valid_tools} return tool_catalog async def _tool_recommendation( @@ -176,8 +149,10 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): tool_type = ( plan.current_task.task_type ) # find tool type from task type through exact match, can extend to retrieval in the future - available_tools = self.available_tools.get(tool_type, {}) - special_prompt = TOOL_TYPE_USAGE_PROMPT.get(tool_type, "") + available_tools = TOOL_REGISTRY.get_tools_by_type(tool_type) + special_prompt = ( + TOOL_REGISTRY.get_tool_type(tool_type).usage_prompt if TOOL_REGISTRY.has_tool_type(tool_type) else "" + ) code_steps = plan.current_task.code_steps finished_tasks = plan.get_finished_tasks() @@ -185,22 +160,17 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): code_context = "\n\n".join(code_context) tool_catalog = {} - module_name = "" - if len(available_tools) > 0: - available_tools = {k: v["description"] for k, v in available_tools.items()} + if available_tools: + available_tools = {tool_name: tool.schema["description"] for tool_name, tool in available_tools.items()} recommend_tools = await self._tool_recommendation( plan.current_task.instruction, code_steps, available_tools ) - tool_catalog = self._parse_recommend_tools(tool_type, recommend_tools) + tool_catalog = self._parse_recommend_tools(recommend_tools) logger.info(f"Recommended tools: \n{recommend_tools}") - module_name = TOOL_TYPE_MODULE[tool_type] - - tools_instruction = TOOL_USAGE_PROMPT.format( - special_prompt=special_prompt, module_name=module_name, tool_catalog=tool_catalog - ) + tools_instruction = TOOL_USAGE_PROMPT.format(special_prompt=special_prompt, tool_catalog=tool_catalog) context.append(Message(content=tools_instruction, role="user")) diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index c7ef541b9..60dcef43b 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -12,7 +12,7 @@ from metagpt.actions import Action from metagpt.logs import logger from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_CONFIG, ASSIGN_TASK_TYPE_PROMPT from metagpt.schema import Message, Plan, Task -from metagpt.tools import TOOL_TYPE_MAPPINGS +from metagpt.tools import TOOL_REGISTRY from metagpt.utils.common import CodeParser, create_func_config @@ -47,13 +47,16 @@ class WritePlan(Action): List[Dict]: tasks with task type assigned """ task_list = "\n".join([f"Task {task['task_id']}: {task['instruction']}" for task in tasks]) - task_type_desc = "\n".join([f"- **{item.name}**: {item.desc}" for item in TOOL_TYPE_MAPPINGS.values()]) + task_type_desc = "\n".join( + [f"- **{tool_type.name}**: {tool_type.desc}" for tool_type in TOOL_REGISTRY.get_tool_types().values()] + ) # task type are binded with tool type now, should be improved in the future prompt = ASSIGN_TASK_TYPE_PROMPT.format( task_list=task_list, task_type_desc=task_type_desc ) # task types are set to be the same as tool types, for now tool_config = create_func_config(ASSIGN_TASK_TYPE_CONFIG) rsp = await self.llm.aask_code(prompt, **tool_config) task_type_list = rsp["task_type"] + print(f"assigned task types: {task_type_list}") for task, task_type in zip(tasks, task_type_list): task["task_type"] = task_type return json.dumps(tasks) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 3baf79843..31d754a9e 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -203,7 +203,7 @@ Specifically, {special_prompt} - You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. # Available Tools (can be empty): -Each Class tool is described in JSON format. When you call a tool, import the tool from `{module_name}` first. +Each Class tool is described in JSON format. When you call a tool, import the tool first. {tool_catalog} # Constraints: diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index 222edf312..f743d63c7 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -8,17 +8,6 @@ from enum import Enum -from pydantic import BaseModel - -from metagpt.const import TOOL_LIBS_PATH -from metagpt.prompts.tool_type import ( - DATA_PREPROCESS_PROMPT, - FEATURE_ENGINEERING_PROMPT, - MODEL_TRAIN_PROMPT, - MODEL_EVALUATE_PROMPT, - VISION_PROMPT, -) - class SearchEngineType(Enum): SERPAPI_GOOGLE = "serpapi" diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index f423f2020..59ede3ffc 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -14,8 +14,13 @@ from sklearn.preprocessing import ( ) from metagpt.tools.functions.libs.base import MLProcess +from metagpt.tools.tool_registry import register_tool +from metagpt.tools.tool_schema import ToolTypeEnum + +TOOL_TYPE = ToolTypeEnum.DATA_PREPROCESS.value +@register_tool(tool_type_name=TOOL_TYPE) class FillMissingValue(MLProcess): def __init__( self, @@ -42,6 +47,7 @@ class FillMissingValue(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class MinMaxScale(MLProcess): def __init__( self, @@ -60,6 +66,7 @@ class MinMaxScale(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class StandardScale(MLProcess): def __init__( self, @@ -78,6 +85,7 @@ class StandardScale(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class MaxAbsScale(MLProcess): def __init__( self, @@ -96,6 +104,7 @@ class MaxAbsScale(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class RobustScale(MLProcess): def __init__( self, @@ -114,6 +123,7 @@ class RobustScale(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class OrdinalEncode(MLProcess): def __init__( self, @@ -132,6 +142,7 @@ class OrdinalEncode(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class OneHotEncode(MLProcess): def __init__( self, @@ -153,6 +164,7 @@ class OneHotEncode(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class LabelEncode(MLProcess): def __init__( self, @@ -181,6 +193,7 @@ class LabelEncode(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) def get_column_info(df: pd.DataFrame) -> dict: column_info = { "Category": [], diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index 0d9584b4a..8b96cbd07 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -6,7 +6,7 @@ # @Desc : Feature Engineering Tools import itertools -import lightgbm as lgb +# import lightgbm as lgb import numpy as np import pandas as pd from joblib import Parallel, delayed @@ -16,8 +16,13 @@ from sklearn.model_selection import KFold from sklearn.preprocessing import KBinsDiscretizer, PolynomialFeatures from metagpt.tools.functions.libs.base import MLProcess +from metagpt.tools.tool_registry import register_tool +from metagpt.tools.tool_schema import ToolTypeEnum + +TOOL_TYPE = ToolTypeEnum.FEATURE_ENGINEERING.value +@register_tool(tool_type_name=TOOL_TYPE) class PolynomialExpansion(MLProcess): def __init__(self, cols: list, degree: int = 2, label_col: str = None): self.cols = cols @@ -48,6 +53,7 @@ class PolynomialExpansion(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class CatCount(MLProcess): def __init__(self, col: str): self.col = col @@ -62,6 +68,7 @@ class CatCount(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class TargetMeanEncoder(MLProcess): def __init__(self, col: str, label: str): self.col = col @@ -77,6 +84,7 @@ class TargetMeanEncoder(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class KFoldTargetMeanEncoder(MLProcess): def __init__(self, col: str, label: str, n_splits: int = 5, random_state: int = 2021): self.col = col @@ -103,6 +111,7 @@ class KFoldTargetMeanEncoder(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class CatCross(MLProcess): def __init__(self, cols: list, max_cat_num: int = 100): self.cols = cols @@ -138,6 +147,7 @@ class CatCross(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class GroupStat(MLProcess): def __init__(self, group_col: str, agg_col: str, agg_funcs: list): self.group_col = group_col @@ -157,6 +167,7 @@ class GroupStat(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class SplitBins(MLProcess): def __init__(self, cols: list, strategy: str = "quantile"): self.cols = cols @@ -173,6 +184,7 @@ class SplitBins(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class ExtractTimeComps(MLProcess): def __init__(self, time_col: str, time_comps: list): self.time_col = time_col @@ -201,6 +213,7 @@ class ExtractTimeComps(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class GeneralSelection(MLProcess): def __init__(self, label_col: str): self.label_col = label_col @@ -228,6 +241,7 @@ class GeneralSelection(MLProcess): return new_df +# skip for now because lgb is needed class TreeBasedSelection(MLProcess): def __init__(self, label_col: str, task_type: str): self.label_col = label_col @@ -270,6 +284,7 @@ class TreeBasedSelection(MLProcess): return new_df +@register_tool(tool_type_name=TOOL_TYPE) class VarianceBasedSelection(MLProcess): def __init__(self, label_col: str, threshold: float = 0): self.label_col = label_col diff --git a/metagpt/tools/functions/schemas/data_preprocess/FillMissingValue.yml b/metagpt/tools/functions/schemas/data_preprocess/FillMissingValue.yml new file mode 100644 index 000000000..44c830a1e --- /dev/null +++ b/metagpt/tools/functions/schemas/data_preprocess/FillMissingValue.yml @@ -0,0 +1,61 @@ +FillMissingValue: + type: class + description: "Completing missing values with simple strategies" + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "columns to be processed" + strategy: + type: str + description: "the imputation strategy, notice mean/median can only be used for numeric features" + default: mean + enum: + - mean + - median + - most_frequent + - constant + fill_value: + type: int + description: "fill_value is used to replace all occurrences of missing_values" + default: null + required: + - features + fit: + description: "Fit the FillMissingValue model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." diff --git a/metagpt/tools/functions/schemas/data_preprocess/LabelEncode.yml b/metagpt/tools/functions/schemas/data_preprocess/LabelEncode.yml new file mode 100644 index 000000000..419ef60a8 --- /dev/null +++ b/metagpt/tools/functions/schemas/data_preprocess/LabelEncode.yml @@ -0,0 +1,48 @@ +LabelEncode: + type: class + description: "Apply label encoding to specified categorical columns in-place." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "Categorical columns to be label encoded" + required: + - features + fit: + description: "Fit the LabelEncode model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." diff --git a/metagpt/tools/functions/schemas/data_preprocess/MaxAbsScale.yml b/metagpt/tools/functions/schemas/data_preprocess/MaxAbsScale.yml new file mode 100644 index 000000000..3e17cfdd0 --- /dev/null +++ b/metagpt/tools/functions/schemas/data_preprocess/MaxAbsScale.yml @@ -0,0 +1,48 @@ +MaxAbsScale: + type: class + description: "cale each feature by its maximum absolute value" + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "columns to be processed" + required: + - features + fit: + description: "Fit the MaxAbsScale model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/data_preprocess/MinMaxScale.yml b/metagpt/tools/functions/schemas/data_preprocess/MinMaxScale.yml new file mode 100644 index 000000000..8f050d942 --- /dev/null +++ b/metagpt/tools/functions/schemas/data_preprocess/MinMaxScale.yml @@ -0,0 +1,48 @@ +MinMaxScale: + type: class + description: "Transform features by scaling each feature to a range, witch is (0, 1)" + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "columns to be processed" + required: + - features + fit: + description: "Fit the MinMaxScale model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." diff --git a/metagpt/tools/functions/schemas/data_preprocess/OneHotEncode.yml b/metagpt/tools/functions/schemas/data_preprocess/OneHotEncode.yml new file mode 100644 index 000000000..f499b2cb8 --- /dev/null +++ b/metagpt/tools/functions/schemas/data_preprocess/OneHotEncode.yml @@ -0,0 +1,48 @@ +OneHotEncode: + type: class + description: "Apply one-hot encoding to specified categorical columns, the original columns will be dropped." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "Categorical columns to be one-hot encoded and dropped" + required: + - features + fit: + description: "Fit the OneHotEncoding model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." diff --git a/metagpt/tools/functions/schemas/data_preprocess/StandardScale.yml b/metagpt/tools/functions/schemas/data_preprocess/StandardScale.yml new file mode 100644 index 000000000..cf6e7d57b --- /dev/null +++ b/metagpt/tools/functions/schemas/data_preprocess/StandardScale.yml @@ -0,0 +1,48 @@ +StandardScale: + type: class + description: "Standardize features by removing the mean and scaling to unit variance" + methods: + __init__: + description: "Initialize self." + parameters: + properties: + features: + type: list + description: "columns to be processed" + required: + - features + fit: + description: "Fit the StandardScale model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." diff --git a/metagpt/tools/functions/schemas/feature_engineering/CatCount.yml b/metagpt/tools/functions/schemas/feature_engineering/CatCount.yml new file mode 100644 index 000000000..049fc7879 --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/CatCount.yml @@ -0,0 +1,48 @@ +CatCount: + type: class + description: "Add value counts of a categorical column as new feature." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + col: + type: str + description: "Column for value counts." + required: + - col + fit: + description: "Fit the CatCount model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering/CatCross.yml b/metagpt/tools/functions/schemas/feature_engineering/CatCross.yml new file mode 100644 index 000000000..5d6303439 --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/CatCross.yml @@ -0,0 +1,52 @@ +CatCross: + type: class + description: "Add pairwise crossed features and convert them to numerical features." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + cols: + type: list + description: "Columns to be pairwise crossed, at least 2 columns." + max_cat_num: + type: int + description: "Maximum unique categories per crossed feature." + default: 100 + required: + - cols + fit: + description: "Fit the CatCross model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering/GeneralSelection.yml b/metagpt/tools/functions/schemas/feature_engineering/GeneralSelection.yml new file mode 100644 index 000000000..2ebf5b397 --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/GeneralSelection.yml @@ -0,0 +1,48 @@ +GeneralSelection: + type: class + description: "Drop all nan feats and feats with only one unique value." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + label_col: + type: str + description: "Label column name." + required: + - label_col + fit: + description: "Fit the GeneralSelection model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering/GroupStat.yml b/metagpt/tools/functions/schemas/feature_engineering/GroupStat.yml new file mode 100644 index 000000000..6e0ba2877 --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/GroupStat.yml @@ -0,0 +1,58 @@ +GroupStat: + type: class + description: "Aggregate specified column in a DataFrame grouped by another column, adding new features named '__by_'." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + group_col: + type: str + description: "Column used for grouping." + agg_col: + type: str + description: "Column on which aggregation is performed." + agg_funcs: + type: list + description: >- + List of aggregation functions to apply, such as ['mean', 'std']. + Each function must be supported by pandas. + required: + - group_col + - agg_col + - agg_funcs + fit: + description: "Fit the GroupStat model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering/KFoldTargetMeanEncoder.yml b/metagpt/tools/functions/schemas/feature_engineering/KFoldTargetMeanEncoder.yml new file mode 100644 index 000000000..79a673f9f --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/KFoldTargetMeanEncoder.yml @@ -0,0 +1,60 @@ +KFoldTargetMeanEncoder: + type: class + description: "Adds a new feature to the DataFrame by k-fold mean encoding of a categorical column using the label column." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + col: + type: str + description: "Column to be k-fold mean encoded." + label: + type: str + description: "Predicted label column." + n_splits: + type: int + description: "Number of splits for K-fold." + default: 5 + random_state: + type: int + description: "Random seed." + default: 2021 + required: + - col + - label + fit: + description: "Fit the KFoldTargetMeanEncoder model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering/PolynomialExpansion.yml b/metagpt/tools/functions/schemas/feature_engineering/PolynomialExpansion.yml new file mode 100644 index 000000000..62e6ad5b3 --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/PolynomialExpansion.yml @@ -0,0 +1,548 @@ +PolynomialExpansion: + type: class + description: "Add polynomial and interaction features from selected numeric columns to input DataFrame." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + cols: + type: list + description: "Columns for polynomial expansion." + label_col: + type: str + description: "Label column name." + degree: + type: int + description: "The degree of the polynomial features." + default: 2 + required: + - cols + - label_col + fit: + description: "Fit the PolynomialExpansion model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame without duplicated columns." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame without duplicated columns." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +CatCount: + type: class + description: "Add value counts of a categorical column as new feature." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + col: + type: str + description: "Column for value counts." + required: + - col + fit: + description: "Fit the CatCount model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +TargetMeanEncoder: + type: class + description: "Encodes a categorical column by the mean of the label column, and adds the result as a new feature." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + col: + type: str + description: "Column to be mean encoded." + label: + type: str + description: "Predicted label column." + required: + - col + - label + fit: + description: "Fit the TargetMeanEncoder model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +KFoldTargetMeanEncoder: + type: class + description: "Adds a new feature to the DataFrame by k-fold mean encoding of a categorical column using the label column." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + col: + type: str + description: "Column to be k-fold mean encoded." + label: + type: str + description: "Predicted label column." + n_splits: + type: int + description: "Number of splits for K-fold." + default: 5 + random_state: + type: int + description: "Random seed." + default: 2021 + required: + - col + - label + fit: + description: "Fit the KFoldTargetMeanEncoder model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +CatCross: + type: class + description: "Add pairwise crossed features and convert them to numerical features." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + cols: + type: list + description: "Columns to be pairwise crossed, at least 2 columns." + max_cat_num: + type: int + description: "Maximum unique categories per crossed feature." + default: 100 + required: + - cols + fit: + description: "Fit the CatCross model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +GroupStat: + type: class + description: "Aggregate specified column in a DataFrame grouped by another column, adding new features named '__by_'." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + group_col: + type: str + description: "Column used for grouping." + agg_col: + type: str + description: "Column on which aggregation is performed." + agg_funcs: + type: list + description: >- + List of aggregation functions to apply, such as ['mean', 'std']. + Each function must be supported by pandas. + required: + - group_col + - agg_col + - agg_funcs + fit: + description: "Fit the GroupStat model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +SplitBins: + type: class + description: "Inplace binning of continuous data into intervals, returning integer-encoded bin identifiers directly." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + cols: + type: list + description: "Columns to be binned inplace." + strategy: + type: str + description: "Strategy used to define the widths of the bins." + default: quantile + enum: + - quantile + - uniform + - kmeans + required: + - cols + fit: + description: "Fit the SplitBins model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + +GeneralSelection: + type: class + description: "Drop all nan feats and feats with only one unique value." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + label_col: + type: str + description: "Label column name." + required: + - label_col + fit: + description: "Fit the GeneralSelection model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + + +TreeBasedSelection: + type: class + description: "Select features based on tree-based model and remove features with low importance." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + label_col: + type: str + description: "Label column name." + task_type: + type: str + description: "Task type, 'cls' for classification, 'mcls' for multi-class classification, 'reg' for regression." + enum: + - cls + - mcls + - reg + required: + - label_col + - task_type + fit: + description: "Fit the TreeBasedSelection model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." + +VarianceBasedSelection: + type: class + description: "Select features based on variance and remove features with low variance." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + label_col: + type: str + description: "Label column name." + threshold: + type: float + description: "Threshold for variance." + default: 0.0 + required: + - label_col + fit: + description: "Fit the VarianceBasedSelection model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering/SplitBins.yml b/metagpt/tools/functions/schemas/feature_engineering/SplitBins.yml new file mode 100644 index 000000000..4e0171406 --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/SplitBins.yml @@ -0,0 +1,56 @@ +SplitBins: + type: class + description: "Inplace binning of continuous data into intervals, returning integer-encoded bin identifiers directly." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + cols: + type: list + description: "Columns to be binned inplace." + strategy: + type: str + description: "Strategy used to define the widths of the bins." + default: quantile + enum: + - quantile + - uniform + - kmeans + required: + - cols + fit: + description: "Fit the SplitBins model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering/TargetMeanEncoder.yml b/metagpt/tools/functions/schemas/feature_engineering/TargetMeanEncoder.yml new file mode 100644 index 000000000..86416ccbb --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/TargetMeanEncoder.yml @@ -0,0 +1,52 @@ +TargetMeanEncoder: + type: class + description: "Encodes a categorical column by the mean of the label column, and adds the result as a new feature." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + col: + type: str + description: "Column to be mean encoded." + label: + type: str + description: "Predicted label column." + required: + - col + - label + fit: + description: "Fit the TargetMeanEncoder model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering/TreeBasedSelection.yml b/metagpt/tools/functions/schemas/feature_engineering/TreeBasedSelection.yml new file mode 100644 index 000000000..c210effea --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/TreeBasedSelection.yml @@ -0,0 +1,56 @@ +TreeBasedSelection: + type: class + description: "Select features based on tree-based model and remove features with low importance." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + label_col: + type: str + description: "Label column name." + task_type: + type: str + description: "Task type, 'cls' for classification, 'mcls' for multi-class classification, 'reg' for regression." + enum: + - cls + - mcls + - reg + required: + - label_col + - task_type + fit: + description: "Fit the TreeBasedSelection model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering/VarianceBasedSelection.yml b/metagpt/tools/functions/schemas/feature_engineering/VarianceBasedSelection.yml new file mode 100644 index 000000000..6da4c3e7f --- /dev/null +++ b/metagpt/tools/functions/schemas/feature_engineering/VarianceBasedSelection.yml @@ -0,0 +1,52 @@ +VarianceBasedSelection: + type: class + description: "Select features based on variance and remove features with low variance." + methods: + __init__: + description: "Initialize self." + parameters: + properties: + label_col: + type: str + description: "Label column name." + threshold: + type: float + description: "Threshold for variance." + default: 0.0 + required: + - label_col + fit: + description: "Fit the VarianceBasedSelection model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + transform: + description: "Transform the input DataFrame with the fitted model." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." + fit_transform: + description: "Fit and transform the input DataFrame." + parameters: + properties: + df: + type: DataFrame + description: "The input DataFrame." + required: + - df + returns: + df: + type: DataFrame + description: "The transformed DataFrame contain label_col." \ No newline at end of file diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py new file mode 100644 index 000000000..201c63c71 --- /dev/null +++ b/metagpt/tools/tool_registry.py @@ -0,0 +1,128 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2023/01/12 17:07 +@Author : garylin2099 +@File : tool_registry.py +""" +import os +from collections import defaultdict +import inspect +import re + +import yaml + +from metagpt.tools.tool_schema import ToolType, ToolSchema, Tool +from metagpt.logs import logger +from metagpt.const import TOOL_SCHEMA_PATH + + +class ToolRegistry: + def __init__(self): + self.tools = {} + self.tool_types = {} + self.tools_by_types = defaultdict( + dict + ) # two-layer k-v, {tool_type_name: {tool_name: {...}, ...}, ...} + + def register_tool_type(self, tool_type: ToolType): + self.tool_types[tool_type.name] = tool_type + + def register_tool( + self, + tool_name, + tool_path, + schema_path=None, + tool_code="", + tool_type_name="other", + make_schema_if_not_exists=False, + ): + if self.has_tool(tool_name): + return + + schema_path = schema_path or TOOL_SCHEMA_PATH / tool_type_name / f"{tool_name}.yml" + + if not os.path.exists(schema_path): + if make_schema_if_not_exists: + logger.warning(f"no schema found, will make schema at {schema_path}") + make_schema(tool_code, schema_path) + else: + logger.warning(f"no schema found at assumed schema_path {schema_path}, skip registering {tool_name}") + return + + with open(schema_path, "r", encoding="utf-8") as f: + schema = yaml.safe_load(f)[tool_name] + schema["tool_path"] = tool_path # corresponding code file path of the tool + try: + ToolSchema(**schema) # validation + except Exception as e: + pass + # logger.warning( + # f"{tool_name} schema not conforms to required format, but will be used anyway. Mismatch: {e}" + # ) + tool = Tool(name=tool_name, path=tool_path, schema=schema, code=tool_code) + self.tools[tool_name] = tool + self.tools_by_types[tool_type_name][tool_name] = tool + logger.info(f"{tool_name} registered") + + def has_tool(self, key): + return key in self.tools + + def get_tool(self, key): + return self.tools.get(key) + + def get_tools_by_type(self, key): + return self.tools_by_types.get(key) + + def has_tool_type(self, key): + return key in self.tool_types + + def get_tool_type(self, key): + return self.tool_types.get(key) + + def get_tool_types(self): + return self.tool_types + + +# Registry instance +TOOL_REGISTRY = ToolRegistry() + + +def register_tool_type(cls): + """register a tool type to registry""" + TOOL_REGISTRY.register_tool_type(tool_type=cls()) + return cls + + +def register_tool(tool_name="", tool_type_name="other", schema_path=None): + """register a tool to registry""" + + def decorator(cls, tool_name=tool_name): + tool_name = tool_name or cls.__name__ + + # Get the file path where the function / class is defined and the source code + file_path = inspect.getfile(cls) + if "metagpt" in file_path: + file_path = re.search("metagpt.+", file_path).group(0) + source_code = inspect.getsource(cls) + + TOOL_REGISTRY.register_tool( + tool_name=tool_name, + tool_path=file_path, + schema_path=schema_path, + tool_code=source_code, + tool_type_name=tool_type_name, + ) + return cls + + return decorator + + +def make_schema(tool_code, path): + os.makedirs( + os.path.dirname(path), exist_ok=True + ) # Create the necessary directories + schema = {} # an empty schema for now + with open(path, "w", encoding="utf-8") as f: + yaml.dump(schema, f) + return path diff --git a/metagpt/tools/tool_schema.py b/metagpt/tools/tool_schema.py new file mode 100644 index 000000000..2b90996e5 --- /dev/null +++ b/metagpt/tools/tool_schema.py @@ -0,0 +1,31 @@ +from enum import Enum + +from pydantic import BaseModel + + +class ToolTypeEnum(Enum): + DATA_PREPROCESS = "data_preprocess" + FEATURE_ENGINEERING = "feature_engineering" + MODEL_TRAIN = "model_train" + MODEL_EVALUATE = "model_evaluate" + OTHER = "other" + + def __missing__(self, key): + return self.OTHER + + +class ToolType(BaseModel): + name: str + desc: str + usage_prompt: str = "" + + +class ToolSchema(BaseModel): + name: str + + +class Tool(BaseModel): + name: str + path: str + schema: dict = {} + code: str = "" diff --git a/metagpt/tools/tool_types.py b/metagpt/tools/tool_types.py new file mode 100644 index 000000000..9104f90b8 --- /dev/null +++ b/metagpt/tools/tool_types.py @@ -0,0 +1,43 @@ +from metagpt.prompts.tool_type import ( + DATA_PREPROCESS_PROMPT, + FEATURE_ENGINEERING_PROMPT, + MODEL_TRAIN_PROMPT, + MODEL_EVALUATE_PROMPT, +) +from metagpt.tools.tool_schema import ToolTypeEnum, ToolType +from metagpt.tools.tool_registry import register_tool_type + + +@register_tool_type +class DataPreprocess(ToolType): + name: str = ToolTypeEnum.DATA_PREPROCESS.value + desc: str = "Only for changing value inplace." + usage_prompt: str = DATA_PREPROCESS_PROMPT + + +@register_tool_type +class FeatureEngineer(ToolType): + name: str = ToolTypeEnum.FEATURE_ENGINEERING.value + desc: str = "Only for creating new columns for input data." + usage_prompt: str = FEATURE_ENGINEERING_PROMPT + + +@register_tool_type +class ModelTrain(ToolType): + name: str = ToolTypeEnum.MODEL_TRAIN.value + desc: str = "Only for training model." + usage_prompt: str = MODEL_TRAIN_PROMPT + + +@register_tool_type +class ModelEvaluate(ToolType): + name: str = ToolTypeEnum.MODEL_EVALUATE.value + desc: str = "Only for evaluating model." + usage_prompt: str = MODEL_EVALUATE_PROMPT + + +@register_tool_type +class Other(ToolType): + name: str = ToolTypeEnum.OTHER.value + desc: str = "Any tools not in the defined categories" + usage_prompt: str = "" From d7ab4d315dd1a58c696733d4912891f1fc7e58d6 Mon Sep 17 00:00:00 2001 From: yzlin Date: Sat, 13 Jan 2024 12:28:52 +0800 Subject: [PATCH 395/637] renaming and integrate sd tool, fix import issue --- metagpt/tools/__init__.py | 66 ++----------------- metagpt/tools/functions/libs/__init__.py | 7 ++ .../tools/functions/libs/data_preprocess.py | 2 +- .../functions/libs/feature_engineering.py | 2 +- metagpt/tools/sd_engine.py | 3 + .../{tool_schema.py => tool_data_type.py} | 1 + metagpt/tools/tool_registry.py | 29 ++++---- metagpt/tools/tool_types.py | 11 +++- 8 files changed, 41 insertions(+), 80 deletions(-) rename metagpt/tools/{tool_schema.py => tool_data_type.py} (92%) diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index f743d63c7..4ca46fc89 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -7,6 +7,13 @@ """ from enum import Enum +from metagpt.tools import tool_types # this registers all tool types +from metagpt.tools.functions import libs # this registers all tools +from metagpt.tools.tool_registry import TOOL_REGISTRY + +_ = tool_types # Avoid pre-commit error +_ = libs # Avoid pre-commit error +_ = TOOL_REGISTRY # Avoid pre-commit error class SearchEngineType(Enum): @@ -26,62 +33,3 @@ class WebBrowserEngineType(Enum): def __missing__(cls, key): """Default type conversion""" return cls.CUSTOM - - -class ToolType(BaseModel): - name: str - module: str = "" - desc: str - usage_prompt: str = "" - - -TOOL_TYPE_MAPPINGS = { - "data_preprocess": ToolType( - name="data_preprocess", - module=str(TOOL_LIBS_PATH / "data_preprocess"), - desc="Only for changing value inplace.", - usage_prompt=DATA_PREPROCESS_PROMPT, - ), - "feature_engineering": ToolType( - name="feature_engineering", - module=str(TOOL_LIBS_PATH / "feature_engineering"), - desc="Only for creating new columns for input data.", - usage_prompt=FEATURE_ENGINEERING_PROMPT, - ), - "model_train": ToolType( - name="model_train", - module="", - desc="Only for training model.", - usage_prompt=MODEL_TRAIN_PROMPT, - ), - "model_evaluate": ToolType( - name="model_evaluate", - module="", - desc="Only for evaluating model.", - usage_prompt=MODEL_EVALUATE_PROMPT, - ), - "stable_diffusion": ToolType( - name="stable_diffusion", - module="metagpt.tools.sd_engine", - desc="Related to text2image, image2image using stable diffusion model.", - usage_prompt="", - ), - "scrape_web": ToolType( - name="scrape_web", - module="metagpt.tools.functions.libs.scrape_web.scrape_web", - desc="Scrape data from web page.", - usage_prompt="", - ), - "vision": ToolType( - name="vision", - module=str(TOOL_LIBS_PATH / "vision"), - desc="Only for converting image into webpage code.", - usage_prompt=VISION_PROMPT, - ), - "other": ToolType( - name="other", - module="", - desc="Any tasks that do not fit into the previous categories", - usage_prompt="", - ), -} diff --git a/metagpt/tools/functions/libs/__init__.py b/metagpt/tools/functions/libs/__init__.py index a0a43f507..f0a61a7d9 100644 --- a/metagpt/tools/functions/libs/__init__.py +++ b/metagpt/tools/functions/libs/__init__.py @@ -4,3 +4,10 @@ # @Author : lidanyang # @File : __init__.py # @Desc : +from metagpt.tools.functions.libs import ( + data_preprocess, + feature_engineering, +) + +_ = data_preprocess # Avoid pre-commit error +_ = feature_engineering # Avoid pre-commit error diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/functions/libs/data_preprocess.py index 59ede3ffc..019ffd34e 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/functions/libs/data_preprocess.py @@ -14,8 +14,8 @@ from sklearn.preprocessing import ( ) from metagpt.tools.functions.libs.base import MLProcess +from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool -from metagpt.tools.tool_schema import ToolTypeEnum TOOL_TYPE = ToolTypeEnum.DATA_PREPROCESS.value diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/functions/libs/feature_engineering.py index 8b96cbd07..cd03592a6 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/functions/libs/feature_engineering.py @@ -16,8 +16,8 @@ from sklearn.model_selection import KFold from sklearn.preprocessing import KBinsDiscretizer, PolynomialFeatures from metagpt.tools.functions.libs.base import MLProcess +from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool -from metagpt.tools.tool_schema import ToolTypeEnum TOOL_TYPE = ToolTypeEnum.FEATURE_ENGINEERING.value diff --git a/metagpt/tools/sd_engine.py b/metagpt/tools/sd_engine.py index ba61fd496..2e3f36ef8 100644 --- a/metagpt/tools/sd_engine.py +++ b/metagpt/tools/sd_engine.py @@ -16,6 +16,8 @@ from PIL import Image, PngImagePlugin from metagpt.config import CONFIG from metagpt.const import SD_OUTPUT_FILE_REPO from metagpt.logs import logger +from metagpt.tools.tool_data_type import ToolTypeEnum +from metagpt.tools.tool_registry import register_tool payload = { "prompt": "", @@ -51,6 +53,7 @@ payload = { default_negative_prompt = "(easynegative:0.8),black, dark,Low resolution" +@register_tool(tool_type_name=ToolTypeEnum.STABLE_DIFFUSION) class SDEngine: def __init__(self, sd_url=""): # Initialize the SDEngine with configuration diff --git a/metagpt/tools/tool_schema.py b/metagpt/tools/tool_data_type.py similarity index 92% rename from metagpt/tools/tool_schema.py rename to metagpt/tools/tool_data_type.py index 2b90996e5..c767fef9b 100644 --- a/metagpt/tools/tool_schema.py +++ b/metagpt/tools/tool_data_type.py @@ -8,6 +8,7 @@ class ToolTypeEnum(Enum): FEATURE_ENGINEERING = "feature_engineering" MODEL_TRAIN = "model_train" MODEL_EVALUATE = "model_evaluate" + STABLE_DIFFUSION = "stable_diffusion" OTHER = "other" def __missing__(self, key): diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index 201c63c71..e6519bba9 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -5,28 +5,27 @@ @Author : garylin2099 @File : tool_registry.py """ -import os -from collections import defaultdict import inspect +import os import re +from collections import defaultdict import yaml -from metagpt.tools.tool_schema import ToolType, ToolSchema, Tool -from metagpt.logs import logger from metagpt.const import TOOL_SCHEMA_PATH +from metagpt.logs import logger +from metagpt.tools.tool_data_type import Tool, ToolSchema, ToolType class ToolRegistry: def __init__(self): self.tools = {} self.tool_types = {} - self.tools_by_types = defaultdict( - dict - ) # two-layer k-v, {tool_type_name: {tool_name: {...}, ...}, ...} + self.tools_by_types = defaultdict(dict) # two-layer k-v, {tool_type_name: {tool_name: {...}, ...}, ...} def register_tool_type(self, tool_type: ToolType): self.tool_types[tool_type.name] = tool_type + logger.info(f"{tool_type.name} registered") def register_tool( self, @@ -55,7 +54,7 @@ class ToolRegistry: schema["tool_path"] = tool_path # corresponding code file path of the tool try: ToolSchema(**schema) # validation - except Exception as e: + except Exception: pass # logger.warning( # f"{tool_name} schema not conforms to required format, but will be used anyway. Mismatch: {e}" @@ -67,19 +66,19 @@ class ToolRegistry: def has_tool(self, key): return key in self.tools - + def get_tool(self, key): return self.tools.get(key) - + def get_tools_by_type(self, key): return self.tools_by_types.get(key) - + def has_tool_type(self, key): return key in self.tool_types def get_tool_type(self, key): return self.tool_types.get(key) - + def get_tool_types(self): return self.tool_types @@ -99,7 +98,7 @@ def register_tool(tool_name="", tool_type_name="other", schema_path=None): def decorator(cls, tool_name=tool_name): tool_name = tool_name or cls.__name__ - + # Get the file path where the function / class is defined and the source code file_path = inspect.getfile(cls) if "metagpt" in file_path: @@ -119,9 +118,7 @@ def register_tool(tool_name="", tool_type_name="other", schema_path=None): def make_schema(tool_code, path): - os.makedirs( - os.path.dirname(path), exist_ok=True - ) # Create the necessary directories + os.makedirs(os.path.dirname(path), exist_ok=True) # Create the necessary directories schema = {} # an empty schema for now with open(path, "w", encoding="utf-8") as f: yaml.dump(schema, f) diff --git a/metagpt/tools/tool_types.py b/metagpt/tools/tool_types.py index 9104f90b8..97eb574da 100644 --- a/metagpt/tools/tool_types.py +++ b/metagpt/tools/tool_types.py @@ -1,10 +1,10 @@ from metagpt.prompts.tool_type import ( DATA_PREPROCESS_PROMPT, FEATURE_ENGINEERING_PROMPT, - MODEL_TRAIN_PROMPT, MODEL_EVALUATE_PROMPT, + MODEL_TRAIN_PROMPT, ) -from metagpt.tools.tool_schema import ToolTypeEnum, ToolType +from metagpt.tools.tool_data_type import ToolType, ToolTypeEnum from metagpt.tools.tool_registry import register_tool_type @@ -36,8 +36,13 @@ class ModelEvaluate(ToolType): usage_prompt: str = MODEL_EVALUATE_PROMPT +@register_tool_type +class StableDiffusion(ToolType): + name: str = ToolTypeEnum.STABLE_DIFFUSION.value + desc: str = "Related to text2image, image2image using stable diffusion model." + + @register_tool_type class Other(ToolType): name: str = ToolTypeEnum.OTHER.value desc: str = "Any tools not in the defined categories" - usage_prompt: str = "" From c8da839afe8f74a3837c49da9a332b415f7e5972 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 15 Jan 2024 11:07:29 +0800 Subject: [PATCH 396/637] moving files --- .gitignore | 1 + docs/FAQ-EN.md | 2 +- metagpt/const.py | 4 +- metagpt/prompts/ml_engineer.py | 4 +- metagpt/tools/__init__.py | 2 +- metagpt/tools/functions/__init__.py | 6 - metagpt/tools/functions/libs/base.py | 16 - metagpt/tools/functions/libs/udf/__init__.py | 126 ---- .../functions/schemas/data_preprocess.yml | 306 ---------- .../functions/schemas/feature_engineering.yml | 548 ------------------ .../tools/{functions => }/libs/__init__.py | 2 +- .../{functions => }/libs/data_preprocess.py | 13 +- .../libs/feature_engineering.py | 2 +- metagpt/tools/{ => libs}/sd_engine.py | 2 +- .../tools/{functions => }/schemas/__init__.py | 0 .../data_preprocess/FillMissingValue.yml | 0 .../schemas/data_preprocess/LabelEncode.yml | 0 .../schemas/data_preprocess/MaxAbsScale.yml | 0 .../schemas/data_preprocess/MinMaxScale.yml | 0 .../schemas/data_preprocess/OneHotEncode.yml | 0 .../schemas/data_preprocess/StandardScale.yml | 0 .../schemas/feature_engineering/CatCount.yml | 0 .../schemas/feature_engineering/CatCross.yml | 0 .../feature_engineering/GeneralSelection.yml | 0 .../schemas/feature_engineering/GroupStat.yml | 0 .../KFoldTargetMeanEncoder.yml | 0 .../PolynomialExpansion.yml | 0 .../schemas/feature_engineering/SplitBins.yml | 0 .../feature_engineering/TargetMeanEncoder.yml | 0 .../TreeBasedSelection.yml | 0 .../VarianceBasedSelection.yml | 0 .../stable_diffusion/SDEngine.yml} | 0 tests/metagpt/tools/functions/__init__.py | 6 - .../tools/{functions => }/libs/__init__.py | 0 .../libs/test_data_preprocess.py | 2 +- .../libs/test_feature_engineering.py | 3 +- .../tools/{functions => libs}/test_sd.py | 2 +- .../tools/{functions => libs}/test_udf.py | 2 +- 38 files changed, 27 insertions(+), 1022 deletions(-) delete mode 100644 metagpt/tools/functions/__init__.py delete mode 100644 metagpt/tools/functions/libs/base.py delete mode 100644 metagpt/tools/functions/libs/udf/__init__.py delete mode 100644 metagpt/tools/functions/schemas/data_preprocess.yml delete mode 100644 metagpt/tools/functions/schemas/feature_engineering.yml rename metagpt/tools/{functions => }/libs/__init__.py (86%) rename metagpt/tools/{functions => }/libs/data_preprocess.py (96%) rename metagpt/tools/{functions => }/libs/feature_engineering.py (99%) rename metagpt/tools/{ => libs}/sd_engine.py (98%) rename metagpt/tools/{functions => }/schemas/__init__.py (100%) rename metagpt/tools/{functions => }/schemas/data_preprocess/FillMissingValue.yml (100%) rename metagpt/tools/{functions => }/schemas/data_preprocess/LabelEncode.yml (100%) rename metagpt/tools/{functions => }/schemas/data_preprocess/MaxAbsScale.yml (100%) rename metagpt/tools/{functions => }/schemas/data_preprocess/MinMaxScale.yml (100%) rename metagpt/tools/{functions => }/schemas/data_preprocess/OneHotEncode.yml (100%) rename metagpt/tools/{functions => }/schemas/data_preprocess/StandardScale.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/CatCount.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/CatCross.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/GeneralSelection.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/GroupStat.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/KFoldTargetMeanEncoder.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/PolynomialExpansion.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/SplitBins.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/TargetMeanEncoder.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/TreeBasedSelection.yml (100%) rename metagpt/tools/{functions => }/schemas/feature_engineering/VarianceBasedSelection.yml (100%) rename metagpt/tools/{functions/schemas/stable_diffusion.yml => schemas/stable_diffusion/SDEngine.yml} (100%) delete mode 100644 tests/metagpt/tools/functions/__init__.py rename tests/metagpt/tools/{functions => }/libs/__init__.py (100%) rename tests/metagpt/tools/{functions => }/libs/test_data_preprocess.py (97%) rename tests/metagpt/tools/{functions => }/libs/test_feature_engineering.py (97%) rename tests/metagpt/tools/{functions => libs}/test_sd.py (93%) rename tests/metagpt/tools/{functions => libs}/test_udf.py (95%) diff --git a/.gitignore b/.gitignore index 87c7b3120..a69b3b1c2 100644 --- a/.gitignore +++ b/.gitignore @@ -173,6 +173,7 @@ tests/metagpt/utils/file_repo_git *.png htmlcov htmlcov.* +cov.xml *.dot *.pkl *-structure.csv diff --git a/docs/FAQ-EN.md b/docs/FAQ-EN.md index d4a9f6097..145d27be9 100644 --- a/docs/FAQ-EN.md +++ b/docs/FAQ-EN.md @@ -130,7 +130,7 @@ 1. HTML Layout: Outputs the HTML code for the page. 1. CSS Styles (styles.css): Outputs the CSS code for the page. - 1. Currently, the SD skill is a tool invoked by UIDesign. It instantiates the SDEngine, with specific code found in metagpt/tools/sd_engine. + 1. Currently, the SD skill is a tool invoked by UIDesign. It instantiates the SDEngine, with specific code found in metagpt/tools/libs/sd_engine.py. 1. Configuration instructions for SD Skills: The SD interface is currently deployed based on *https://github.com/AUTOMATIC1111/stable-diffusion-webui* **For environmental configurations and model downloads, please refer to the aforementioned GitHub repository. To initiate the SD service that supports API calls, run the command specified in cmd with the parameter nowebui, i.e., diff --git a/metagpt/const.py b/metagpt/const.py index a57464a19..7a19e81d0 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -70,8 +70,8 @@ TMP = METAGPT_ROOT / "tmp" SOURCE_ROOT = METAGPT_ROOT / "metagpt" PROMPT_PATH = SOURCE_ROOT / "prompts" SKILL_DIRECTORY = SOURCE_ROOT / "skills" -TOOL_SCHEMA_PATH = METAGPT_ROOT / "metagpt/tools/functions/schemas" -TOOL_LIBS_PATH = METAGPT_ROOT / "metagpt/tools/functions/libs" +TOOL_SCHEMA_PATH = METAGPT_ROOT / "metagpt/tools/schemas" +TOOL_LIBS_PATH = METAGPT_ROOT / "metagpt/tools/libs" # REAL CONSTS diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 31d754a9e..ff29d5ed4 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -15,7 +15,7 @@ Keep dataset column information updated before model train. # Task Update and print the dataset's column information only if the train or test data has changed. Use the following code: ```python -from metagpt.tools.functions.libs.data_preprocess import get_column_info +from metagpt.tools.libs.data_preprocess import get_column_info column_info = get_column_info(df) print("column_info") @@ -248,7 +248,7 @@ when current task is "do data preprocess, like fill missing value, handle outlie ```python # Step 1: fill missing value # Tools used: ['FillMissingValue'] -from metagpt.tools.functions.libs.data_preprocess import FillMissingValue +from metagpt.tools.libs.data_preprocess import FillMissingValue train_processed = train.copy() test_processed = test.copy() diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index 4ca46fc89..23b51533d 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -8,7 +8,7 @@ from enum import Enum from metagpt.tools import tool_types # this registers all tool types -from metagpt.tools.functions import libs # this registers all tools +from metagpt.tools import libs # this registers all tools from metagpt.tools.tool_registry import TOOL_REGISTRY _ = tool_types # Avoid pre-commit error diff --git a/metagpt/tools/functions/__init__.py b/metagpt/tools/functions/__init__.py deleted file mode 100644 index a0a43f507..000000000 --- a/metagpt/tools/functions/__init__.py +++ /dev/null @@ -1,6 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/16 16:32 -# @Author : lidanyang -# @File : __init__.py -# @Desc : diff --git a/metagpt/tools/functions/libs/base.py b/metagpt/tools/functions/libs/base.py deleted file mode 100644 index c39adc66b..000000000 --- a/metagpt/tools/functions/libs/base.py +++ /dev/null @@ -1,16 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/12/10 20:12 -# @Author : lidanyang -# @File : base -# @Desc : -class MLProcess(object): - def fit(self, df): - raise NotImplementedError - - def transform(self, df): - raise NotImplementedError - - def fit_transform(self, df): - self.fit(df) - return self.transform(df) diff --git a/metagpt/tools/functions/libs/udf/__init__.py b/metagpt/tools/functions/libs/udf/__init__.py deleted file mode 100644 index 6644565d7..000000000 --- a/metagpt/tools/functions/libs/udf/__init__.py +++ /dev/null @@ -1,126 +0,0 @@ -import ast -import os -import re -import yaml -import inspect -import importlib -from pathlib import Path -from typing import List -from metagpt.logs import logger - - -def extract_function_signatures(file_path): - with open(file_path, "r", encoding="utf-8") as file: - source_code = file.read() - - tree = ast.parse(source_code) - function_signatures = [] - function_returns = [] - for node in ast.walk(tree): - if isinstance(node, ast.FunctionDef): - # 只提取用户自定义函数,排除内置函数 - if not (node.name.startswith("__") and node.name.endswith("__")): - # 获取函数名 - function_name = node.name - # 获取参数列表 - args = [arg.arg for arg in node.args.args] - # 获取函数签名 - function_signature = f"{function_name}({', '.join(args)})" - # 导入函数 - module_name = Path(file_path).parts[-1][: -len(Path(file_path).suffix)] - module = importlib.import_module(f"metagpt.tools.functions.libs.udf.{module_name}") - # 将函数导入到当前命名空间 - globals().update({function_name: getattr(module, function_name)}) - # 获取函数注释和函数路径 - function_schema = { - "udf_name": function_signature, - "udf_path": f"from metagpt.tools.functions.libs.udf.{module_name} import {function_name}", - "udf_doc": inspect.getdoc(getattr(module, function_name)), - } - function_signatures.append(function_schema) - # 获取函数返回变量名 - source_lines, _ = inspect.getsourcelines(getattr(module, function_name)) - for line in source_lines: - if line.strip().startswith("return "): - function_returns.append( - { - "udf_name": function_name, - "udf_returns": [var.strip() for var in line.strip()[len("return ") :].split(",")], - } - ) - break - - # 没有返回值的函数 - if not function_returns or function_returns[-1]["udf_name"] != function_name: - function_returns.append({"udf_name": function_name, "udf_returns": [None]}) - return function_signatures, function_returns - - -def get_function_signatures_in_folder(folder_path): - python_files = [f for f in os.listdir(folder_path) if f.endswith(".py") and f != "__init__.py"] - all_function_signatures = [] - all_function_returns = [] - - for file_name in python_files: - file_path = os.path.join(folder_path, file_name) - function_signatures, function_returns = extract_function_signatures(file_path) - all_function_signatures.extend(function_signatures) - all_function_returns.extend(function_returns) - return all_function_signatures, all_function_returns - - -# Create Tools Yaml Style Schema -def docstring_to_yaml(docstring: str, return_vars: List[str] = None): - logger.debug(f"\n\nFunction Docstring: \n{'-'*60}\n {docstring} \n\nFunction Returns: \n{'-'*60}\n{return_vars}\n") - if docstring is None: - return {} - # 匹配简介部分 - description_match = re.search(r"^(.*?)(?:Args:|Returns:|Raises:|$)", docstring, re.DOTALL) - description = description_match.group(1).strip() if description_match else "" - - # 匹配Args部分 - args_match = re.search(r"Args:\s*(.*?)(?:Returns:|Raises:|$)", docstring, re.DOTALL) - _args = args_match.group(1).strip() if args_match else "" - variable_pattern = re.compile(r"(\w+)\s*\((.*?)\):\s*(.*)") - params = variable_pattern.findall(_args) - if not params: - params = ((None, None, None),) - # 匹配Returns部分 - returns_match = re.search(r"Returns:\s*(.*?)(?:Raises:|$)", docstring, re.DOTALL) - returns = returns_match.group(1).strip() if returns_match else "" - return_pattern = re.compile(r"^(.*)\s*:\s*(.*)$") - # 添加返回值变量名 - return_vars = return_vars if isinstance(return_vars, list) else [return_vars] - returns = [(r, *r_desc) for r_desc, r in zip(return_pattern.findall(returns), return_vars)] - # 构建YAML字典 - yaml_data = { - "description": description.strip(".").strip(), - "parameters": { - "properties": { - param[0]: {"type": param[1], "description": param[2]} for param in params if param[0] is not None - }, - "required": [param[0] for param in params if param[0] is not None], - }, - "returns": {ret[0]: {"type": ret[1], "description": ret[2]} for ret in returns}, - } - return yaml_data - - -def extract_function_schema_yaml_in_folder(folder_path: str): - function_signatures, function_returns = get_function_signatures_in_folder(folder_path) - function_schema_yaml_data = {} - for func_docstring, func_returns in zip(function_signatures, function_returns): - if func_docstring["udf_doc"]: - fun_yaml_data = docstring_to_yaml(func_docstring["udf_doc"], func_returns["udf_returns"]) - fun_yaml_data.update({"type": "function"}) - function_schema_yaml_data.update({func_returns["udf_name"]: fun_yaml_data}) - return yaml.dump(function_schema_yaml_data, default_flow_style=False) - - -folder_path = str(Path(__file__).parent.absolute()) -function_signatures, function_returns = get_function_signatures_in_folder(folder_path) - -UDFS = [func for func in function_signatures] - -UDFS_YAML_STR: str = extract_function_schema_yaml_in_folder(folder_path) -UDFS_YAML: dict = yaml.load(UDFS_YAML_STR, Loader=yaml.FullLoader) diff --git a/metagpt/tools/functions/schemas/data_preprocess.yml b/metagpt/tools/functions/schemas/data_preprocess.yml deleted file mode 100644 index 4de697abd..000000000 --- a/metagpt/tools/functions/schemas/data_preprocess.yml +++ /dev/null @@ -1,306 +0,0 @@ -FillMissingValue: - type: class - description: "Completing missing values with simple strategies" - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "columns to be processed" - strategy: - type: str - description: "the imputation strategy, notice mean/median can only be used for numeric features" - default: mean - enum: - - mean - - median - - most_frequent - - constant - fill_value: - type: int - description: "fill_value is used to replace all occurrences of missing_values" - default: null - required: - - features - fit: - description: "Fit the FillMissingValue model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -MinMaxScale: - type: class - description: "Transform features by scaling each feature to a range, witch is (0, 1)" - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "columns to be processed" - required: - - features - fit: - description: "Fit the MinMaxScale model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -StandardScale: - type: class - description: "Standardize features by removing the mean and scaling to unit variance" - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "columns to be processed" - required: - - features - fit: - description: "Fit the StandardScale model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -MaxAbsScale: - type: class - description: "cale each feature by its maximum absolute value" - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "columns to be processed" - required: - - features - fit: - description: "Fit the MaxAbsScale model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -LabelEncode: - type: class - description: "Apply label encoding to specified categorical columns in-place." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "Categorical columns to be label encoded" - required: - - features - fit: - description: "Fit the LabelEncode model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -OneHotEncode: - type: class - description: "Apply one-hot encoding to specified categorical columns, the original columns will be dropped." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "Categorical columns to be one-hot encoded and dropped" - required: - - features - fit: - description: "Fit the OneHotEncoding model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/functions/schemas/feature_engineering.yml b/metagpt/tools/functions/schemas/feature_engineering.yml deleted file mode 100644 index 62e6ad5b3..000000000 --- a/metagpt/tools/functions/schemas/feature_engineering.yml +++ /dev/null @@ -1,548 +0,0 @@ -PolynomialExpansion: - type: class - description: "Add polynomial and interaction features from selected numeric columns to input DataFrame." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - cols: - type: list - description: "Columns for polynomial expansion." - label_col: - type: str - description: "Label column name." - degree: - type: int - description: "The degree of the polynomial features." - default: 2 - required: - - cols - - label_col - fit: - description: "Fit the PolynomialExpansion model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame without duplicated columns." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame without duplicated columns." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -CatCount: - type: class - description: "Add value counts of a categorical column as new feature." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - col: - type: str - description: "Column for value counts." - required: - - col - fit: - description: "Fit the CatCount model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -TargetMeanEncoder: - type: class - description: "Encodes a categorical column by the mean of the label column, and adds the result as a new feature." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - col: - type: str - description: "Column to be mean encoded." - label: - type: str - description: "Predicted label column." - required: - - col - - label - fit: - description: "Fit the TargetMeanEncoder model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -KFoldTargetMeanEncoder: - type: class - description: "Adds a new feature to the DataFrame by k-fold mean encoding of a categorical column using the label column." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - col: - type: str - description: "Column to be k-fold mean encoded." - label: - type: str - description: "Predicted label column." - n_splits: - type: int - description: "Number of splits for K-fold." - default: 5 - random_state: - type: int - description: "Random seed." - default: 2021 - required: - - col - - label - fit: - description: "Fit the KFoldTargetMeanEncoder model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -CatCross: - type: class - description: "Add pairwise crossed features and convert them to numerical features." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - cols: - type: list - description: "Columns to be pairwise crossed, at least 2 columns." - max_cat_num: - type: int - description: "Maximum unique categories per crossed feature." - default: 100 - required: - - cols - fit: - description: "Fit the CatCross model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -GroupStat: - type: class - description: "Aggregate specified column in a DataFrame grouped by another column, adding new features named '__by_'." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - group_col: - type: str - description: "Column used for grouping." - agg_col: - type: str - description: "Column on which aggregation is performed." - agg_funcs: - type: list - description: >- - List of aggregation functions to apply, such as ['mean', 'std']. - Each function must be supported by pandas. - required: - - group_col - - agg_col - - agg_funcs - fit: - description: "Fit the GroupStat model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -SplitBins: - type: class - description: "Inplace binning of continuous data into intervals, returning integer-encoded bin identifiers directly." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - cols: - type: list - description: "Columns to be binned inplace." - strategy: - type: str - description: "Strategy used to define the widths of the bins." - default: quantile - enum: - - quantile - - uniform - - kmeans - required: - - cols - fit: - description: "Fit the SplitBins model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -GeneralSelection: - type: class - description: "Drop all nan feats and feats with only one unique value." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - label_col: - type: str - description: "Label column name." - required: - - label_col - fit: - description: "Fit the GeneralSelection model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - - -TreeBasedSelection: - type: class - description: "Select features based on tree-based model and remove features with low importance." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - label_col: - type: str - description: "Label column name." - task_type: - type: str - description: "Task type, 'cls' for classification, 'mcls' for multi-class classification, 'reg' for regression." - enum: - - cls - - mcls - - reg - required: - - label_col - - task_type - fit: - description: "Fit the TreeBasedSelection model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." - -VarianceBasedSelection: - type: class - description: "Select features based on variance and remove features with low variance." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - label_col: - type: str - description: "Label column name." - threshold: - type: float - description: "Threshold for variance." - default: 0.0 - required: - - label_col - fit: - description: "Fit the VarianceBasedSelection model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." \ No newline at end of file diff --git a/metagpt/tools/functions/libs/__init__.py b/metagpt/tools/libs/__init__.py similarity index 86% rename from metagpt/tools/functions/libs/__init__.py rename to metagpt/tools/libs/__init__.py index f0a61a7d9..3d74674aa 100644 --- a/metagpt/tools/functions/libs/__init__.py +++ b/metagpt/tools/libs/__init__.py @@ -4,7 +4,7 @@ # @Author : lidanyang # @File : __init__.py # @Desc : -from metagpt.tools.functions.libs import ( +from metagpt.tools.libs import ( data_preprocess, feature_engineering, ) diff --git a/metagpt/tools/functions/libs/data_preprocess.py b/metagpt/tools/libs/data_preprocess.py similarity index 96% rename from metagpt/tools/functions/libs/data_preprocess.py rename to metagpt/tools/libs/data_preprocess.py index 019ffd34e..7cc44263d 100644 --- a/metagpt/tools/functions/libs/data_preprocess.py +++ b/metagpt/tools/libs/data_preprocess.py @@ -13,13 +13,24 @@ from sklearn.preprocessing import ( StandardScaler, ) -from metagpt.tools.functions.libs.base import MLProcess from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool TOOL_TYPE = ToolTypeEnum.DATA_PREPROCESS.value +class MLProcess(object): + def fit(self, df): + raise NotImplementedError + + def transform(self, df): + raise NotImplementedError + + def fit_transform(self, df): + self.fit(df) + return self.transform(df) + + @register_tool(tool_type_name=TOOL_TYPE) class FillMissingValue(MLProcess): def __init__( diff --git a/metagpt/tools/functions/libs/feature_engineering.py b/metagpt/tools/libs/feature_engineering.py similarity index 99% rename from metagpt/tools/functions/libs/feature_engineering.py rename to metagpt/tools/libs/feature_engineering.py index cd03592a6..ed5c1be72 100644 --- a/metagpt/tools/functions/libs/feature_engineering.py +++ b/metagpt/tools/libs/feature_engineering.py @@ -15,7 +15,7 @@ from sklearn.feature_selection import VarianceThreshold from sklearn.model_selection import KFold from sklearn.preprocessing import KBinsDiscretizer, PolynomialFeatures -from metagpt.tools.functions.libs.base import MLProcess +from metagpt.tools.libs.data_preprocess import MLProcess from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool diff --git a/metagpt/tools/sd_engine.py b/metagpt/tools/libs/sd_engine.py similarity index 98% rename from metagpt/tools/sd_engine.py rename to metagpt/tools/libs/sd_engine.py index 2e3f36ef8..ad63c2505 100644 --- a/metagpt/tools/sd_engine.py +++ b/metagpt/tools/libs/sd_engine.py @@ -53,7 +53,7 @@ payload = { default_negative_prompt = "(easynegative:0.8),black, dark,Low resolution" -@register_tool(tool_type_name=ToolTypeEnum.STABLE_DIFFUSION) +@register_tool(tool_type_name=ToolTypeEnum.STABLE_DIFFUSION.value) class SDEngine: def __init__(self, sd_url=""): # Initialize the SDEngine with configuration diff --git a/metagpt/tools/functions/schemas/__init__.py b/metagpt/tools/schemas/__init__.py similarity index 100% rename from metagpt/tools/functions/schemas/__init__.py rename to metagpt/tools/schemas/__init__.py diff --git a/metagpt/tools/functions/schemas/data_preprocess/FillMissingValue.yml b/metagpt/tools/schemas/data_preprocess/FillMissingValue.yml similarity index 100% rename from metagpt/tools/functions/schemas/data_preprocess/FillMissingValue.yml rename to metagpt/tools/schemas/data_preprocess/FillMissingValue.yml diff --git a/metagpt/tools/functions/schemas/data_preprocess/LabelEncode.yml b/metagpt/tools/schemas/data_preprocess/LabelEncode.yml similarity index 100% rename from metagpt/tools/functions/schemas/data_preprocess/LabelEncode.yml rename to metagpt/tools/schemas/data_preprocess/LabelEncode.yml diff --git a/metagpt/tools/functions/schemas/data_preprocess/MaxAbsScale.yml b/metagpt/tools/schemas/data_preprocess/MaxAbsScale.yml similarity index 100% rename from metagpt/tools/functions/schemas/data_preprocess/MaxAbsScale.yml rename to metagpt/tools/schemas/data_preprocess/MaxAbsScale.yml diff --git a/metagpt/tools/functions/schemas/data_preprocess/MinMaxScale.yml b/metagpt/tools/schemas/data_preprocess/MinMaxScale.yml similarity index 100% rename from metagpt/tools/functions/schemas/data_preprocess/MinMaxScale.yml rename to metagpt/tools/schemas/data_preprocess/MinMaxScale.yml diff --git a/metagpt/tools/functions/schemas/data_preprocess/OneHotEncode.yml b/metagpt/tools/schemas/data_preprocess/OneHotEncode.yml similarity index 100% rename from metagpt/tools/functions/schemas/data_preprocess/OneHotEncode.yml rename to metagpt/tools/schemas/data_preprocess/OneHotEncode.yml diff --git a/metagpt/tools/functions/schemas/data_preprocess/StandardScale.yml b/metagpt/tools/schemas/data_preprocess/StandardScale.yml similarity index 100% rename from metagpt/tools/functions/schemas/data_preprocess/StandardScale.yml rename to metagpt/tools/schemas/data_preprocess/StandardScale.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/CatCount.yml b/metagpt/tools/schemas/feature_engineering/CatCount.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/CatCount.yml rename to metagpt/tools/schemas/feature_engineering/CatCount.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/CatCross.yml b/metagpt/tools/schemas/feature_engineering/CatCross.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/CatCross.yml rename to metagpt/tools/schemas/feature_engineering/CatCross.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/GeneralSelection.yml b/metagpt/tools/schemas/feature_engineering/GeneralSelection.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/GeneralSelection.yml rename to metagpt/tools/schemas/feature_engineering/GeneralSelection.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/GroupStat.yml b/metagpt/tools/schemas/feature_engineering/GroupStat.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/GroupStat.yml rename to metagpt/tools/schemas/feature_engineering/GroupStat.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/KFoldTargetMeanEncoder.yml b/metagpt/tools/schemas/feature_engineering/KFoldTargetMeanEncoder.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/KFoldTargetMeanEncoder.yml rename to metagpt/tools/schemas/feature_engineering/KFoldTargetMeanEncoder.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/PolynomialExpansion.yml b/metagpt/tools/schemas/feature_engineering/PolynomialExpansion.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/PolynomialExpansion.yml rename to metagpt/tools/schemas/feature_engineering/PolynomialExpansion.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/SplitBins.yml b/metagpt/tools/schemas/feature_engineering/SplitBins.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/SplitBins.yml rename to metagpt/tools/schemas/feature_engineering/SplitBins.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/TargetMeanEncoder.yml b/metagpt/tools/schemas/feature_engineering/TargetMeanEncoder.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/TargetMeanEncoder.yml rename to metagpt/tools/schemas/feature_engineering/TargetMeanEncoder.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/TreeBasedSelection.yml b/metagpt/tools/schemas/feature_engineering/TreeBasedSelection.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/TreeBasedSelection.yml rename to metagpt/tools/schemas/feature_engineering/TreeBasedSelection.yml diff --git a/metagpt/tools/functions/schemas/feature_engineering/VarianceBasedSelection.yml b/metagpt/tools/schemas/feature_engineering/VarianceBasedSelection.yml similarity index 100% rename from metagpt/tools/functions/schemas/feature_engineering/VarianceBasedSelection.yml rename to metagpt/tools/schemas/feature_engineering/VarianceBasedSelection.yml diff --git a/metagpt/tools/functions/schemas/stable_diffusion.yml b/metagpt/tools/schemas/stable_diffusion/SDEngine.yml similarity index 100% rename from metagpt/tools/functions/schemas/stable_diffusion.yml rename to metagpt/tools/schemas/stable_diffusion/SDEngine.yml diff --git a/tests/metagpt/tools/functions/__init__.py b/tests/metagpt/tools/functions/__init__.py deleted file mode 100644 index 7d36f3404..000000000 --- a/tests/metagpt/tools/functions/__init__.py +++ /dev/null @@ -1,6 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/17 10:24 -# @Author : lidanyang -# @File : __init__.py -# @Desc : diff --git a/tests/metagpt/tools/functions/libs/__init__.py b/tests/metagpt/tools/libs/__init__.py similarity index 100% rename from tests/metagpt/tools/functions/libs/__init__.py rename to tests/metagpt/tools/libs/__init__.py diff --git a/tests/metagpt/tools/functions/libs/test_data_preprocess.py b/tests/metagpt/tools/libs/test_data_preprocess.py similarity index 97% rename from tests/metagpt/tools/functions/libs/test_data_preprocess.py rename to tests/metagpt/tools/libs/test_data_preprocess.py index 3c2d661ab..418f8adee 100644 --- a/tests/metagpt/tools/functions/libs/test_data_preprocess.py +++ b/tests/metagpt/tools/libs/test_data_preprocess.py @@ -5,7 +5,7 @@ import numpy.testing as npt import pandas as pd import pytest -from metagpt.tools.functions.libs.data_preprocess import ( +from metagpt.tools.libs.data_preprocess import ( FillMissingValue, LabelEncode, MaxAbsScale, diff --git a/tests/metagpt/tools/functions/libs/test_feature_engineering.py b/tests/metagpt/tools/libs/test_feature_engineering.py similarity index 97% rename from tests/metagpt/tools/functions/libs/test_feature_engineering.py rename to tests/metagpt/tools/libs/test_feature_engineering.py index 5b45aeb0c..3cfd5dacd 100644 --- a/tests/metagpt/tools/functions/libs/test_feature_engineering.py +++ b/tests/metagpt/tools/libs/test_feature_engineering.py @@ -3,7 +3,7 @@ import pandas as pd import pytest from sklearn.datasets import fetch_california_housing, load_breast_cancer, load_iris -from metagpt.tools.functions.libs.feature_engineering import ( +from metagpt.tools.libs.feature_engineering import ( CatCount, CatCross, ExtractTimeComps, @@ -147,6 +147,7 @@ def test_general_selection(mock_dataset): assert "cat2" not in transformed.columns +@pytest.mark.skip # skip because TreeBasedSelection needs lgb as dependency def test_tree_based_selection(mock_dataset): # regression data = load_sklearn_data("housing") diff --git a/tests/metagpt/tools/functions/test_sd.py b/tests/metagpt/tools/libs/test_sd.py similarity index 93% rename from tests/metagpt/tools/functions/test_sd.py rename to tests/metagpt/tools/libs/test_sd.py index 142101cad..363cf96b9 100644 --- a/tests/metagpt/tools/functions/test_sd.py +++ b/tests/metagpt/tools/libs/test_sd.py @@ -4,7 +4,7 @@ # @Desc : import pytest -from metagpt.tools.sd_engine import SDEngine +from metagpt.tools.libs.sd_engine import SDEngine def test_sd_tools(): diff --git a/tests/metagpt/tools/functions/test_udf.py b/tests/metagpt/tools/libs/test_udf.py similarity index 95% rename from tests/metagpt/tools/functions/test_udf.py rename to tests/metagpt/tools/libs/test_udf.py index 741bd9a9f..19e523448 100644 --- a/tests/metagpt/tools/functions/test_udf.py +++ b/tests/metagpt/tools/libs/test_udf.py @@ -3,7 +3,7 @@ import json import yaml from metagpt.logs import logger -from metagpt.tools.functions.libs.udf import UDFS, UDFS_YAML, docstring_to_yaml +from metagpt.tools.libs.udf import UDFS, UDFS_YAML, docstring_to_yaml def test_udfs(): From 638dda31cf0c3d1b2fc3834174cd80b3c086abab Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 15 Jan 2024 11:58:07 +0800 Subject: [PATCH 397/637] add unit tests for tool registry --- metagpt/tools/tool_registry.py | 3 +- tests/metagpt/tools/test_tool_registry.py | 101 ++++++++++++++++++++++ 2 files changed, 103 insertions(+), 1 deletion(-) create mode 100644 tests/metagpt/tools/test_tool_registry.py diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index e6519bba9..2c59cd198 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -50,7 +50,8 @@ class ToolRegistry: return with open(schema_path, "r", encoding="utf-8") as f: - schema = yaml.safe_load(f)[tool_name] + schema_dict = yaml.safe_load(f) + schema = schema_dict.get(tool_name) or dict(schema_dict.values()) schema["tool_path"] = tool_path # corresponding code file path of the tool try: ToolSchema(**schema) # validation diff --git a/tests/metagpt/tools/test_tool_registry.py b/tests/metagpt/tools/test_tool_registry.py new file mode 100644 index 000000000..fd758b141 --- /dev/null +++ b/tests/metagpt/tools/test_tool_registry.py @@ -0,0 +1,101 @@ +import pytest + +from metagpt.tools.tool_registry import ToolRegistry +from metagpt.tools.tool_types import ToolType + + +@pytest.fixture +def tool_registry(): + return ToolRegistry() + + +@pytest.fixture +def schema_yaml(mocker): + mock_yaml_content = """ + tool_name: + key1: value1 + key2: value2 + """ + mocker.patch("os.path.exists", return_value=True) + mocker.patch("builtins.open", mocker.mock_open(read_data=mock_yaml_content)) + return mocker + + +# Test Initialization +def test_initialization(tool_registry): + assert isinstance(tool_registry, ToolRegistry) + assert tool_registry.tools == {} + assert tool_registry.tool_types == {} + assert tool_registry.tools_by_types == {} + + +# Test Tool Type Registration +def test_register_tool_type(tool_registry): + tool_type = ToolType(name="TestType", desc="test") + tool_registry.register_tool_type(tool_type) + assert "TestType" in tool_registry.tool_types + + +# Test Tool Registration +def test_register_tool(tool_registry, schema_yaml): + tool_registry.register_tool("TestTool", "/path/to/tool") + assert "TestTool" in tool_registry.tools + + +# Test Tool Registration with Non-existing Schema +def test_register_tool_no_schema(tool_registry, mocker): + mocker.patch("os.path.exists", return_value=False) + tool_registry.register_tool("TestTool", "/path/to/tool") + assert "TestTool" not in tool_registry.tools + + +# Test Tool Existence Checks +def test_has_tool(tool_registry, schema_yaml): + tool_registry.register_tool("TestTool", "/path/to/tool") + assert tool_registry.has_tool("TestTool") + assert not tool_registry.has_tool("NonexistentTool") + + +# Test Tool Retrieval +def test_get_tool(tool_registry, schema_yaml): + tool_registry.register_tool("TestTool", "/path/to/tool") + tool = tool_registry.get_tool("TestTool") + assert tool is not None + assert tool.name == "TestTool" + assert tool.path == "/path/to/tool" + + +# Similar tests for has_tool_type, get_tool_type, get_tools_by_type +def test_has_tool_type(tool_registry): + tool_type = ToolType(name="TestType", desc="test") + tool_registry.register_tool_type(tool_type) + assert tool_registry.has_tool_type("TestType") + assert not tool_registry.has_tool_type("NonexistentType") + + +def test_get_tool_type(tool_registry): + tool_type = ToolType(name="TestType", desc="test") + tool_registry.register_tool_type(tool_type) + retrieved_type = tool_registry.get_tool_type("TestType") + assert retrieved_type is not None + assert retrieved_type.name == "TestType" + + +def test_get_tools_by_type(tool_registry, schema_yaml): + tool_type_name = "TestType" + tool_name = "TestTool" + tool_path = "/path/to/tool" + tool_type = ToolType(name=tool_type_name, desc="test") + tool_registry.register_tool_type(tool_type) + + tool_registry.register_tool(tool_name, tool_path, tool_type_name=tool_type_name) + + tools_by_type = tool_registry.get_tools_by_type(tool_type_name) + assert tools_by_type is not None + assert tool_name in tools_by_type + + +# Test case for when the tool type does not exist +def test_get_tools_by_nonexistent_type(tool_registry): + tools_by_type = tool_registry.get_tools_by_type("NonexistentType") + assert tools_by_type is None From 8a14dde219f8ec03531c21f0f62c75bcc680ae60 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 16 Jan 2024 15:46:13 +0800 Subject: [PATCH 398/637] tool_type renaming --- metagpt/prompts/{tool_type.py => tool_types.py} | 0 metagpt/roles/code_interpreter.py | 7 +++++++ metagpt/tools/tool_types.py | 2 +- 3 files changed, 8 insertions(+), 1 deletion(-) rename metagpt/prompts/{tool_type.py => tool_types.py} (100%) diff --git a/metagpt/prompts/tool_type.py b/metagpt/prompts/tool_types.py similarity index 100% rename from metagpt/prompts/tool_type.py rename to metagpt/prompts/tool_types.py diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index afd51a575..46cc00d5e 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -5,6 +5,7 @@ from pydantic import Field from metagpt.actions.ask_review import ReviewConst from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.logs import logger from metagpt.roles import Role from metagpt.roles.tool_maker import ToolMaker @@ -16,6 +17,7 @@ class CodeInterpreter(Role): auto_run: bool = True use_tools: bool = False make_udfs: bool = False # whether to save user-defined functions + use_code_steps: bool = False execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) def __init__( @@ -56,6 +58,10 @@ class CodeInterpreter(Role): return task_result async def _write_and_exec_code(self, max_retry: int = 3): + self.planner.current_task.code_steps = ( + await WriteCodeSteps().run(self.planner.plan) if self.use_code_steps else "" + ) + counter = 0 success = False @@ -90,6 +96,7 @@ class CodeInterpreter(Role): logger.info(f"ready to {todo.name}") context = self.planner.get_useful_memories() + # print(*context, sep="\n***\n") code = await todo.run(context=context, plan=self.planner.plan, temperature=0.0) # 暂时在这里转换 WriteCodeWithTools 的输出 if isinstance(code, str): diff --git a/metagpt/tools/tool_types.py b/metagpt/tools/tool_types.py index 97eb574da..289271985 100644 --- a/metagpt/tools/tool_types.py +++ b/metagpt/tools/tool_types.py @@ -1,4 +1,4 @@ -from metagpt.prompts.tool_type import ( +from metagpt.prompts.tool_types import ( DATA_PREPROCESS_PROMPT, FEATURE_ENGINEERING_PROMPT, MODEL_EVALUATE_PROMPT, From c8858cd8d464ef2c477770f927310e1a84cc7b3c Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 16 Jan 2024 17:54:38 +0800 Subject: [PATCH 399/637] minimize ml_engineer --- metagpt/actions/ml_da_action.py | 2 +- metagpt/actions/write_analysis_code.py | 23 +++-- metagpt/prompts/ml_engineer.py | 8 +- metagpt/roles/ml_engineer.py | 111 +++++++------------------ metagpt/tools/tool_data_type.py | 1 + metagpt/tools/tool_types.py | 6 ++ 6 files changed, 51 insertions(+), 100 deletions(-) diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py index d4e77773f..584c4db7a 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_da_action.py @@ -63,4 +63,4 @@ class UpdateDataColumns(Action): prompt = UPDATE_DATA_COLUMNS.format(history_code=code_context) tool_config = create_func_config(PRINT_DATA_COLUMNS) rsp = await self.llm.aask_code(prompt, **tool_config) - return rsp + return rsp["code"] diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index f4ae1e572..efd1ea163 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -155,10 +155,6 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): ) code_steps = plan.current_task.code_steps - finished_tasks = plan.get_finished_tasks() - code_context = [remove_comments(task.code) for task in finished_tasks] - code_context = "\n\n".join(code_context) - tool_catalog = {} if available_tools: @@ -189,26 +185,28 @@ class WriteCodeWithToolsML(WriteCodeWithTools): column_info: str = "", **kwargs, ) -> Tuple[List[Message], str]: - tool_type = plan.current_task.task_type - available_tools = self.available_tools.get(tool_type, {}) - special_prompt = TOOL_TYPE_USAGE_PROMPT.get(tool_type, "") + tool_type = ( + plan.current_task.task_type + ) # find tool type from task type through exact match, can extend to retrieval in the future + available_tools = TOOL_REGISTRY.get_tools_by_type(tool_type) + special_prompt = ( + TOOL_REGISTRY.get_tool_type(tool_type).usage_prompt if TOOL_REGISTRY.has_tool_type(tool_type) else "" + ) code_steps = plan.current_task.code_steps finished_tasks = plan.get_finished_tasks() code_context = [remove_comments(task.code) for task in finished_tasks] code_context = "\n\n".join(code_context) - if len(available_tools) > 0: - available_tools = {k: v["description"] for k, v in available_tools.items()} + if available_tools: + available_tools = {tool_name: tool.schema["description"] for tool_name, tool in available_tools.items()} recommend_tools = await self._tool_recommendation( plan.current_task.instruction, code_steps, available_tools ) - tool_catalog = self._parse_recommend_tools(tool_type, recommend_tools) + tool_catalog = self._parse_recommend_tools(recommend_tools) logger.info(f"Recommended tools: \n{recommend_tools}") - module_name = TOOL_TYPE_MODULE[tool_type] - prompt = ML_TOOL_USAGE_PROMPT.format( user_requirement=plan.goal, history_code=code_context, @@ -216,7 +214,6 @@ class WriteCodeWithToolsML(WriteCodeWithTools): column_info=column_info, special_prompt=special_prompt, code_steps=code_steps, - module_name=module_name, tool_catalog=tool_catalog, ) diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index ff29d5ed4..3fd895e6e 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -134,16 +134,12 @@ PRINT_DATA_COLUMNS = { "parameters": { "type": "object", "properties": { - "is_update": { - "type": "boolean", - "description": "Whether need to update the column info.", - }, "code": { "type": "string", "description": "The code to be added to a new cell in jupyter.", }, }, - "required": ["is_update", "code"], + "required": ["code"], }, } @@ -240,7 +236,7 @@ Strictly follow steps below when you writing code if it's convenient. - You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. # Available Tools: -Each Class tool is described in JSON format. When you call a tool, import the tool from `{module_name}` first. +Each Class tool is described in JSON format. When you call a tool, import the tool from its path first. {tool_catalog} # Output Example: diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index a60642bff..aeea39c0c 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,64 +1,43 @@ -from metagpt.actions.ask_review import ReviewConst from metagpt.actions.debug_code import DebugCode from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.ml_da_action import Reflect, SummarizeAnalysis, UpdateDataColumns +from metagpt.actions.ml_da_action import UpdateDataColumns from metagpt.actions.write_analysis_code import WriteCodeWithToolsML -from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter -from metagpt.roles.kaggle_manager import DownloadData, SubmitResult -from metagpt.schema import Message +from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.utils.common import any_to_str class MLEngineer(CodeInterpreter): - use_code_steps: bool = False - use_udfs: bool = False - data_desc: dict = {} debug_context: list = [] latest_code: str = "" def __init__(self, name="Mark", profile="MLEngineer", **kwargs): super().__init__(name=name, profile=profile, **kwargs) - # self._watch([DownloadData, SubmitResult]) # in multi-agent settings - - async def _plan_and_act(self): - ### a new attempt on the data, relevant in a multi-agent multi-turn setting ### - await self._prepare_data_context() - - ### general plan process ### - await super()._plan_and_act() - - ### summarize analysis ### - summary = await SummarizeAnalysis().run(self.planner.plan) - rsp = Message(content=summary, cause_by=SummarizeAnalysis) - self.rc.memory.add(rsp) - - return rsp - - async def _write_and_exec_code(self, max_retry: int = 3): - self.planner.current_task.code_steps = ( - await WriteCodeSteps().run(self.planner.plan) if self.use_code_steps else "" - ) - - code, result, success = await super()._write_and_exec_code(max_retry=max_retry) - - if success: - if self.use_tools and self.planner.current_task.task_type in ["data_preprocess", "feature_engineering"]: - update_success, new_code = await self._update_data_columns() - if update_success: - code = code + "\n\n" + new_code - - return code, result, success async def _write_code(self): if not self.use_tools: return await super()._write_code() - code_execution_count = sum([msg.cause_by == any_to_str(ExecutePyCode) for msg in self.working_memory.get()]) + # In a trial and errors settings, check whether this is our first attempt to tackle the task. If there is no code execution before, then it is. + is_first_trial = any_to_str(ExecutePyCode) not in [msg.cause_by for msg in self.working_memory.get()] - if code_execution_count > 0: - logger.warning("We got a bug code, now start to debug...") + if is_first_trial: + # For the first trial, write task code from scratch + column_info = await self._update_data_columns() + + logger.info("Write code with tools") + tool_context, code = await WriteCodeWithToolsML().run( + context=[], # context assembled inside the Action + plan=self.planner.plan, + column_info=column_info, + ) + self.debug_context = tool_context + cause_by = WriteCodeWithToolsML + + else: + # Previous trials resulted in error, debug and rewrite the code + logger.warning("We got a bug, now start to debug...") code = await DebugCode().run( code=self.latest_code, runtime_result=self.working_memory.get(), @@ -67,49 +46,21 @@ class MLEngineer(CodeInterpreter): logger.info(f"new code \n{code}") cause_by = DebugCode - else: - logger.info("Write code with tools") - tool_context, code = await WriteCodeWithToolsML().run( - context=[], # context assembled inside the Action - plan=self.planner.plan, - column_info=self.data_desc.get("column_info", ""), - ) - self.debug_context = tool_context - cause_by = WriteCodeWithToolsML - self.latest_code = code return code, cause_by async def _update_data_columns(self): + current_task = self.planner.plan.current_task + if current_task.task_type not in [ + ToolTypeEnum.DATA_PREPROCESS.value, + ToolTypeEnum.FEATURE_ENGINEERING.value, + ToolTypeEnum.MODEL_TRAIN.value, + ]: + return "" logger.info("Check columns in updated data") - rsp = await UpdateDataColumns().run(self.planner.plan) - is_update, code = rsp["is_update"], rsp["code"] + code = await UpdateDataColumns().run(self.planner.plan) success = False - if is_update: - result, success = await self.execute_code.run(code) - if success: - print(result) - self.data_desc["column_info"] = result - return success, code - - async def _prepare_data_context(self): - memories = self.get_memories() - if memories: - latest_event = memories[-1].cause_by - if latest_event == DownloadData: - self.planner.plan.context = memories[-1].content - elif latest_event == SubmitResult: - # self reflect on previous plan outcomes and think about how to improve the plan, add to working memory - await self._reflect() - - # get feedback for improvement from human, add to working memory - await self.planner.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - - async def _reflect(self): - context = self.get_memories() - context = "\n".join([str(msg) for msg in context]) - - reflection = await Reflect().run(context=context) - self.working_memory.add(Message(content=reflection, role="assistant")) - self.working_memory.add(Message(content=Reflect.REWRITE_PLAN_INSTRUCTION, role="user")) + result, success = await self.execute_code.run(code) + print(result) + return result if success else "" diff --git a/metagpt/tools/tool_data_type.py b/metagpt/tools/tool_data_type.py index c767fef9b..a3ab20a4e 100644 --- a/metagpt/tools/tool_data_type.py +++ b/metagpt/tools/tool_data_type.py @@ -4,6 +4,7 @@ from pydantic import BaseModel class ToolTypeEnum(Enum): + EDA = "eda" DATA_PREPROCESS = "data_preprocess" FEATURE_ENGINEERING = "feature_engineering" MODEL_TRAIN = "model_train" diff --git a/metagpt/tools/tool_types.py b/metagpt/tools/tool_types.py index 289271985..2e22adc40 100644 --- a/metagpt/tools/tool_types.py +++ b/metagpt/tools/tool_types.py @@ -8,6 +8,12 @@ from metagpt.tools.tool_data_type import ToolType, ToolTypeEnum from metagpt.tools.tool_registry import register_tool_type +@register_tool_type +class EDA(ToolType): + name: str = ToolTypeEnum.EDA.value + desc: str = "Useful for performing exploratory data analysis" + + @register_tool_type class DataPreprocess(ToolType): name: str = ToolTypeEnum.DATA_PREPROCESS.value From 9dc421b1229bc88fb9b5f2c8307fd98b16874ab5 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 16 Jan 2024 19:18:03 +0800 Subject: [PATCH 400/637] rename schema to schemas to avoid pydantic warning --- metagpt/actions/write_analysis_code.py | 6 +++--- metagpt/tools/tool_data_type.py | 2 +- metagpt/tools/tool_registry.py | 10 +++++----- 3 files changed, 9 insertions(+), 9 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index efd1ea163..65be198ef 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -110,7 +110,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): if TOOL_REGISTRY.has_tool(tool_name): valid_tools.append(TOOL_REGISTRY.get_tool(tool_name)) - tool_catalog = {tool.name: tool.schema for tool in valid_tools} + tool_catalog = {tool.name: tool.schemas for tool in valid_tools} return tool_catalog async def _tool_recommendation( @@ -158,7 +158,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): tool_catalog = {} if available_tools: - available_tools = {tool_name: tool.schema["description"] for tool_name, tool in available_tools.items()} + available_tools = {tool_name: tool.schemas["description"] for tool_name, tool in available_tools.items()} recommend_tools = await self._tool_recommendation( plan.current_task.instruction, code_steps, available_tools @@ -199,7 +199,7 @@ class WriteCodeWithToolsML(WriteCodeWithTools): code_context = "\n\n".join(code_context) if available_tools: - available_tools = {tool_name: tool.schema["description"] for tool_name, tool in available_tools.items()} + available_tools = {tool_name: tool.schemas["description"] for tool_name, tool in available_tools.items()} recommend_tools = await self._tool_recommendation( plan.current_task.instruction, code_steps, available_tools diff --git a/metagpt/tools/tool_data_type.py b/metagpt/tools/tool_data_type.py index a3ab20a4e..8206afa59 100644 --- a/metagpt/tools/tool_data_type.py +++ b/metagpt/tools/tool_data_type.py @@ -29,5 +29,5 @@ class ToolSchema(BaseModel): class Tool(BaseModel): name: str path: str - schema: dict = {} + schemas: dict = {} code: str = "" diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index 2c59cd198..5d743358c 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -25,7 +25,7 @@ class ToolRegistry: def register_tool_type(self, tool_type: ToolType): self.tool_types[tool_type.name] = tool_type - logger.info(f"{tool_type.name} registered") + logger.info(f"tool type {tool_type.name} registered") def register_tool( self, @@ -51,16 +51,16 @@ class ToolRegistry: with open(schema_path, "r", encoding="utf-8") as f: schema_dict = yaml.safe_load(f) - schema = schema_dict.get(tool_name) or dict(schema_dict.values()) - schema["tool_path"] = tool_path # corresponding code file path of the tool + schemas = schema_dict.get(tool_name) or dict(schema_dict.values()) + schemas["tool_path"] = tool_path # corresponding code file path of the tool try: - ToolSchema(**schema) # validation + ToolSchema(**schemas) # validation except Exception: pass # logger.warning( # f"{tool_name} schema not conforms to required format, but will be used anyway. Mismatch: {e}" # ) - tool = Tool(name=tool_name, path=tool_path, schema=schema, code=tool_code) + tool = Tool(name=tool_name, path=tool_path, schemas=schemas, code=tool_code) self.tools[tool_name] = tool self.tools_by_types[tool_type_name][tool_name] = tool logger.info(f"{tool_name} registered") From 1cabf2c503f2de5c037049af78923ad2faa2be4a Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 18 Jan 2024 20:34:32 +0800 Subject: [PATCH 401/637] change register arg name, integrate image2web tool --- metagpt/prompts/tool_types.py | 4 +- metagpt/tools/__init__.py | 4 +- metagpt/tools/libs/__init__.py | 5 +- metagpt/tools/libs/data_preprocess.py | 18 +++---- metagpt/tools/libs/feature_engineering.py | 20 ++++---- .../vision.py => libs/gpt_v_generator.py} | 34 ++++++------- metagpt/tools/libs/sd_engine.py | 5 +- .../image2webpage/GPTvGenerator.yml} | 2 +- metagpt/tools/tool_data_type.py | 1 + metagpt/tools/tool_registry.py | 12 ++--- metagpt/tools/tool_types.py | 8 ++++ .../tools/functions/libs/test_vision.py | 48 ------------------- .../tools/libs/test_gpt_v_generator.py | 40 ++++++++++++++++ .../libs/{test_sd.py => test_sd_engine.py} | 0 tests/metagpt/tools/test_tool_registry.py | 2 +- 15 files changed, 100 insertions(+), 103 deletions(-) rename metagpt/tools/{functions/libs/vision.py => libs/gpt_v_generator.py} (85%) rename metagpt/tools/{functions/schemas/vision.yml => schemas/image2webpage/GPTvGenerator.yml} (93%) delete mode 100644 tests/metagpt/tools/functions/libs/test_vision.py create mode 100644 tests/metagpt/tools/libs/test_gpt_v_generator.py rename tests/metagpt/tools/libs/{test_sd.py => test_sd_engine.py} (100%) diff --git a/metagpt/prompts/tool_types.py b/metagpt/prompts/tool_types.py index 43ead78a6..c01a80310 100644 --- a/metagpt/prompts/tool_types.py +++ b/metagpt/prompts/tool_types.py @@ -39,7 +39,7 @@ The current task is about evaluating a model, please note the following: """ # Prompt for using tools of "vision" type -VISION_PROMPT = """ +IMAGE2WEBPAGE_PROMPT = """ The current task is about converting image into webpage code. please note the following: - Single-Step Code Generation: Execute the entire code generation process in a single step, encompassing HTML, CSS, and JavaScript. Avoid fragmenting the code generation into multiple separate steps to maintain consistency and simplify the development workflow. -""" \ No newline at end of file +""" diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index 23b51533d..f18d1d276 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -11,9 +11,7 @@ from metagpt.tools import tool_types # this registers all tool types from metagpt.tools import libs # this registers all tools from metagpt.tools.tool_registry import TOOL_REGISTRY -_ = tool_types # Avoid pre-commit error -_ = libs # Avoid pre-commit error -_ = TOOL_REGISTRY # Avoid pre-commit error +_, _, _ = tool_types, libs, TOOL_REGISTRY # Avoid pre-commit error class SearchEngineType(Enum): diff --git a/metagpt/tools/libs/__init__.py b/metagpt/tools/libs/__init__.py index 3d74674aa..b576997c9 100644 --- a/metagpt/tools/libs/__init__.py +++ b/metagpt/tools/libs/__init__.py @@ -7,7 +7,8 @@ from metagpt.tools.libs import ( data_preprocess, feature_engineering, + sd_engine, + gpt_v_generator, ) -_ = data_preprocess # Avoid pre-commit error -_ = feature_engineering # Avoid pre-commit error +_, _, _, _ = data_preprocess, feature_engineering, sd_engine, gpt_v_generator # Avoid pre-commit error diff --git a/metagpt/tools/libs/data_preprocess.py b/metagpt/tools/libs/data_preprocess.py index 7cc44263d..3891f9df0 100644 --- a/metagpt/tools/libs/data_preprocess.py +++ b/metagpt/tools/libs/data_preprocess.py @@ -31,7 +31,7 @@ class MLProcess(object): return self.transform(df) -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class FillMissingValue(MLProcess): def __init__( self, @@ -58,7 +58,7 @@ class FillMissingValue(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class MinMaxScale(MLProcess): def __init__( self, @@ -77,7 +77,7 @@ class MinMaxScale(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class StandardScale(MLProcess): def __init__( self, @@ -96,7 +96,7 @@ class StandardScale(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class MaxAbsScale(MLProcess): def __init__( self, @@ -115,7 +115,7 @@ class MaxAbsScale(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class RobustScale(MLProcess): def __init__( self, @@ -134,7 +134,7 @@ class RobustScale(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class OrdinalEncode(MLProcess): def __init__( self, @@ -153,7 +153,7 @@ class OrdinalEncode(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class OneHotEncode(MLProcess): def __init__( self, @@ -175,7 +175,7 @@ class OneHotEncode(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class LabelEncode(MLProcess): def __init__( self, @@ -204,7 +204,7 @@ class LabelEncode(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) def get_column_info(df: pd.DataFrame) -> dict: column_info = { "Category": [], diff --git a/metagpt/tools/libs/feature_engineering.py b/metagpt/tools/libs/feature_engineering.py index ed5c1be72..308150f9b 100644 --- a/metagpt/tools/libs/feature_engineering.py +++ b/metagpt/tools/libs/feature_engineering.py @@ -22,7 +22,7 @@ from metagpt.tools.tool_registry import register_tool TOOL_TYPE = ToolTypeEnum.FEATURE_ENGINEERING.value -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class PolynomialExpansion(MLProcess): def __init__(self, cols: list, degree: int = 2, label_col: str = None): self.cols = cols @@ -53,7 +53,7 @@ class PolynomialExpansion(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class CatCount(MLProcess): def __init__(self, col: str): self.col = col @@ -68,7 +68,7 @@ class CatCount(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class TargetMeanEncoder(MLProcess): def __init__(self, col: str, label: str): self.col = col @@ -84,7 +84,7 @@ class TargetMeanEncoder(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class KFoldTargetMeanEncoder(MLProcess): def __init__(self, col: str, label: str, n_splits: int = 5, random_state: int = 2021): self.col = col @@ -111,7 +111,7 @@ class KFoldTargetMeanEncoder(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class CatCross(MLProcess): def __init__(self, cols: list, max_cat_num: int = 100): self.cols = cols @@ -147,7 +147,7 @@ class CatCross(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class GroupStat(MLProcess): def __init__(self, group_col: str, agg_col: str, agg_funcs: list): self.group_col = group_col @@ -167,7 +167,7 @@ class GroupStat(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class SplitBins(MLProcess): def __init__(self, cols: list, strategy: str = "quantile"): self.cols = cols @@ -184,7 +184,7 @@ class SplitBins(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class ExtractTimeComps(MLProcess): def __init__(self, time_col: str, time_comps: list): self.time_col = time_col @@ -213,7 +213,7 @@ class ExtractTimeComps(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class GeneralSelection(MLProcess): def __init__(self, label_col: str): self.label_col = label_col @@ -284,7 +284,7 @@ class TreeBasedSelection(MLProcess): return new_df -@register_tool(tool_type_name=TOOL_TYPE) +@register_tool(tool_type=TOOL_TYPE) class VarianceBasedSelection(MLProcess): def __init__(self, label_col: str, threshold: float = 0): self.label_col = label_col diff --git a/metagpt/tools/functions/libs/vision.py b/metagpt/tools/libs/gpt_v_generator.py similarity index 85% rename from metagpt/tools/functions/libs/vision.py rename to metagpt/tools/libs/gpt_v_generator.py index b10ad7608..58e547840 100644 --- a/metagpt/tools/functions/libs/vision.py +++ b/metagpt/tools/libs/gpt_v_generator.py @@ -5,18 +5,13 @@ @Author : mannaandpoem @File : vision.py """ +import base64 from pathlib import Path import requests -import base64 - -from metagpt.config import CONFIG - -OPENAI_API_BASE = CONFIG.OPENAI_BASE_URL -API_KEY = CONFIG.OPENAI_API_KEY -MODEL = CONFIG.OPENAI_VISION_MODEL -MAX_TOKENS = CONFIG.VISION_MAX_TOKENS +from metagpt.tools.tool_data_type import ToolTypeEnum +from metagpt.tools.tool_registry import register_tool ANALYZE_LAYOUT_PROMPT = """You are now a UI/UX, please generate layout information for this image: @@ -33,8 +28,15 @@ As the design pays tribute to large companies, sometimes it is normal for some c Now, please generate the corresponding webpage code including HTML, CSS and JavaScript:""" -class Vision: +@register_tool(tool_type=ToolTypeEnum.IMAGE2WEBPAGE.value) +class GPTvGenerator: def __init__(self): + from metagpt.config import CONFIG + + OPENAI_API_BASE = CONFIG.OPENAI_BASE_URL + API_KEY = CONFIG.OPENAI_API_KEY + MODEL = CONFIG.OPENAI_VISION_MODEL + MAX_TOKENS = CONFIG.VISION_MAX_TOKENS self.api_key = API_KEY self.api_base = OPENAI_API_BASE self.model = MODEL @@ -51,10 +53,7 @@ class Vision: def get_result(self, image_path, prompt): base64_image = self.encode_image(image_path) - headers = { - "Content-Type": "application/json", - "Authorization": f"Bearer {self.api_key}" - } + headers = {"Content-Type": "application/json", "Authorization": f"Bearer {self.api_key}"} payload = { "model": self.model, "messages": [ @@ -62,11 +61,8 @@ class Vision: "role": "user", "content": [ {"type": "text", "text": prompt}, - { - "type": "image_url", - "image_url": {"url": f"data:image/jpeg;base64,{base64_image}"} - } - ] + {"type": "image_url", "image_url": {"url": f"data:image/jpeg;base64,{base64_image}"}}, + ], } ], "max_tokens": self.max_tokens, @@ -81,7 +77,7 @@ class Vision: @staticmethod def encode_image(image_path): with open(image_path, "rb") as image_file: - return base64.b64encode(image_file.read()).decode('utf-8') + return base64.b64encode(image_file.read()).decode("utf-8") @staticmethod def save_webpages(image_path, webpages) -> Path: diff --git a/metagpt/tools/libs/sd_engine.py b/metagpt/tools/libs/sd_engine.py index ad63c2505..794758f77 100644 --- a/metagpt/tools/libs/sd_engine.py +++ b/metagpt/tools/libs/sd_engine.py @@ -13,7 +13,6 @@ import requests from aiohttp import ClientSession from PIL import Image, PngImagePlugin -from metagpt.config import CONFIG from metagpt.const import SD_OUTPUT_FILE_REPO from metagpt.logs import logger from metagpt.tools.tool_data_type import ToolTypeEnum @@ -53,9 +52,11 @@ payload = { default_negative_prompt = "(easynegative:0.8),black, dark,Low resolution" -@register_tool(tool_type_name=ToolTypeEnum.STABLE_DIFFUSION.value) +@register_tool(tool_type=ToolTypeEnum.STABLE_DIFFUSION.value) class SDEngine: def __init__(self, sd_url=""): + from metagpt.config import CONFIG + # Initialize the SDEngine with configuration self.sd_url = sd_url if sd_url else CONFIG.get("SD_URL") self.sd_t2i_url = f"{self.sd_url}{CONFIG.get('SD_T2I_API')}" diff --git a/metagpt/tools/functions/schemas/vision.yml b/metagpt/tools/schemas/image2webpage/GPTvGenerator.yml similarity index 93% rename from metagpt/tools/functions/schemas/vision.yml rename to metagpt/tools/schemas/image2webpage/GPTvGenerator.yml index 4cb247419..4087f7c12 100644 --- a/metagpt/tools/functions/schemas/vision.yml +++ b/metagpt/tools/schemas/image2webpage/GPTvGenerator.yml @@ -1,4 +1,4 @@ -Vision: +GPTvGenerator: type: class description: "Class for generating web pages at once." methods: diff --git a/metagpt/tools/tool_data_type.py b/metagpt/tools/tool_data_type.py index 8206afa59..45fb539a6 100644 --- a/metagpt/tools/tool_data_type.py +++ b/metagpt/tools/tool_data_type.py @@ -10,6 +10,7 @@ class ToolTypeEnum(Enum): MODEL_TRAIN = "model_train" MODEL_EVALUATE = "model_evaluate" STABLE_DIFFUSION = "stable_diffusion" + IMAGE2WEBPAGE = "image2webpage" OTHER = "other" def __missing__(self, key): diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index 5d743358c..0544d25ee 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -21,7 +21,7 @@ class ToolRegistry: def __init__(self): self.tools = {} self.tool_types = {} - self.tools_by_types = defaultdict(dict) # two-layer k-v, {tool_type_name: {tool_name: {...}, ...}, ...} + self.tools_by_types = defaultdict(dict) # two-layer k-v, {tool_type: {tool_name: {...}, ...}, ...} def register_tool_type(self, tool_type: ToolType): self.tool_types[tool_type.name] = tool_type @@ -33,13 +33,13 @@ class ToolRegistry: tool_path, schema_path=None, tool_code="", - tool_type_name="other", + tool_type="other", make_schema_if_not_exists=False, ): if self.has_tool(tool_name): return - schema_path = schema_path or TOOL_SCHEMA_PATH / tool_type_name / f"{tool_name}.yml" + schema_path = schema_path or TOOL_SCHEMA_PATH / tool_type / f"{tool_name}.yml" if not os.path.exists(schema_path): if make_schema_if_not_exists: @@ -62,7 +62,7 @@ class ToolRegistry: # ) tool = Tool(name=tool_name, path=tool_path, schemas=schemas, code=tool_code) self.tools[tool_name] = tool - self.tools_by_types[tool_type_name][tool_name] = tool + self.tools_by_types[tool_type][tool_name] = tool logger.info(f"{tool_name} registered") def has_tool(self, key): @@ -94,7 +94,7 @@ def register_tool_type(cls): return cls -def register_tool(tool_name="", tool_type_name="other", schema_path=None): +def register_tool(tool_name="", tool_type="other", schema_path=None): """register a tool to registry""" def decorator(cls, tool_name=tool_name): @@ -111,7 +111,7 @@ def register_tool(tool_name="", tool_type_name="other", schema_path=None): tool_path=file_path, schema_path=schema_path, tool_code=source_code, - tool_type_name=tool_type_name, + tool_type=tool_type, ) return cls diff --git a/metagpt/tools/tool_types.py b/metagpt/tools/tool_types.py index 2e22adc40..b5b233d53 100644 --- a/metagpt/tools/tool_types.py +++ b/metagpt/tools/tool_types.py @@ -1,6 +1,7 @@ from metagpt.prompts.tool_types import ( DATA_PREPROCESS_PROMPT, FEATURE_ENGINEERING_PROMPT, + IMAGE2WEBPAGE_PROMPT, MODEL_EVALUATE_PROMPT, MODEL_TRAIN_PROMPT, ) @@ -48,6 +49,13 @@ class StableDiffusion(ToolType): desc: str = "Related to text2image, image2image using stable diffusion model." +@register_tool_type +class Image2Webpage(ToolType): + name: str = ToolTypeEnum.IMAGE2WEBPAGE.value + desc: str = "For converting image into webpage code." + usage_prompt: str = IMAGE2WEBPAGE_PROMPT + + @register_tool_type class Other(ToolType): name: str = ToolTypeEnum.OTHER.value diff --git a/tests/metagpt/tools/functions/libs/test_vision.py b/tests/metagpt/tools/functions/libs/test_vision.py deleted file mode 100644 index f4f97c46a..000000000 --- a/tests/metagpt/tools/functions/libs/test_vision.py +++ /dev/null @@ -1,48 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -""" -@Time : 2024/01/15 -@Author : mannaandpoem -@File : test_vision.py -""" -import pytest - -from metagpt import logs -from metagpt.tools.functions.libs.vision import Vision - - -@pytest.fixture -def mock_webpages(): - return """```html\n\n -\n\n```\n -```css\n.class { ... }\n```\n -```javascript\nfunction() { ... }\n```\n""" - - -def test_vision_generate_webpages(mocker, mock_webpages): - mocker.patch( - "metagpt.tools.functions.libs.vision.Vision.generate_web_pages", - return_value=mock_webpages - ) - image_path = "image.png" - vision = Vision() - rsp = vision.generate_web_pages(image_path=image_path) - logs.logger.info(rsp) - assert "html" in rsp - assert "css" in rsp - assert "javascript" in rsp - - -def test_save_webpages(mocker, mock_webpages): - mocker.patch( - "metagpt.tools.functions.libs.vision.Vision.generate_web_pages", - return_value=mock_webpages - ) - image_path = "image.png" - vision = Vision() - webpages = vision.generate_web_pages(image_path) - webpages_dir = vision.save_webpages(image_path=image_path, webpages=webpages) - logs.logger.info(webpages_dir) - assert webpages_dir.exists() - - diff --git a/tests/metagpt/tools/libs/test_gpt_v_generator.py b/tests/metagpt/tools/libs/test_gpt_v_generator.py new file mode 100644 index 000000000..360ca4a75 --- /dev/null +++ b/tests/metagpt/tools/libs/test_gpt_v_generator.py @@ -0,0 +1,40 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/01/15 +@Author : mannaandpoem +@File : test_vision.py +""" +import pytest + +from metagpt import logs +from metagpt.tools.libs.gpt_v_generator import GPTvGenerator + + +@pytest.fixture +def mock_webpages(mocker): + mock_data = """```html\n\n +\n\n```\n +```css\n.class { ... }\n```\n +```javascript\nfunction() { ... }\n```\n""" + mocker.patch("metagpt.tools.libs.gpt_v_generator.GPTvGenerator.generate_web_pages", return_value=mock_data) + return mocker + + +def test_vision_generate_webpages(mock_webpages): + image_path = "image.png" + generator = GPTvGenerator() + rsp = generator.generate_web_pages(image_path=image_path) + logs.logger.info(rsp) + assert "html" in rsp + assert "css" in rsp + assert "javascript" in rsp + + +def test_save_webpages(mock_webpages): + image_path = "image.png" + generator = GPTvGenerator() + webpages = generator.generate_web_pages(image_path) + webpages_dir = generator.save_webpages(image_path=image_path, webpages=webpages) + logs.logger.info(webpages_dir) + assert webpages_dir.exists() diff --git a/tests/metagpt/tools/libs/test_sd.py b/tests/metagpt/tools/libs/test_sd_engine.py similarity index 100% rename from tests/metagpt/tools/libs/test_sd.py rename to tests/metagpt/tools/libs/test_sd_engine.py diff --git a/tests/metagpt/tools/test_tool_registry.py b/tests/metagpt/tools/test_tool_registry.py index fd758b141..582c368a8 100644 --- a/tests/metagpt/tools/test_tool_registry.py +++ b/tests/metagpt/tools/test_tool_registry.py @@ -88,7 +88,7 @@ def test_get_tools_by_type(tool_registry, schema_yaml): tool_type = ToolType(name=tool_type_name, desc="test") tool_registry.register_tool_type(tool_type) - tool_registry.register_tool(tool_name, tool_path, tool_type_name=tool_type_name) + tool_registry.register_tool(tool_name, tool_path, tool_type=tool_type_name) tools_by_type = tool_registry.get_tools_by_type(tool_type_name) assert tools_by_type is not None From c32dcca293e2431cecd147e670951a8bb2a8c13d Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 18 Jan 2024 21:17:34 +0800 Subject: [PATCH 402/637] fix schema reading bug --- metagpt/tools/tool_registry.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index 0544d25ee..52ad25ce4 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -51,7 +51,7 @@ class ToolRegistry: with open(schema_path, "r", encoding="utf-8") as f: schema_dict = yaml.safe_load(f) - schemas = schema_dict.get(tool_name) or dict(schema_dict.values()) + schemas = schema_dict.get(tool_name) or list(schema_dict.values())[0] schemas["tool_path"] = tool_path # corresponding code file path of the tool try: ToolSchema(**schemas) # validation From 1f7567e3c44e051e470542ea447b88d397a27519 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 18 Jan 2024 22:58:56 +0800 Subject: [PATCH 403/637] Update README.md --- README.md | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index 3ede740f7..61d03f692 100644 --- a/README.md +++ b/README.md @@ -34,17 +34,19 @@ # MetaGPT: The Multi-Agent Framework

Software Company Multi-Role Schematic (Gradually Implementing)

## News -🚀 Jan 16: Our paper: [MetaGPT: Meta Programming for A Multi-Agent Collaborative Framework](https://arxiv.org/abs/2308.00352) has been accepted by ICLR 2024 for **oral presentation (top 1.2%)**! More details are [here](https://openreview.net/forum?id=VtmBAGCN7o). +🚀 Jan. 16, 2024: [MetaGPT paper](https://arxiv.org/abs/2308.00352) accepted for oral presentation **(top 1.2%)** at ICLR 2024, **ranking #1** in the LLM-based Agent category. +🚀 Jan. 03, 2024: [v0.6.0](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0) released, new features include serialization, upgraded OpenAI package and supported multiple LLM etc. -🚀 Jan 03: Here comes [v0.6.0](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! In this version, we added serialization and deserialization of important objects and enabled breakpoint recovery. We upgraded OpenAI package to v1.6.0 and supported Gemini, ZhipuAI, Ollama, OpenLLM, etc. Moreover, we provided extremely simple examples where you need only 7 lines to implement a general election [debate](https://github.com/geekan/MetaGPT/blob/main/examples/debate_simple.py). Check out more details [here](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0)! +🚀 Dec. 15, 2023: [v0.5.0](https://github.com/geekan/MetaGPT/releases/tag/v0.5.0) released, introducing **incremental development**, **multilingual**, **multiple programming languages**, etc. +🔥 Nov. 08, 2023: MetaGPT is selected into [Open100: Top 100 Open Source achievements](https://www.benchcouncil.org/evaluation/opencs/annual.html). -🚀 Dec 15: [v0.5.0](https://github.com/geekan/MetaGPT/releases/tag/v0.5.0) is released! We introduced **incremental development**, facilitating agents to build up larger projects on top of their previous efforts or existing codebase. We also launched a whole collection of important features, including **multilingual support** (experimental), multiple **programming languages support** (experimental), **incremental development** (experimental), CLI support, pip support, enhanced code review, documentation mechanism, and optimized messaging mechanism! +🔥 Sep. 01, 2023: MetaGPT tops GitHub Trending Monthly for the **17th time** in August 2023. -🔥 Nov 8: MetaGPT is selected into [Open100: Top 100 Open Source achievements](https://www.benchcouncil.org/evaluation/opencs/annual.html). +🌟 Jun. 30, 2023: MetaGPT is now open source. -🔥 Sep 1: MetaGPT clinched the top spot for the **17th time** in GitHub's Trending Monthly for August 2023. +🌟 Apr. 24, 2023: First line of MetaGPT code committed. ## Install From 1460cff7295bff3234338b5233c8db3d312d5810 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 18 Jan 2024 23:07:43 +0800 Subject: [PATCH 404/637] Update README.md --- README.md | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index 61d03f692..6206b7f70 100644 --- a/README.md +++ b/README.md @@ -34,11 +34,12 @@ # MetaGPT: The Multi-Agent Framework

Software Company Multi-Role Schematic (Gradually Implementing)

## News -🚀 Jan. 16, 2024: [MetaGPT paper](https://arxiv.org/abs/2308.00352) accepted for oral presentation **(top 1.2%)** at ICLR 2024, **ranking #1** in the LLM-based Agent category. +🚀 Jan. 16, 2024: Our paper [MetaGPT: Meta Programming for A Multi-Agent Collaborative Framework +](https://arxiv.org/abs/2308.00352) accepted for oral presentation **(top 1.2%)** at ICLR 2024, **ranking #1** in the LLM-based Agent category. -🚀 Jan. 03, 2024: [v0.6.0](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0) released, new features include serialization, upgraded OpenAI package and supported multiple LLM etc. +🚀 Jan. 03, 2024: [v0.6.0](https://github.com/geekan/MetaGPT/releases/tag/v0.6.0) released, new features include serialization, upgraded OpenAI package and supported multiple LLM, provided [minimal example for debate](https://github.com/geekan/MetaGPT/blob/main/examples/debate_simple.py) etc. -🚀 Dec. 15, 2023: [v0.5.0](https://github.com/geekan/MetaGPT/releases/tag/v0.5.0) released, introducing **incremental development**, **multilingual**, **multiple programming languages**, etc. +🚀 Dec. 15, 2023: [v0.5.0](https://github.com/geekan/MetaGPT/releases/tag/v0.5.0) released, introducing some experimental features such as **incremental development**, **multilingual**, **multiple programming languages**, etc. 🔥 Nov. 08, 2023: MetaGPT is selected into [Open100: Top 100 Open Source achievements](https://www.benchcouncil.org/evaluation/opencs/annual.html). From 88c4c8c90d25e7d7b46ba453df55106345be6843 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 18 Jan 2024 23:26:34 +0800 Subject: [PATCH 405/637] integrate web scraping tool --- metagpt/tools/__init__.py | 2 +- metagpt/tools/functions/libs/scrape_web/__init__.py | 1 - metagpt/tools/libs/__init__.py | 3 ++- .../scrape_web/scrape_web.py => libs/web_scrapping.py} | 9 ++++----- .../web_scrapping/scrape_web_playwright.yml} | 2 +- metagpt/tools/tool_data_type.py | 1 + metagpt/tools/tool_types.py | 8 +++++++- metagpt/tools/web_browser_engine_playwright.py | 3 ++- 8 files changed, 18 insertions(+), 11 deletions(-) delete mode 100644 metagpt/tools/functions/libs/scrape_web/__init__.py rename metagpt/tools/{functions/libs/scrape_web/scrape_web.py => libs/web_scrapping.py} (76%) rename metagpt/tools/{functions/schemas/scrape_web.yml => schemas/web_scrapping/scrape_web_playwright.yml} (96%) diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index f18d1d276..bb87f1b62 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -11,7 +11,7 @@ from metagpt.tools import tool_types # this registers all tool types from metagpt.tools import libs # this registers all tools from metagpt.tools.tool_registry import TOOL_REGISTRY -_, _, _ = tool_types, libs, TOOL_REGISTRY # Avoid pre-commit error +_ = tool_types, libs, TOOL_REGISTRY # Avoid pre-commit error class SearchEngineType(Enum): diff --git a/metagpt/tools/functions/libs/scrape_web/__init__.py b/metagpt/tools/functions/libs/scrape_web/__init__.py deleted file mode 100644 index d5cd1524b..000000000 --- a/metagpt/tools/functions/libs/scrape_web/__init__.py +++ /dev/null @@ -1 +0,0 @@ -from metagpt.tools.functions.libs.scrape_web.scrape_web import scrape_web diff --git a/metagpt/tools/libs/__init__.py b/metagpt/tools/libs/__init__.py index b576997c9..442f57149 100644 --- a/metagpt/tools/libs/__init__.py +++ b/metagpt/tools/libs/__init__.py @@ -9,6 +9,7 @@ from metagpt.tools.libs import ( feature_engineering, sd_engine, gpt_v_generator, + web_scrapping, ) -_, _, _, _ = data_preprocess, feature_engineering, sd_engine, gpt_v_generator # Avoid pre-commit error +_ = data_preprocess, feature_engineering, sd_engine, gpt_v_generator, web_scrapping # Avoid pre-commit error diff --git a/metagpt/tools/functions/libs/scrape_web/scrape_web.py b/metagpt/tools/libs/web_scrapping.py similarity index 76% rename from metagpt/tools/functions/libs/scrape_web/scrape_web.py rename to metagpt/tools/libs/web_scrapping.py index e68ce0e64..e8e73f123 100644 --- a/metagpt/tools/functions/libs/scrape_web/scrape_web.py +++ b/metagpt/tools/libs/web_scrapping.py @@ -1,9 +1,10 @@ -import asyncio - +from metagpt.tools.tool_data_type import ToolTypeEnum +from metagpt.tools.tool_registry import register_tool from metagpt.tools.web_browser_engine_playwright import PlaywrightWrapper -async def scrape_web(url, *urls): +@register_tool(tool_type=ToolTypeEnum.WEBSCRAPING.value) +async def scrape_web_playwright(url, *urls): """ Scrape and save the HTML structure and inner text content of a web page using Playwright. @@ -19,5 +20,3 @@ async def scrape_web(url, *urls): # Return the inner text content of the web page return {"inner_text": web.inner_text, "html": web.html} - -# 需要改三个地方: yaml, 对应路径下init, MetaGPT/metagpt/prompts/ml_engineer.py中ML_MODULE_MAP diff --git a/metagpt/tools/functions/schemas/scrape_web.yml b/metagpt/tools/schemas/web_scrapping/scrape_web_playwright.yml similarity index 96% rename from metagpt/tools/functions/schemas/scrape_web.yml rename to metagpt/tools/schemas/web_scrapping/scrape_web_playwright.yml index ecca3fbed..a6ff7d6c7 100644 --- a/metagpt/tools/functions/schemas/scrape_web.yml +++ b/metagpt/tools/schemas/web_scrapping/scrape_web_playwright.yml @@ -1,4 +1,4 @@ -scrape_web: +scrape_web_playwright: type: async funciton description: "Scrape and save the HTML structure and inner text content of a web page using Playwright." parameters: diff --git a/metagpt/tools/tool_data_type.py b/metagpt/tools/tool_data_type.py index 45fb539a6..0c4eea4cc 100644 --- a/metagpt/tools/tool_data_type.py +++ b/metagpt/tools/tool_data_type.py @@ -11,6 +11,7 @@ class ToolTypeEnum(Enum): MODEL_EVALUATE = "model_evaluate" STABLE_DIFFUSION = "stable_diffusion" IMAGE2WEBPAGE = "image2webpage" + WEBSCRAPING = "web_scraping" OTHER = "other" def __missing__(self, key): diff --git a/metagpt/tools/tool_types.py b/metagpt/tools/tool_types.py index b5b233d53..35c0772b1 100644 --- a/metagpt/tools/tool_types.py +++ b/metagpt/tools/tool_types.py @@ -12,7 +12,7 @@ from metagpt.tools.tool_registry import register_tool_type @register_tool_type class EDA(ToolType): name: str = ToolTypeEnum.EDA.value - desc: str = "Useful for performing exploratory data analysis" + desc: str = "For performing exploratory data analysis" @register_tool_type @@ -56,6 +56,12 @@ class Image2Webpage(ToolType): usage_prompt: str = IMAGE2WEBPAGE_PROMPT +@register_tool_type +class WebScraping(ToolType): + name: str = ToolTypeEnum.WEBSCRAPING.value + desc: str = "For scraping data from web pages." + + @register_tool_type class Other(ToolType): name: str = ToolTypeEnum.OTHER.value diff --git a/metagpt/tools/web_browser_engine_playwright.py b/metagpt/tools/web_browser_engine_playwright.py index a45f6a12e..15c8a78d7 100644 --- a/metagpt/tools/web_browser_engine_playwright.py +++ b/metagpt/tools/web_browser_engine_playwright.py @@ -12,7 +12,6 @@ from typing import Literal from playwright.async_api import async_playwright -from metagpt.config import CONFIG from metagpt.logs import logger from metagpt.utils.parse_html import WebPage @@ -32,6 +31,8 @@ class PlaywrightWrapper: launch_kwargs: dict | None = None, **kwargs, ) -> None: + from metagpt.config import CONFIG + if browser_type is None: browser_type = CONFIG.playwright_browser_type self.browser_type = browser_type From 5389c52556c2065208ef4f67560931ab6c9112a9 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 18 Jan 2024 23:34:46 +0800 Subject: [PATCH 406/637] Update README.md --- README.md | 22 ++++++++++++---------- 1 file changed, 12 insertions(+), 10 deletions(-) diff --git a/README.md b/README.md index 6206b7f70..90c586068 100644 --- a/README.md +++ b/README.md @@ -6,16 +6,16 @@ # MetaGPT: The Multi-Agent Framework

-Assign different roles to GPTs to form a collaborative software entity for complex tasks. +Assign different roles to GPTs to form a collaborative entity for complex tasks.

@@ -25,14 +25,6 @@ # MetaGPT: The Multi-Agent Framework Hugging Face

-1. MetaGPT takes a **one line requirement** as input and outputs **user stories / competitive analysis / requirements / data structures / APIs / documents, etc.** -2. Internally, MetaGPT includes **product managers / architects / project managers / engineers.** It provides the entire process of a **software company along with carefully orchestrated SOPs.** - 1. `Code = SOP(Team)` is the core philosophy. We materialize SOP and apply it to teams composed of LLMs. - -![A software company consists of LLM-based roles](docs/resources/software_company_cd.jpeg) - -

Software Company Multi-Role Schematic (Gradually Implementing)

- ## News 🚀 Jan. 16, 2024: Our paper [MetaGPT: Meta Programming for A Multi-Agent Collaborative Framework ](https://arxiv.org/abs/2308.00352) accepted for oral presentation **(top 1.2%)** at ICLR 2024, **ranking #1** in the LLM-based Agent category. @@ -49,6 +41,16 @@ ## News 🌟 Apr. 24, 2023: First line of MetaGPT code committed. +## Software Company as Multi-Agent System + +1. MetaGPT takes a **one line requirement** as input and outputs **user stories / competitive analysis / requirements / data structures / APIs / documents, etc.** +2. Internally, MetaGPT includes **product managers / architects / project managers / engineers.** It provides the entire process of a **software company along with carefully orchestrated SOPs.** + 1. `Code = SOP(Team)` is the core philosophy. We materialize SOP and apply it to teams composed of LLMs. + +![A software company consists of LLM-based roles](docs/resources/software_company_cd.jpeg) + +

Software Company Multi-Agent Schematic (Gradually Implementing)

+ ## Install ### Pip installation From 3faa094248d819a178156471c9990089b9a8d5a7 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 18 Jan 2024 23:45:37 +0800 Subject: [PATCH 407/637] fix aask_code issues in ml_engineer --- metagpt/actions/debug_code.py | 3 +-- metagpt/actions/ml_da_action.py | 2 +- metagpt/actions/write_analysis_code.py | 8 ++++---- metagpt/roles/code_interpreter.py | 11 ++++------- metagpt/roles/ml_engineer.py | 4 ++-- 5 files changed, 12 insertions(+), 16 deletions(-) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index e5e0ac5d4..121c126c4 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -119,5 +119,4 @@ class DebugCode(BaseWriteAnalysisCode): runtime_result=runtime_result, ) # 根据reflection结果重写代码 - improv_code = reflection["improved_impl"] - return improv_code + return {"code": reflection["improved_impl"]} diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_da_action.py index 584c4db7a..d4e77773f 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_da_action.py @@ -63,4 +63,4 @@ class UpdateDataColumns(Action): prompt = UPDATE_DATA_COLUMNS.format(history_code=code_context) tool_config = create_func_config(PRINT_DATA_COLUMNS) rsp = await self.llm.aask_code(prompt, **tool_config) - return rsp["code"] + return rsp diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 65be198ef..cf806a986 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -59,7 +59,7 @@ class BaseWriteAnalysisCode(Action): } return messages - async def run(self, context: List[Message], plan: Plan = None) -> str: + async def run(self, context: List[Message], plan: Plan = None) -> dict: """Run of a code writing action, used in data analysis or modeling Args: @@ -67,7 +67,7 @@ class BaseWriteAnalysisCode(Action): plan (Plan, optional): Overall plan. Defaults to None. Returns: - str: The code string. + dict: code result in the format of {"code": "print('hello world')", "language": "python"} """ @@ -174,7 +174,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) - return rsp["code"] + return rsp class WriteCodeWithToolsML(WriteCodeWithTools): @@ -230,7 +230,7 @@ class WriteCodeWithToolsML(WriteCodeWithTools): tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) context = [Message(content=prompt, role="user")] - return context, rsp["code"] + return context, rsp class MakeTools(WriteCodeByGenerate): diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 46cc00d5e..f972e72e2 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -54,7 +54,7 @@ class CodeInterpreter(Role): async def _act_on_task(self, current_task: Task) -> TaskResult: code, result, is_success = await self._write_and_exec_code() - task_result = TaskResult(code=code['code'], result=result, is_success=is_success) + task_result = TaskResult(code=code, result=result, is_success=is_success) return task_result async def _write_and_exec_code(self, max_retry: int = 3): @@ -69,7 +69,7 @@ class CodeInterpreter(Role): ### write code ### code, cause_by = await self._write_code() - self.working_memory.add(Message(content=code['code'], role="assistant", cause_by=cause_by)) + self.working_memory.add(Message(content=code["code"], role="assistant", cause_by=cause_by)) ### execute code ### result, success = await self.execute_code.run(**code) @@ -78,7 +78,7 @@ class CodeInterpreter(Role): self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) ### process execution result ### - if "!pip" in code: + if "!pip" in code["code"]: success = False counter += 1 @@ -89,7 +89,7 @@ class CodeInterpreter(Role): if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions - return code, result, success + return code["code"], result, success async def _write_code(self): todo = WriteCodeByGenerate() if not self.use_tools else WriteCodeWithTools() @@ -98,9 +98,6 @@ class CodeInterpreter(Role): context = self.planner.get_useful_memories() # print(*context, sep="\n***\n") code = await todo.run(context=context, plan=self.planner.plan, temperature=0.0) - # 暂时在这里转换 WriteCodeWithTools 的输出 - if isinstance(code, str): - code = {'code': code, 'language': 'python'} return code, todo diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index aeea39c0c..6b671f9c2 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -46,7 +46,7 @@ class MLEngineer(CodeInterpreter): logger.info(f"new code \n{code}") cause_by = DebugCode - self.latest_code = code + self.latest_code = code["code"] return code, cause_by @@ -61,6 +61,6 @@ class MLEngineer(CodeInterpreter): logger.info("Check columns in updated data") code = await UpdateDataColumns().run(self.planner.plan) success = False - result, success = await self.execute_code.run(code) + result, success = await self.execute_code.run(**code) print(result) return result if success else "" From 23fccdde67f50fed24906f22c5f3f8c0a58002da Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 19 Jan 2024 00:09:58 +0800 Subject: [PATCH 408/637] update mock llm aask_code --- tests/mock/mock_llm.py | 1 - 1 file changed, 1 deletion(-) diff --git a/tests/mock/mock_llm.py b/tests/mock/mock_llm.py index 45b28c63b..a52aeed09 100644 --- a/tests/mock/mock_llm.py +++ b/tests/mock/mock_llm.py @@ -69,7 +69,6 @@ class MockLLM(OriginalLLM): A copy of metagpt.provider.openai_api.OpenAILLM.aask_code, we can't use super().aask because it will be mocked. Since openai_api.OpenAILLM.aask_code is different from base_llm.BaseLLM.aask_code, we use the former. """ - messages = self._process_message(messages) rsp = await self._achat_completion_function(messages, **kwargs) return self.get_choice_function_arguments(rsp) From 5190dc446238055107725897786969316cb6ee30 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 09:03:24 +0800 Subject: [PATCH 409/637] 1. rename and modify guideline to plan 2. update prompt in ActionNode 3. add code comment 4. refactor Guideline code structure --- metagpt/actions/design_api_an.py | 6 +- metagpt/actions/project_management_an.py | 2 +- metagpt/actions/write_code.py | 39 ++++++----- ..._guideline_an.py => write_code_plan_an.py} | 48 +++++-------- metagpt/actions/write_code_review.py | 27 +++++--- metagpt/const.py | 5 +- metagpt/roles/engineer.py | 64 ++++++++--------- metagpt/utils/mermaid.py | 69 ------------------- tests/data/incremental_dev_project/mock.py | 4 +- ...eline_an.py => test_write_code_plan_an.py} | 32 ++++----- tests/metagpt/test_incremental_dev.py | 2 +- 11 files changed, 112 insertions(+), 186 deletions(-) rename metagpt/actions/{write_code_guideline_an.py => write_code_plan_an.py} (68%) rename tests/metagpt/actions/{test_write_code_guideline_an.py => test_write_code_plan_an.py} (61%) diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index 3e65cee1b..b872159e1 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -9,7 +9,7 @@ from typing import List from metagpt.actions.action_node import ActionNode from metagpt.logs import logger -from metagpt.utils.mermaid import MMC1, MMC1_REFINE, MMC2, MMC2_REFINE +from metagpt.utils.mermaid import MMC1, MMC2 IMPLEMENTATION_APPROACH = ActionNode( key="Implementation approach", @@ -62,7 +62,7 @@ REFINED_DATA_STRUCTURES_AND_INTERFACES = ActionNode( "methods (including __init__), and functions with precise type annotations. Delineate additional " "relationships between classes, ensuring clarity and adherence to PEP8 standards." "Retain content that is not related to incremental development but important for consistency and clarity.", - example=MMC1_REFINE, + example=MMC1, ) PROGRAM_CALL_FLOW = ActionNode( @@ -80,7 +80,7 @@ REFINED_PROGRAM_CALL_FLOW = ActionNode( "CRUD and initialization of each object. Ensure correct syntax usage and reflect the incremental changes introduced" "in the classes and API defined above. " "Retain content that is not related to incremental development but important for consistency and clarity.", - example=MMC2_REFINE, + example=MMC2, ) ANYTHING_UNCLEAR = ActionNode( diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index 8b0196707..e3d6537ab 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -118,7 +118,7 @@ REFINE_NODES = [ ] PM_NODE = ActionNode.from_children("PM_NODE", NODES) -REFINED_PM_NODES = ActionNode.from_children("Refined_PM_NODES", REFINE_NODES) +REFINED_PM_NODES = ActionNode.from_children("REFINED_PM_NODES", REFINE_NODES) def main(): diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 30ebf09f8..93c7a2f65 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -16,18 +16,21 @@ """ import json +from typing import Literal from pydantic import Field from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions.action import Action from metagpt.actions.project_management_an import REFINED_TASK_LIST, TASK_LIST -from metagpt.actions.write_code_guideline_an import REFINED_CODE_TEMPLATE +from metagpt.actions.write_code_plan_an import REFINED_CODE_TEMPLATE from metagpt.config import CONFIG from metagpt.const import ( BUGFIX_FILENAME, CODE_SUMMARIES_FILE_REPO, DOCS_FILE_REPO, + PLAN_FILENAME, + PLAN_PDF_FILE_REPO, REQUIREMENT_FILENAME, TASK_FILE_REPO, TEST_OUTPUTS_FILE_REPO, @@ -104,6 +107,9 @@ class WriteCode(Action): test_doc = await FileRepository.get_file( filename="test_" + coding_context.filename + ".json", relative_path=TEST_OUTPUTS_FILE_REPO ) + plan_doc = await FileRepository.get_file(filename=PLAN_FILENAME, relative_path=PLAN_PDF_FILE_REPO) + plan = plan_doc.content if plan_doc else "" + requirement_doc = await FileRepository.get_file(filename=REQUIREMENT_FILENAME, relative_path=DOCS_FILE_REPO) summary_doc = None if coding_context.design_doc and coding_context.design_doc.filename: summary_doc = await FileRepository.get_file( @@ -114,21 +120,17 @@ class WriteCode(Action): test_detail = RunCodeResult.loads(test_doc.content) logs = test_detail.stderr - docs_file_repo = CONFIG.git_repo.new_file_repository(relative_path=DOCS_FILE_REPO) - requirement_doc = await docs_file_repo.get(filename=REQUIREMENT_FILENAME) - - guideline = kwargs.get("guideline", "") if bug_feedback: code_context = coding_context.code_doc.content - elif guideline: - code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename, mode="guide") + elif plan: + code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename, mode="plan") else: code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename) - if guideline: + if plan: prompt = REFINED_CODE_TEMPLATE.format( user_requirement=requirement_doc.content if requirement_doc else "", - guideline=guideline, + plan=plan, design=coding_context.design_doc.content if coding_context.design_doc else "", tasks=coding_context.task_doc.content if coding_context.task_doc else "", code=code_context, @@ -157,14 +159,14 @@ class WriteCode(Action): return coding_context @staticmethod - async def get_codes(task_doc, exclude, mode="normal") -> str: + async def get_codes(task_doc: Document, exclude: str, mode: Literal["normal", "plan"] = "normal") -> str: """ Get code snippets based on different modes. Attributes: task_doc (Document): Document object of the task file. exclude (str): Specifies the filename to be excluded from the code snippets. - mode (str): Specifies the mode, either "normal" or "guide" (default is "normal"). + mode (str): Specifies the mode, either "normal" or "plan" (default is "normal"). Returns: str: Code snippets. @@ -173,7 +175,7 @@ class WriteCode(Action): If mode is set to "normal", it returns code snippets for the regular coding phase, i.e., all the code generated before writing the current file. - If mode is set to "guide", it returns code snippets for incremental development, + If mode is set to "plan", it returns code snippets for incremental development, building upon the existing code in the "normal" mode and adding code for the current file's older versions. """ if not task_doc: @@ -185,31 +187,36 @@ class WriteCode(Action): codes = [] src_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.src_workspace) - if mode == "guide": + if mode == "plan": src_files = src_file_repo.all_files old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) old_files = old_file_repo.all_files + # Get the union of the files in the src and old workspaces union_files_list = list(set(src_files) | set(old_files)) for filename in union_files_list: + # Exclude the current file from the all code snippets to get the context code snippets for generating if filename == exclude: + # If the file is in the old workspace, use the legacy code # Exclude unnecessary code to maintain a clean and focused main.py file, ensuring only relevant and # essential functionality is included for the project’s requirements if filename in old_files and filename != "main.py": # Use legacy code doc = await old_file_repo.get(filename=filename) + # If the file is in the src workspace, skip it else: continue codes.insert(0, f"-----Now, {filename} to be rewritten\n```{doc.content}```\n=====") - + # The context code snippets are generated from the src workspace else: - # Use new code doc = await src_file_repo.get(filename=filename) + # If the file does not exist in the src workspace, skip it if not doc: continue codes.append(f"----- {filename}\n```{doc.content}```") - else: + elif mode == "normal": for filename in code_filenames: + # Exclude the current file to get the context code snippets for generating the current file if filename == exclude: continue doc = await src_file_repo.get(filename=filename) diff --git a/metagpt/actions/write_code_guideline_an.py b/metagpt/actions/write_code_plan_an.py similarity index 68% rename from metagpt/actions/write_code_guideline_an.py rename to metagpt/actions/write_code_plan_an.py index 28af592c0..f3f4177e4 100644 --- a/metagpt/actions/write_code_guideline_an.py +++ b/metagpt/actions/write_code_plan_an.py @@ -3,23 +3,21 @@ """ @Time : 2023/12/26 @Author : mannaandpoem -@File : write_code_guideline_an.py +@File : write_code_plan_an.py """ from metagpt.actions.action import Action -from metagpt.actions.action_node import ActionNode, dict_to_markdown -from metagpt.config import CONFIG -from metagpt.const import CODE_GUIDELINE_FILE_REPO, CODE_GUIDELINE_PDF_FILE_REPO +from metagpt.actions.action_node import ActionNode -GUIDELINES_AND_INCREMENTAL_CHANGE = ActionNode( - key="Guidelines and Incremental Change", +Plan = ActionNode( + key="Plan", expected_type=str, - instruction="Developing comprehensive and step-by-step incremental development guideline, and Write Incremental " + instruction="Developing comprehensive and step-by-step incremental development plan, and write Incremental " "Change by making a code draft that how to implement incremental development including detailed steps based on the " "context. Note: Track incremental changes using mark of '+' or '-' for add/modify/delete code, and conforms to the " "output format of git diff", example=""" -1. Guideline for calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Additionally, implement robust error handling for the division operation to mitigate potential issues related to division by zero. +1. Plan for calculator.py: Enhance the functionality of `calculator.py` by extending it to incorporate methods for subtraction, multiplication, and division. Additionally, implement robust error handling for the division operation to mitigate potential issues related to division by zero. ```python class Calculator: self.result = number1 + number2 @@ -75,7 +73,7 @@ class Calculator: self.result = 0.0 ``` -2. Guideline for main.py: Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Then, ensure seamless integration with the overall application architecture and maintain consistency with coding standards. +2. Plan for main.py: Integrate new API endpoints for subtraction, multiplication, and division into the existing codebase of `main.py`. Then, ensure seamless integration with the overall application architecture and maintain consistency with coding standards. ```python def add_numbers(): result = calculator.add_numbers(num1, num2) @@ -106,7 +104,7 @@ def add_numbers(): ```""", ) -CODE_GUIDELINE_CONTEXT = """ +CODE_PLAN_CONTEXT = """ ## User New Requirements {user_requirement} @@ -125,14 +123,14 @@ CODE_GUIDELINE_CONTEXT = """ REFINED_CODE_TEMPLATE = """ NOTICE -Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and Guidelines and Incremental Change, ensuring the integration of new features. +Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and plan and Incremental Change, ensuring the integration of new features. # Context ## User New Requirements {user_requirement} -## Guidelines and Incremental Change -{guideline} +## Plan +{plan} ## Design {design} @@ -170,32 +168,18 @@ Role: You are a professional engineer; The main goal is to complete incremental 2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets. 3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import. 4. Follow design: YOU MUST FOLLOW "Data structures and interfaces". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design. -5. Follow Guidelines and Incremental Change: If there is any Incremental Change or Legacy Code files contain "{filename} to be rewritten", you must merge it into the code file according to the guidelines. +5. Follow plan and Incremental Change: If there is any Incremental Change or Legacy Code files contain "{filename} to be rewritten", you must merge it into the code file according to the plan. 6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 7. Before using a external variable/module, make sure you import it first. 8. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. 9. Attention: Retain content that is not related to incremental development but important for consistency and clarity.". """ -WRITE_CODE_GUIDELINE_NODE = ActionNode.from_children("WriteCodeGuideline", [GUIDELINES_AND_INCREMENTAL_CHANGE]) +WRITE_CODE_PLAN_NODE = ActionNode.from_children("WriteCodePlan", [Plan]) -class WriteCodeGuideline(Action): +class WriteCodePlan(Action): async def run(self, context): self.llm.system_prompt = "You are a professional software engineer, your primary responsibility is to " - "meticulously craft comprehensive incremental development guidelines and deliver detailed Incremental Change" - return await WRITE_CODE_GUIDELINE_NODE.fill(context=context, llm=self.llm, schema="json") - - @staticmethod - async def save_json(guideline, filename="code_guideline.json"): - await CONFIG.git_repo.new_file_repository(CODE_GUIDELINE_FILE_REPO).save( - filename=filename, content=str(guideline) - ) - - @staticmethod - async def save_md(guideline, filename="code_guideline.md"): - guideline_md = dict_to_markdown(guideline) - await CONFIG.git_repo.new_file_repository(CODE_GUIDELINE_PDF_FILE_REPO).save( - filename=filename, content=guideline_md - ) - return guideline_md + "meticulously craft comprehensive incremental development plan and deliver detailed Incremental Change" + return await WRITE_CODE_PLAN_NODE.fill(context=context, llm=self.llm, schema="json") diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index 27e4c280e..72424c037 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -14,10 +14,16 @@ from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions import WriteCode from metagpt.actions.action import Action from metagpt.config import CONFIG -from metagpt.const import DOCS_FILE_REPO, REQUIREMENT_FILENAME +from metagpt.const import ( + DOCS_FILE_REPO, + PLAN_FILENAME, + PLAN_PDF_FILE_REPO, + REQUIREMENT_FILENAME, +) from metagpt.logs import logger from metagpt.schema import CodingContext from metagpt.utils.common import CodeParser +from metagpt.utils.file_repository import FileRepository PROMPT_TEMPLATE = """ # System @@ -138,16 +144,17 @@ class WriteCodeReview(Action): async def run(self, *args, **kwargs) -> CodingContext: iterative_code = self.context.code_doc.content - # k = CONFIG.code_review_k_times or 1 - k = 1 - guideline = kwargs.get("guideline") - mode = "guide" if guideline else "normal" + k = CONFIG.code_review_k_times or 1 + plan_doc = await FileRepository.get_file(filename=PLAN_FILENAME, relative_path=PLAN_PDF_FILE_REPO) + plan = plan_doc.content if plan_doc else "" + mode = "plan" if plan else "normal" + for i in range(k): format_example = FORMAT_EXAMPLE.format(filename=self.context.code_doc.filename) task_content = self.context.task_doc.content if self.context.task_doc else "" code_context = await WriteCode.get_codes(self.context.task_doc, exclude=self.context.filename, mode=mode) - if not guideline: + if not plan: context = "\n".join( [ "## System Design\n" + str(self.context.design_doc) + "\n", @@ -156,15 +163,15 @@ class WriteCodeReview(Action): ] ) else: - requirement_doc = await CONFIG.git_repo.new_file_repository(relative_path=DOCS_FILE_REPO).get( - filename=REQUIREMENT_FILENAME + requirement_doc = await FileRepository.get_file( + filename=REQUIREMENT_FILENAME, relative_path=DOCS_FILE_REPO ) user_requirement = requirement_doc.content if requirement_doc else "" context = "\n".join( [ - "## User New Requirements\n" + str(user_requirement) + "\n", - "## Guidelines and Incremental Change\n" + guideline + "\n", + "## User New Requirements\n" + user_requirement + "\n", + "## Plan\n" + plan + "\n", "## System Design\n" + str(self.context.design_doc) + "\n", "## Tasks\n" + task_content + "\n", "## Code Files\n" + code_context + "\n", diff --git a/metagpt/const.py b/metagpt/const.py index 2ee1267d8..014193b59 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -87,19 +87,20 @@ MESSAGE_ROUTE_TO_NONE = "" REQUIREMENT_FILENAME = "requirement.txt" BUGFIX_FILENAME = "bugfix.txt" PACKAGE_REQUIREMENTS_FILENAME = "requirements.txt" +PLAN_FILENAME = "plan.json" DOCS_FILE_REPO = "docs" PRDS_FILE_REPO = "docs/prds" SYSTEM_DESIGN_FILE_REPO = "docs/system_design" TASK_FILE_REPO = "docs/tasks" -CODE_GUIDELINE_FILE_REPO = "docs/code_guideline" +PLAN_FILE_REPO = "docs/plan" COMPETITIVE_ANALYSIS_FILE_REPO = "resources/competitive_analysis" DATA_API_DESIGN_FILE_REPO = "resources/data_api_design" SEQ_FLOW_FILE_REPO = "resources/seq_flow" SYSTEM_DESIGN_PDF_FILE_REPO = "resources/system_design" PRD_PDF_FILE_REPO = "resources/prd" TASK_PDF_FILE_REPO = "resources/api_spec_and_tasks" -CODE_GUIDELINE_PDF_FILE_REPO = "resources/code_guideline" +PLAN_PDF_FILE_REPO = "resources/plan" TEST_CODES_FILE_REPO = "tests" TEST_OUTPUTS_FILE_REPO = "test_outputs" CODE_SUMMARIES_FILE_REPO = "docs/code_summaries" diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index d767b1ebf..695e9dd2a 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -23,21 +23,21 @@ import json import os from collections import defaultdict from pathlib import Path -from typing import Set +from typing import Literal, Set from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks from metagpt.actions.fix_bug import FixBug +from metagpt.actions.project_management_an import REFINED_TASK_LIST, TASK_LIST from metagpt.actions.summarize_code import SummarizeCode -from metagpt.actions.write_code_guideline_an import ( - CODE_GUIDELINE_CONTEXT, - WriteCodeGuideline, -) +from metagpt.actions.write_code_plan_an import CODE_PLAN_CONTEXT, WriteCodePlan +from metagpt.actions.write_prd_an import REFINED_REQUIREMENT_POOL, REQUIREMENT_POOL from metagpt.config import CONFIG from metagpt.const import ( - CODE_GUIDELINE_PDF_FILE_REPO, CODE_SUMMARIES_FILE_REPO, CODE_SUMMARIES_PDF_FILE_REPO, - PRDS_FILE_REPO, + PLAN_FILE_REPO, + PLAN_FILENAME, + PLAN_PDF_FILE_REPO, SYSTEM_DESIGN_FILE_REPO, TASK_FILE_REPO, ) @@ -51,6 +51,7 @@ from metagpt.schema import ( Message, ) from metagpt.utils.common import any_to_name, any_to_str, any_to_str_set +from metagpt.utils.file_repository import FileRepository IS_PASS_PROMPT = """ {context} @@ -100,9 +101,9 @@ class Engineer(Role): @staticmethod def _parse_tasks(task_msg: Document) -> list[str]: m = json.loads(task_msg.content) - return m.get("Task list") or m.get("Refined Task list") + return m.get(TASK_LIST.key) or m.get(REFINED_TASK_LIST.key) - async def _act_sp_with_cr(self, review=False, guideline=Document()) -> Set[str]: + async def _act_sp_with_cr(self, review=False, mode: Literal["normal", "plan"] = "normal") -> Set[str]: changed_files = set() src_file_repo = CONFIG.git_repo.new_file_repository(CONFIG.src_workspace) for todo in self.code_todos: @@ -113,16 +114,16 @@ class Engineer(Role): 3. Do we need other codes (currently needed)? TODO: The goal is not to need it. After clear task decomposition, based on the design idea, you should be able to write a single file without needing other codes. If you can't, it means you need a clearer definition. This is the key to writing longer code. """ - coding_context = await todo.run(guideline=guideline.content) + coding_context = await todo.run() # Code review if review: action = WriteCodeReview(context=coding_context, llm=self.llm) self._init_action_system_message(action) - coding_context = await action.run(guideline=guideline.content) + coding_context = await action.run() dependencies = {coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path} - if guideline.content: - dependencies.add(guideline.root_relative_path) + if mode == "plan": + dependencies.add(os.path.join(PLAN_PDF_FILE_REPO, PLAN_FILENAME)) await src_file_repo.save( coding_context.filename, dependencies=dependencies, @@ -155,10 +156,8 @@ class Engineer(Role): async def _act_write_code(self): if CONFIG.inc: - code_guideline = await self._write_code_guideline() - changed_files = await self._act_sp_with_cr(review=self.use_code_review, guideline=code_guideline) - else: - changed_files = await self._act_sp_with_cr(review=self.use_code_review) + await self._write_code_plan() + changed_files = await self._act_sp_with_cr(review=self.use_code_review) return Message( content="\n".join(changed_files), role=self.profile, @@ -335,41 +334,38 @@ class Engineer(Role): """AgentStore uses this attribute to display to the user what actions the current role should take.""" return self.next_todo_action - async def _write_code_guideline(self): - """Write some guidelines that guides subsequent WriteCode and WriteCodeReview""" - logger.info("Writing code guideline..") + async def _write_code_plan(self): + """Write code plan that guides subsequent WriteCode and WriteCodeReview""" + logger.info("Writing code plan..") user_requirement = str(self.rc.memory.get_by_role("Human")[0]) - contents = [] - prd = await CONFIG.git_repo.new_file_repository(PRDS_FILE_REPO).get_all() + pool_contents = [] + prd = await FileRepository.get_all_files(relative_path=PLAN_PDF_FILE_REPO) for doc in prd: prd_json = json.loads(doc.content) - product_requirement_pool = prd_json.get("Requirement Pool", prd_json.get("Refined Requirement Pool")) - contents.append(str(product_requirement_pool)) + product_requirement_pool = prd_json.get(REFINED_REQUIREMENT_POOL.key) or prd_json.get(REQUIREMENT_POOL.key) + pool_contents.append(str(product_requirement_pool)) - product_requirement_pools = "\n".join(contents) + product_requirement_pools = "\n".join(pool_contents) - design = await CONFIG.git_repo.new_file_repository(SYSTEM_DESIGN_FILE_REPO).get_all() + design = await FileRepository.get_all_files(relative_path=SYSTEM_DESIGN_FILE_REPO) design = "\n".join([doc.content for doc in design]) - tasks = await CONFIG.git_repo.new_file_repository(TASK_FILE_REPO).get_all() + tasks = await FileRepository.get_all_files(relative_path=TASK_FILE_REPO) tasks = "\n".join([doc.content for doc in tasks]) old_codes = await self.get_old_codes() - context = CODE_GUIDELINE_CONTEXT.format( + context = CODE_PLAN_CONTEXT.format( user_requirement=user_requirement, product_requirement_pools=product_requirement_pools, tasks=tasks, design=design, code=old_codes, ) - node = await WriteCodeGuideline().run(context=context) - guideline = node.instruct_content.model_dump() - await WriteCodeGuideline.save_json(guideline) - guideline_md = await WriteCodeGuideline.save_md(guideline) - - return Document(root_path=CODE_GUIDELINE_PDF_FILE_REPO, filename="code_guideline.md", content=guideline_md) + node = await WriteCodePlan().run(context=context) + plan = node.instruct_content.model_dump_json() + CONFIG.git_repo.new_file_repository(PLAN_FILE_REPO).save(filename=PLAN_FILENAME, content=plan) @staticmethod async def get_old_codes() -> str: diff --git a/metagpt/utils/mermaid.py b/metagpt/utils/mermaid.py index 25c593e6c..7762784d8 100644 --- a/metagpt/utils/mermaid.py +++ b/metagpt/utils/mermaid.py @@ -121,46 +121,6 @@ classDiagram Index --> KnowledgeBase """ -MMC1_REFINE = """ -classDiagram - class Main { - -SearchEngine search_engine - +main() str - +newMethod() str # Incremental change - } - class SearchEngine { - -Index index - -Ranking ranking - -Summary summary - +search(query: str) str - +newMethod() str # Incremental change - } - class Index { - -KnowledgeBase knowledge_base - +create_index(data: dict) - +query_index(query: str) list - +newMethod() list # Incremental change - } - class Ranking { - +rank_results(results: list) list - +newMethod() list # Incremental change - } - class Summary { - +summarize_results(results: list) str - +newMethod() str # Incremental change - } - class KnowledgeBase { - +update(data: dict) - +fetch_data(query: str) dict - +newMethod() # Incremental change - } - Main --> SearchEngine - SearchEngine --> Index - SearchEngine --> Ranking - SearchEngine --> Summary - Index --> KnowledgeBase -""" - MMC2 = """ sequenceDiagram participant M as Main @@ -180,32 +140,3 @@ sequenceDiagram S-->>SE: return summary SE-->>M: return summary """ - -MMC2_REFINE = """ -sequenceDiagram - participant M as Main - participant SE as SearchEngine - participant I as Index - participant R as Ranking - participant S as Summary - participant KB as KnowledgeBase - M->>SE: search(query) - SE->>I: query_index(query) - I->>KB: fetch_data(query) - KB-->>I: return data - I-->>SE: return results - SE->>R: rank_results(results) - R-->>SE: return ranked_results - SE->>S: summarize_results(ranked_results) - S-->>SE: return summary - SE-->>M: return summary - M->>SE: newMethod() # Incremental change - SE->>I: newMethod() # Incremental change - I->>KB: newMethod() # Incremental change - KB-->>I: newMethod() # Incremental change - SE->>R: newMethod() # Incremental change - R-->>SE: newMethod() # Incremental change - SE->>S: newMethod() # Incremental change - S-->>SE: newMethod() # Incremental change - SE-->>M: newMethod() # Incremental change -""" diff --git a/tests/data/incremental_dev_project/mock.py b/tests/data/incremental_dev_project/mock.py index a7aa2bbe4..bb3d008a1 100644 --- a/tests/data/incremental_dev_project/mock.py +++ b/tests/data/incremental_dev_project/mock.py @@ -372,8 +372,8 @@ REFINED_TASKS_JSON = { "Anything UNCLEAR": "", } -GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE = { - "Guidelines and Incremental Change": '\n1. Guideline for gui.py: Develop the GUI using Tkinter to replace the command-line interface. Start by setting up the main window and event handling. Then, add widgets for displaying the game status, results, and feedback. Implement interactive elements for difficulty selection and visualize the guess history. Finally, create animations for guess feedback and ensure responsiveness across different screen sizes.\n```python\nclass GUI:\n- pass\n+ def __init__(self):\n+ self.setup_window()\n+\n+ def setup_window(self):\n+ # Initialize the main window using Tkinter\n+ pass\n+\n+ def bind_events(self):\n+ # Bind button clicks and other events\n+ pass\n+\n+ def update_feedback(self, message: str):\n+ # Update the feedback label with the given message\n+ pass\n+\n+ def update_attempts(self, attempts: int):\n+ # Update the attempts label with the number of attempts\n+ pass\n+\n+ def update_history(self, history: list):\n+ # Update the history view with the list of past guesses\n+ pass\n+\n+ def show_difficulty_selector(self):\n+ # Show buttons or a dropdown for difficulty selection\n+ pass\n+\n+ def animate_guess_result(self, correct: bool):\n+ # Trigger an animation for correct or incorrect guesses\n+ pass\n```\n\n2. Guideline for main.py: Modify the main.py to initialize the GUI and start the event-driven game loop. Ensure that the GUI is the primary interface for user interaction.\n```python\nclass Main:\n def main(self):\n- user_interface = UI()\n- user_interface.start()\n+ graphical_user_interface = GUI()\n+ graphical_user_interface.setup_window()\n+ graphical_user_interface.bind_events()\n+ # Start the Tkinter main loop\n+ pass\n\n if __name__ == "__main__":\n main_instance = Main()\n main_instance.main()\n```\n\n3. Guideline for ui.py: Refactor ui.py to work with the new GUI class. Remove command-line interactions and delegate display and input tasks to the GUI.\n```python\nclass UI:\n- def display_message(self, message: str):\n- print(message)\n+\n+ def display_message(self, message: str):\n+ # This method will now pass the message to the GUI to display\n+ pass\n\n- def get_user_input(self, prompt: str) -> str:\n- return input(prompt)\n+\n+ def get_user_input(self, prompt: str) -> str:\n+ # This method will now trigger the GUI to get user input\n+ pass\n\n- def show_attempts(self, attempts: int):\n- print(f"Number of attempts: {attempts}")\n+\n+ def show_attempts(self, attempts: int):\n+ # This method will now update the GUI with the number of attempts\n+ pass\n\n- def show_history(self, history: list):\n- print("Guess history:")\n- for guess in history:\n- print(guess)\n+\n+ def show_history(self, history: list):\n+ # This method will now update the GUI with the guess history\n+ pass\n```\n\n4. Guideline for game.py: Ensure game.py remains mostly unchanged as it contains the core game logic. However, make minor adjustments if necessary to integrate with the new GUI.\n```python\nclass Game:\n # No changes required for now\n```\n' +PLAN_SAMPLE = { + "Plan": '\n1. Plan for gui.py: Develop the GUI using Tkinter to replace the command-line interface. Start by setting up the main window and event handling. Then, add widgets for displaying the game status, results, and feedback. Implement interactive elements for difficulty selection and visualize the guess history. Finally, create animations for guess feedback and ensure responsiveness across different screen sizes.\n```python\nclass GUI:\n- pass\n+ def __init__(self):\n+ self.setup_window()\n+\n+ def setup_window(self):\n+ # Initialize the main window using Tkinter\n+ pass\n+\n+ def bind_events(self):\n+ # Bind button clicks and other events\n+ pass\n+\n+ def update_feedback(self, message: str):\n+ # Update the feedback label with the given message\n+ pass\n+\n+ def update_attempts(self, attempts: int):\n+ # Update the attempts label with the number of attempts\n+ pass\n+\n+ def update_history(self, history: list):\n+ # Update the history view with the list of past guesses\n+ pass\n+\n+ def show_difficulty_selector(self):\n+ # Show buttons or a dropdown for difficulty selection\n+ pass\n+\n+ def animate_guess_result(self, correct: bool):\n+ # Trigger an animation for correct or incorrect guesses\n+ pass\n```\n\n2. Plan for main.py: Modify the main.py to initialize the GUI and start the event-driven game loop. Ensure that the GUI is the primary interface for user interaction.\n```python\nclass Main:\n def main(self):\n- user_interface = UI()\n- user_interface.start()\n+ graphical_user_interface = GUI()\n+ graphical_user_interface.setup_window()\n+ graphical_user_interface.bind_events()\n+ # Start the Tkinter main loop\n+ pass\n\n if __name__ == "__main__":\n main_instance = Main()\n main_instance.main()\n```\n\n3. Plan for ui.py: Refactor ui.py to work with the new GUI class. Remove command-line interactions and delegate display and input tasks to the GUI.\n```python\nclass UI:\n- def display_message(self, message: str):\n- print(message)\n+\n+ def display_message(self, message: str):\n+ # This method will now pass the message to the GUI to display\n+ pass\n\n- def get_user_input(self, prompt: str) -> str:\n- return input(prompt)\n+\n+ def get_user_input(self, prompt: str) -> str:\n+ # This method will now trigger the GUI to get user input\n+ pass\n\n- def show_attempts(self, attempts: int):\n- print(f"Number of attempts: {attempts}")\n+\n+ def show_attempts(self, attempts: int):\n+ # This method will now update the GUI with the number of attempts\n+ pass\n\n- def show_history(self, history: list):\n- print("Guess history:")\n- for guess in history:\n- print(guess)\n+\n+ def show_history(self, history: list):\n+ # This method will now update the GUI with the guess history\n+ pass\n```\n\n4. Plan for game.py: Ensure game.py remains mostly unchanged as it contains the core game logic. However, make minor adjustments if necessary to integrate with the new GUI.\n```python\nclass Game:\n # No changes required for now\n```\n' } REFINED_CODE_INPUT_SAMPLE = """ diff --git a/tests/metagpt/actions/test_write_code_guideline_an.py b/tests/metagpt/actions/test_write_code_plan_an.py similarity index 61% rename from tests/metagpt/actions/test_write_code_guideline_an.py rename to tests/metagpt/actions/test_write_code_plan_an.py index 5a4e19d57..0babab7c5 100644 --- a/tests/metagpt/actions/test_write_code_guideline_an.py +++ b/tests/metagpt/actions/test_write_code_plan_an.py @@ -3,23 +3,23 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_write_code_guideline_an.py +@File : test_write_code_plan_an.py """ import pytest from openai._models import BaseModel from metagpt.actions.action_node import ActionNode from metagpt.actions.write_code import WriteCode -from metagpt.actions.write_code_guideline_an import ( - CODE_GUIDELINE_CONTEXT, +from metagpt.actions.write_code_plan_an import ( + CODE_PLAN_CONTEXT, REFINED_CODE_TEMPLATE, - WriteCodeGuideline, + WriteCodePlan, ) from tests.data.incremental_dev_project.mock import ( DESIGN_SAMPLE, - GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE, NEW_REQUIREMENT_SAMPLE, OLD_CODE_SAMPLE, + PLAN_SAMPLE, REFINED_CODE_INPUT_SAMPLE, REFINED_CODE_SAMPLE, REFINED_DESIGN_JSON, @@ -29,30 +29,30 @@ from tests.data.incremental_dev_project.mock import ( ) -def mock_guidelines_and_incremental_change(): - return GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE +def mock_plan(): + return PLAN_SAMPLE @pytest.mark.asyncio -async def test_write_code_guideline_an(mocker): +async def test_write_code_plan_an(mocker): root = ActionNode.from_children( - "WriteCodeGuideline", [ActionNode(key="", expected_type=str, instruction="", example="")] + "WriteCodePlan", [ActionNode(key="", expected_type=str, instruction="", example="")] ) root.instruct_content = BaseModel() - root.instruct_content.model_dump = mock_guidelines_and_incremental_change - mocker.patch("metagpt.actions.write_code_guideline_an.WriteCodeGuideline.run", return_value=root) + root.instruct_content.model_dump = mock_plan + mocker.patch("metagpt.actions.write_code_plan_an.WriteCodePlan.run", return_value=root) - write_code_guideline = WriteCodeGuideline() - context = CODE_GUIDELINE_CONTEXT.format( + write_code_plan = WriteCodePlan() + context = CODE_PLAN_CONTEXT.format( user_requirement=NEW_REQUIREMENT_SAMPLE, product_requirement_pools=REFINED_PRD_JSON.get("Refined Requirement Pool", ""), design=REFINED_DESIGN_JSON, tasks=REFINED_TASKS_JSON, code=OLD_CODE_SAMPLE, ) - node = await write_code_guideline.run(context=context) + node = await write_code_plan.run(context=context) - assert "Guidelines and Incremental Change" in node.instruct_content.model_dump() + assert "Plan" in node.instruct_content.model_dump() @pytest.mark.asyncio @@ -60,7 +60,7 @@ async def test_refine_code(mocker): mocker.patch("metagpt.actions.write_code.WriteCode.write_code", return_value=REFINED_CODE_SAMPLE) prompt = REFINED_CODE_TEMPLATE.format( user_requirement=NEW_REQUIREMENT_SAMPLE, - guideline=GUIDELINES_AND_INCREMENTAL_CHANGE_SAMPLE, + plan=PLAN_SAMPLE, design=DESIGN_SAMPLE, tasks=TASKS_SAMPLE, code=REFINED_CODE_INPUT_SAMPLE, diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index ed8fc7e78..41ba785c4 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -105,7 +105,7 @@ def log_and_check_result(result, tag_name="refine"): def get_incremental_dev_result(idea, project_name, use_review=True): project_path = TEST_DATA_PATH / "incremental_dev_project" / project_name - if not os.path.exists(project_path): + if project_path.exists(): raise Exception(f"Project {project_name} not exists") check_or_create_base_tag(project_path) args = [idea, "--inc", "--project-path", project_path] From 06b35a34a6e1ea287ba78008844d0dc7d9578744 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 10:00:56 +0800 Subject: [PATCH 410/637] fix bug of save file and update prompt for gpt_v_generator tool --- metagpt/prompts/tool_types.py | 1 + metagpt/tools/libs/gpt_v_generator.py | 23 +++++++++++-------- .../schemas/image2webpage/GPTvGenerator.yml | 6 ++--- .../tools/libs/test_gpt_v_generator.py | 8 +++---- 4 files changed, 21 insertions(+), 17 deletions(-) diff --git a/metagpt/prompts/tool_types.py b/metagpt/prompts/tool_types.py index c01a80310..718eefd51 100644 --- a/metagpt/prompts/tool_types.py +++ b/metagpt/prompts/tool_types.py @@ -42,4 +42,5 @@ The current task is about evaluating a model, please note the following: IMAGE2WEBPAGE_PROMPT = """ The current task is about converting image into webpage code. please note the following: - Single-Step Code Generation: Execute the entire code generation process in a single step, encompassing HTML, CSS, and JavaScript. Avoid fragmenting the code generation into multiple separate steps to maintain consistency and simplify the development workflow. +- Save webpages: Be sure to use the save method inside Vision. """ diff --git a/metagpt/tools/libs/gpt_v_generator.py b/metagpt/tools/libs/gpt_v_generator.py index 58e547840..adc3b1051 100644 --- a/metagpt/tools/libs/gpt_v_generator.py +++ b/metagpt/tools/libs/gpt_v_generator.py @@ -3,13 +3,15 @@ """ @Time : 2024/01/12 @Author : mannaandpoem -@File : vision.py +@File : gpt_v_generator.py """ import base64 +import os from pathlib import Path import requests +from metagpt.const import DEFAULT_WORKSPACE_ROOT from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool @@ -45,7 +47,7 @@ class GPTvGenerator: def analyze_layout(self, image_path): return self.get_result(image_path, ANALYZE_LAYOUT_PROMPT) - def generate_web_pages(self, image_path): + def generate_webpages(self, image_path): layout = self.analyze_layout(image_path) prompt = GENERATE_PROMPT + "\n\n # Context\n The layout information of the sketch image is: \n" + layout result = self.get_result(image_path, prompt) @@ -81,15 +83,16 @@ class GPTvGenerator: @staticmethod def save_webpages(image_path, webpages) -> Path: - # 在当前目录下创建一个名为webpages的文件夹,用于存储html、css和js文件 - webpages_path = Path(image_path).parent / "webpages" - webpages_path.mkdir(exist_ok=True) + # 在workspace目录下,创建一个名为下webpages的文件夹,用于存储html、css和js文件 + webpages_path = DEFAULT_WORKSPACE_ROOT / "webpages" / Path(image_path).stem + os.makedirs(webpages_path, exist_ok=True) + + index_path = webpages_path / "index.html" try: - index_path = webpages_path / "index.html" index = webpages.split("```html")[1].split("```")[0] except IndexError: - raise ValueError("No html code found in the result, please check your image and try again.") + index = "No html code found in the result, please check your image and try again." + "\n" + webpages try: if "styles.css" in index: @@ -111,13 +114,13 @@ class GPTvGenerator: raise ValueError("No css or js code found in the result, please check your image and try again.") try: - with open(index_path, "w") as f: + with open(index_path, "w", encoding="utf-8") as f: f.write(index) if style_path: - with open(style_path, "w") as f: + with open(style_path, "w", encoding="utf-8") as f: f.write(style) if js_path: - with open(js_path, "w") as f: + with open(js_path, "w", encoding="utf-8") as f: f.write(js) except FileNotFoundError as e: raise FileNotFoundError(f"Cannot save the webpages to {str(webpages_path)}") from e diff --git a/metagpt/tools/schemas/image2webpage/GPTvGenerator.yml b/metagpt/tools/schemas/image2webpage/GPTvGenerator.yml index 4087f7c12..1ba2c2b08 100644 --- a/metagpt/tools/schemas/image2webpage/GPTvGenerator.yml +++ b/metagpt/tools/schemas/image2webpage/GPTvGenerator.yml @@ -1,12 +1,12 @@ GPTvGenerator: type: class - description: "Class for generating web pages at once." + description: "Class for generating webpages at once." methods: __init__: description: "Initialize Vision class with default values." - generate_web_pages: - description: "Generate web pages including all code(HTML, CSS and JavaScript) in one go based on the image." + generate_webpages: + description: "Generate webpages including all code(HTML, CSS and JavaScript) in one go based on the image." parameters: properties: image_path: diff --git a/tests/metagpt/tools/libs/test_gpt_v_generator.py b/tests/metagpt/tools/libs/test_gpt_v_generator.py index 360ca4a75..d686d38ba 100644 --- a/tests/metagpt/tools/libs/test_gpt_v_generator.py +++ b/tests/metagpt/tools/libs/test_gpt_v_generator.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/15 @Author : mannaandpoem -@File : test_vision.py +@File : test_gpt_v_generator.py """ import pytest @@ -17,14 +17,14 @@ def mock_webpages(mocker): \n\n```\n ```css\n.class { ... }\n```\n ```javascript\nfunction() { ... }\n```\n""" - mocker.patch("metagpt.tools.libs.gpt_v_generator.GPTvGenerator.generate_web_pages", return_value=mock_data) + mocker.patch("metagpt.tools.libs.gpt_v_generator.GPTvGenerator.generate_webpages", return_value=mock_data) return mocker def test_vision_generate_webpages(mock_webpages): image_path = "image.png" generator = GPTvGenerator() - rsp = generator.generate_web_pages(image_path=image_path) + rsp = generator.generate_webpages(image_path=image_path) logs.logger.info(rsp) assert "html" in rsp assert "css" in rsp @@ -34,7 +34,7 @@ def test_vision_generate_webpages(mock_webpages): def test_save_webpages(mock_webpages): image_path = "image.png" generator = GPTvGenerator() - webpages = generator.generate_web_pages(image_path) + webpages = generator.generate_webpages(image_path) webpages_dir = generator.save_webpages(image_path=image_path, webpages=webpages) logs.logger.info(webpages_dir) assert webpages_dir.exists() From 42565c39e3f51682dc0521a0a122dd1f1c978feb Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 09:08:29 +0800 Subject: [PATCH 411/637] 1. rename and modify guideline to plan 2. update prompt in ActionNode 3. add code comment 4. refactor Guideline code structure --- metagpt/actions/write_code.py | 4 ++-- metagpt/actions/write_code_review.py | 4 ++-- metagpt/const.py | 1 - metagpt/roles/engineer.py | 6 +++--- 4 files changed, 7 insertions(+), 8 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 93c7a2f65..093633a8b 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -29,8 +29,8 @@ from metagpt.const import ( BUGFIX_FILENAME, CODE_SUMMARIES_FILE_REPO, DOCS_FILE_REPO, + PLAN_FILE_REPO, PLAN_FILENAME, - PLAN_PDF_FILE_REPO, REQUIREMENT_FILENAME, TASK_FILE_REPO, TEST_OUTPUTS_FILE_REPO, @@ -107,7 +107,7 @@ class WriteCode(Action): test_doc = await FileRepository.get_file( filename="test_" + coding_context.filename + ".json", relative_path=TEST_OUTPUTS_FILE_REPO ) - plan_doc = await FileRepository.get_file(filename=PLAN_FILENAME, relative_path=PLAN_PDF_FILE_REPO) + plan_doc = await FileRepository.get_file(filename=PLAN_FILENAME, relative_path=PLAN_FILE_REPO) plan = plan_doc.content if plan_doc else "" requirement_doc = await FileRepository.get_file(filename=REQUIREMENT_FILENAME, relative_path=DOCS_FILE_REPO) summary_doc = None diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index 72424c037..2b9c7bd10 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -16,8 +16,8 @@ from metagpt.actions.action import Action from metagpt.config import CONFIG from metagpt.const import ( DOCS_FILE_REPO, + PLAN_FILE_REPO, PLAN_FILENAME, - PLAN_PDF_FILE_REPO, REQUIREMENT_FILENAME, ) from metagpt.logs import logger @@ -145,7 +145,7 @@ class WriteCodeReview(Action): async def run(self, *args, **kwargs) -> CodingContext: iterative_code = self.context.code_doc.content k = CONFIG.code_review_k_times or 1 - plan_doc = await FileRepository.get_file(filename=PLAN_FILENAME, relative_path=PLAN_PDF_FILE_REPO) + plan_doc = await FileRepository.get_file(filename=PLAN_FILENAME, relative_path=PLAN_FILE_REPO) plan = plan_doc.content if plan_doc else "" mode = "plan" if plan else "normal" diff --git a/metagpt/const.py b/metagpt/const.py index 014193b59..8f0eeef23 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -100,7 +100,6 @@ SEQ_FLOW_FILE_REPO = "resources/seq_flow" SYSTEM_DESIGN_PDF_FILE_REPO = "resources/system_design" PRD_PDF_FILE_REPO = "resources/prd" TASK_PDF_FILE_REPO = "resources/api_spec_and_tasks" -PLAN_PDF_FILE_REPO = "resources/plan" TEST_CODES_FILE_REPO = "tests" TEST_OUTPUTS_FILE_REPO = "test_outputs" CODE_SUMMARIES_FILE_REPO = "docs/code_summaries" diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 695e9dd2a..2803c8668 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -37,7 +37,7 @@ from metagpt.const import ( CODE_SUMMARIES_PDF_FILE_REPO, PLAN_FILE_REPO, PLAN_FILENAME, - PLAN_PDF_FILE_REPO, + PRDS_FILE_REPO, SYSTEM_DESIGN_FILE_REPO, TASK_FILE_REPO, ) @@ -123,7 +123,7 @@ class Engineer(Role): dependencies = {coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path} if mode == "plan": - dependencies.add(os.path.join(PLAN_PDF_FILE_REPO, PLAN_FILENAME)) + dependencies.add(os.path.join(PLAN_FILE_REPO, PLAN_FILENAME)) await src_file_repo.save( coding_context.filename, dependencies=dependencies, @@ -340,7 +340,7 @@ class Engineer(Role): user_requirement = str(self.rc.memory.get_by_role("Human")[0]) pool_contents = [] - prd = await FileRepository.get_all_files(relative_path=PLAN_PDF_FILE_REPO) + prd = await FileRepository.get_all_files(relative_path=PRDS_FILE_REPO) for doc in prd: prd_json = json.loads(doc.content) product_requirement_pool = prd_json.get(REFINED_REQUIREMENT_POOL.key) or prd_json.get(REQUIREMENT_POOL.key) From 3486b9d1d3e248bda33fee0a73629d4e92f1476c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Fri, 19 Jan 2024 11:20:05 +0800 Subject: [PATCH 412/637] feat: Maintain the original exceptions of OpenAI and HTTPX during exception handling. --- metagpt/utils/common.py | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index c7751c2af..3295603b4 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -501,7 +501,7 @@ def role_raise_decorator(func): self.rc.memory.delete(self.latest_observed_msg) # raise again to make it captured outside raise Exception(format_trackback_info(limit=None)) - except Exception: + except Exception as e: if self.latest_observed_msg: logger.warning( "There is a exception in role's execution, in order to resume, " @@ -510,6 +510,11 @@ def role_raise_decorator(func): # remove role newest observed msg to make it observed again self.rc.memory.delete(self.latest_observed_msg) # raise again to make it captured outside + last_error = e.last_attempt._exception + name = any_to_str(last_error) + if re.match(r"^openai\.", name) or re.match(r"^httpx\.", name): + raise last_error + raise Exception(format_trackback_info(limit=None)) return wrapper From 95ccd980f8ad2781c0e90e820959548e16b7eb3b Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 11:26:58 +0800 Subject: [PATCH 413/637] 1. rename and modify plan to code plan and change 2. modify name of ActionNode instance --- metagpt/actions/design_api.py | 4 +-- metagpt/actions/design_api_an.py | 6 ++-- metagpt/actions/project_management.py | 4 +-- metagpt/actions/project_management_an.py | 6 ++-- metagpt/actions/write_code.py | 30 +++++++++-------- ...an.py => write_code_plan_and_change_an.py} | 24 +++++++------- metagpt/actions/write_code_review.py | 16 +++++---- metagpt/actions/write_prd.py | 8 ++--- metagpt/actions/write_prd_an.py | 8 ++--- metagpt/const.py | 4 +-- metagpt/roles/engineer.py | 33 +++++++++++-------- tests/data/incremental_dev_project/mock.py | 2 +- tests/metagpt/actions/test_design_api_an.py | 4 +-- .../actions/test_project_management_an.py | 4 +-- .../actions/test_write_code_plan_an.py | 33 ++++++++++--------- tests/metagpt/actions/test_write_prd_an.py | 6 ++-- 16 files changed, 101 insertions(+), 91 deletions(-) rename metagpt/actions/{write_code_plan_an.py => write_code_plan_and_change_an.py} (88%) diff --git a/metagpt/actions/design_api.py b/metagpt/actions/design_api.py index 933c59b74..d24cd2ae3 100644 --- a/metagpt/actions/design_api.py +++ b/metagpt/actions/design_api.py @@ -19,7 +19,7 @@ from metagpt.actions.design_api_an import ( DESIGN_API_NODE, PROGRAM_CALL_FLOW, REFINED_DATA_STRUCTURES_AND_INTERFACES, - REFINED_DESIGN_NODES, + REFINED_DESIGN_NODE, REFINED_PROGRAM_CALL_FLOW, ) from metagpt.config import CONFIG @@ -89,7 +89,7 @@ class WriteDesign(Action): async def _merge(self, prd_doc, system_design_doc, schema=CONFIG.prompt_schema): context = NEW_REQ_TEMPLATE.format(old_design=system_design_doc.content, context=prd_doc.content) - node = await REFINED_DESIGN_NODES.fill(context=context, llm=self.llm, schema=schema) + node = await REFINED_DESIGN_NODE.fill(context=context, llm=self.llm, schema=schema) system_design_doc.content = node.instruct_content.model_dump_json() return system_design_doc diff --git a/metagpt/actions/design_api_an.py b/metagpt/actions/design_api_an.py index b872159e1..35b50ef8f 100644 --- a/metagpt/actions/design_api_an.py +++ b/metagpt/actions/design_api_an.py @@ -99,7 +99,7 @@ NODES = [ ANYTHING_UNCLEAR, ] -REFINE_NODES = [ +REFINED_NODES = [ REFINED_IMPLEMENTATION_APPROACH, REFINED_FILE_LIST, REFINED_DATA_STRUCTURES_AND_INTERFACES, @@ -108,13 +108,13 @@ REFINE_NODES = [ ] DESIGN_API_NODE = ActionNode.from_children("DesignAPI", NODES) -REFINED_DESIGN_NODES = ActionNode.from_children("RefinedDesignAPI", REFINE_NODES) +REFINED_DESIGN_NODE = ActionNode.from_children("RefinedDesignAPI", REFINED_NODES) def main(): prompt = DESIGN_API_NODE.compile(context="") logger.info(prompt) - prompt = REFINED_DESIGN_NODES.compile(context="") + prompt = REFINED_DESIGN_NODE.compile(context="") logger.info(prompt) diff --git a/metagpt/actions/project_management.py b/metagpt/actions/project_management.py index d6b351d18..448a8afcd 100644 --- a/metagpt/actions/project_management.py +++ b/metagpt/actions/project_management.py @@ -15,7 +15,7 @@ from typing import Optional from metagpt.actions import ActionOutput from metagpt.actions.action import Action -from metagpt.actions.project_management_an import PM_NODE, REFINED_PM_NODES +from metagpt.actions.project_management_an import PM_NODE, REFINED_PM_NODE from metagpt.config import CONFIG from metagpt.const import ( PACKAGE_REQUIREMENTS_FILENAME, @@ -93,7 +93,7 @@ class WriteTasks(Action): async def _merge(self, system_design_doc, task_doc, schema=CONFIG.prompt_schema) -> Document: context = NEW_REQ_TEMPLATE.format(context=system_design_doc.content, old_tasks=task_doc.content) - node = await REFINED_PM_NODES.fill(context, self.llm, schema) + node = await REFINED_PM_NODE.fill(context, self.llm, schema) task_doc.content = node.instruct_content.model_dump_json() return task_doc diff --git a/metagpt/actions/project_management_an.py b/metagpt/actions/project_management_an.py index e3d6537ab..379a23384 100644 --- a/metagpt/actions/project_management_an.py +++ b/metagpt/actions/project_management_an.py @@ -107,7 +107,7 @@ NODES = [ ANYTHING_UNCLEAR_PM, ] -REFINE_NODES = [ +REFINED_NODES = [ REQUIRED_PYTHON_PACKAGES, REQUIRED_OTHER_LANGUAGE_PACKAGES, REFINED_LOGIC_ANALYSIS, @@ -118,13 +118,13 @@ REFINE_NODES = [ ] PM_NODE = ActionNode.from_children("PM_NODE", NODES) -REFINED_PM_NODES = ActionNode.from_children("REFINED_PM_NODES", REFINE_NODES) +REFINED_PM_NODE = ActionNode.from_children("REFINED_PM_NODE", REFINED_NODES) def main(): prompt = PM_NODE.compile(context="") logger.info(prompt) - prompt = REFINED_PM_NODES.compile(context="") + prompt = REFINED_PM_NODE.compile(context="") logger.info(prompt) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 093633a8b..662524518 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -23,14 +23,14 @@ from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions.action import Action from metagpt.actions.project_management_an import REFINED_TASK_LIST, TASK_LIST -from metagpt.actions.write_code_plan_an import REFINED_CODE_TEMPLATE +from metagpt.actions.write_code_plan_and_change_an import REFINED_TEMPLATE from metagpt.config import CONFIG from metagpt.const import ( BUGFIX_FILENAME, + CODE_PLAN_AND_CHANGE_FILE_REPO, + CODE_PLAN_AND_CHANGE_FILENAME, CODE_SUMMARIES_FILE_REPO, DOCS_FILE_REPO, - PLAN_FILE_REPO, - PLAN_FILENAME, REQUIREMENT_FILENAME, TASK_FILE_REPO, TEST_OUTPUTS_FILE_REPO, @@ -107,8 +107,10 @@ class WriteCode(Action): test_doc = await FileRepository.get_file( filename="test_" + coding_context.filename + ".json", relative_path=TEST_OUTPUTS_FILE_REPO ) - plan_doc = await FileRepository.get_file(filename=PLAN_FILENAME, relative_path=PLAN_FILE_REPO) - plan = plan_doc.content if plan_doc else "" + code_plan_and_change_doc = await FileRepository.get_file( + filename=CODE_PLAN_AND_CHANGE_FILENAME, relative_path=CODE_PLAN_AND_CHANGE_FILE_REPO + ) + code_plan_and_change = code_plan_and_change_doc.content if code_plan_and_change_doc else "" requirement_doc = await FileRepository.get_file(filename=REQUIREMENT_FILENAME, relative_path=DOCS_FILE_REPO) summary_doc = None if coding_context.design_doc and coding_context.design_doc.filename: @@ -122,15 +124,15 @@ class WriteCode(Action): if bug_feedback: code_context = coding_context.code_doc.content - elif plan: - code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename, mode="plan") + elif code_plan_and_change: + code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename, mode="guide") else: code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename) - if plan: - prompt = REFINED_CODE_TEMPLATE.format( + if code_plan_and_change: + prompt = REFINED_TEMPLATE.format( user_requirement=requirement_doc.content if requirement_doc else "", - plan=plan, + code_plan_and_change=code_plan_and_change, design=coding_context.design_doc.content if coding_context.design_doc else "", tasks=coding_context.task_doc.content if coding_context.task_doc else "", code=code_context, @@ -159,14 +161,14 @@ class WriteCode(Action): return coding_context @staticmethod - async def get_codes(task_doc: Document, exclude: str, mode: Literal["normal", "plan"] = "normal") -> str: + async def get_codes(task_doc: Document, exclude: str, mode: Literal["normal", "guide"] = "normal") -> str: """ Get code snippets based on different modes. Attributes: task_doc (Document): Document object of the task file. exclude (str): Specifies the filename to be excluded from the code snippets. - mode (str): Specifies the mode, either "normal" or "plan" (default is "normal"). + mode (str): Specifies the mode, either "normal" or "guide" (default is "normal"). Returns: str: Code snippets. @@ -175,7 +177,7 @@ class WriteCode(Action): If mode is set to "normal", it returns code snippets for the regular coding phase, i.e., all the code generated before writing the current file. - If mode is set to "plan", it returns code snippets for incremental development, + If mode is set to "guide", it returns code snippets for generating the code plan and change, building upon the existing code in the "normal" mode and adding code for the current file's older versions. """ if not task_doc: @@ -187,7 +189,7 @@ class WriteCode(Action): codes = [] src_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.src_workspace) - if mode == "plan": + if mode == "guide": src_files = src_file_repo.all_files old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) old_files = old_file_repo.all_files diff --git a/metagpt/actions/write_code_plan_an.py b/metagpt/actions/write_code_plan_and_change_an.py similarity index 88% rename from metagpt/actions/write_code_plan_an.py rename to metagpt/actions/write_code_plan_and_change_an.py index f3f4177e4..1722855d2 100644 --- a/metagpt/actions/write_code_plan_an.py +++ b/metagpt/actions/write_code_plan_and_change_an.py @@ -3,14 +3,14 @@ """ @Time : 2023/12/26 @Author : mannaandpoem -@File : write_code_plan_an.py +@File : write_code_plan_and_change_an.py """ from metagpt.actions.action import Action from metagpt.actions.action_node import ActionNode -Plan = ActionNode( - key="Plan", +CODE_PLAN_AND_CHANGE = ActionNode( + key="Code Plan And Change", expected_type=str, instruction="Developing comprehensive and step-by-step incremental development plan, and write Incremental " "Change by making a code draft that how to implement incremental development including detailed steps based on the " @@ -104,7 +104,7 @@ def add_numbers(): ```""", ) -CODE_PLAN_CONTEXT = """ +CODE_PLAN_AND_CHANGE_CONTEXT = """ ## User New Requirements {user_requirement} @@ -121,7 +121,7 @@ CODE_PLAN_CONTEXT = """ {code} """ -REFINED_CODE_TEMPLATE = """ +REFINED_TEMPLATE = """ NOTICE Role: You are a professional engineer; The main goal is to complete incremental development by combining legacy code and plan and Incremental Change, ensuring the integration of new features. @@ -129,8 +129,8 @@ Role: You are a professional engineer; The main goal is to complete incremental ## User New Requirements {user_requirement} -## Plan -{plan} +## Code Plan And Change +{code_plan_and_change} ## Design {design} @@ -168,18 +168,18 @@ Role: You are a professional engineer; The main goal is to complete incremental 2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets. 3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import. 4. Follow design: YOU MUST FOLLOW "Data structures and interfaces". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design. -5. Follow plan and Incremental Change: If there is any Incremental Change or Legacy Code files contain "{filename} to be rewritten", you must merge it into the code file according to the plan. +5. Follow Code Plan And Change: If there is any Incremental Change or Legacy Code files contain "{filename} to be rewritten", you must merge it into the code file according to the plan. 6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 7. Before using a external variable/module, make sure you import it first. 8. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. 9. Attention: Retain content that is not related to incremental development but important for consistency and clarity.". """ -WRITE_CODE_PLAN_NODE = ActionNode.from_children("WriteCodePlan", [Plan]) +WRITE_CODE_PLAN_AND_CHANGE_NODE = ActionNode.from_children("WriteCodePlanAndChange", [CODE_PLAN_AND_CHANGE]) -class WriteCodePlan(Action): +class WriteCodePlanAndChange(Action): async def run(self, context): self.llm.system_prompt = "You are a professional software engineer, your primary responsibility is to " - "meticulously craft comprehensive incremental development plan and deliver detailed Incremental Change" - return await WRITE_CODE_PLAN_NODE.fill(context=context, llm=self.llm, schema="json") + "meticulously craft comprehensive incremental development plan and deliver detailed incremental change" + return await WRITE_CODE_PLAN_AND_CHANGE_NODE.fill(context=context, llm=self.llm, schema="json") diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index 2b9c7bd10..6dbcb03fa 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -15,9 +15,9 @@ from metagpt.actions import WriteCode from metagpt.actions.action import Action from metagpt.config import CONFIG from metagpt.const import ( + CODE_PLAN_AND_CHANGE_FILE_REPO, + CODE_PLAN_AND_CHANGE_FILENAME, DOCS_FILE_REPO, - PLAN_FILE_REPO, - PLAN_FILENAME, REQUIREMENT_FILENAME, ) from metagpt.logs import logger @@ -145,16 +145,18 @@ class WriteCodeReview(Action): async def run(self, *args, **kwargs) -> CodingContext: iterative_code = self.context.code_doc.content k = CONFIG.code_review_k_times or 1 - plan_doc = await FileRepository.get_file(filename=PLAN_FILENAME, relative_path=PLAN_FILE_REPO) - plan = plan_doc.content if plan_doc else "" - mode = "plan" if plan else "normal" + code_plan_and_change_doc = await FileRepository.get_file( + filename=CODE_PLAN_AND_CHANGE_FILENAME, relative_path=CODE_PLAN_AND_CHANGE_FILE_REPO + ) + code_plan_and_change = code_plan_and_change_doc.content if code_plan_and_change_doc else "" + mode = "guide" if code_plan_and_change else "normal" for i in range(k): format_example = FORMAT_EXAMPLE.format(filename=self.context.code_doc.filename) task_content = self.context.task_doc.content if self.context.task_doc else "" code_context = await WriteCode.get_codes(self.context.task_doc, exclude=self.context.filename, mode=mode) - if not plan: + if not code_plan_and_change: context = "\n".join( [ "## System Design\n" + str(self.context.design_doc) + "\n", @@ -171,7 +173,7 @@ class WriteCodeReview(Action): context = "\n".join( [ "## User New Requirements\n" + user_requirement + "\n", - "## Plan\n" + plan + "\n", + "## Code Plan And Change\n" + code_plan_and_change + "\n", "## System Design\n" + str(self.context.design_doc) + "\n", "## Tasks\n" + task_content + "\n", "## Code Files\n" + code_context + "\n", diff --git a/metagpt/actions/write_prd.py b/metagpt/actions/write_prd.py index e12c1e1ec..c92749da0 100644 --- a/metagpt/actions/write_prd.py +++ b/metagpt/actions/write_prd.py @@ -23,8 +23,8 @@ from metagpt.actions.fix_bug import FixBug from metagpt.actions.write_prd_an import ( COMPETITIVE_QUADRANT_CHART, PROJECT_NAME, - REFINE_PRD_NODE, - REFINE_PRD_TEMPLATE, + REFINED_PRD_NODE, + REFINED_TEMPLATE, WP_IS_RELATIVE_NODE, WP_ISSUE_TYPE_NODE, WRITE_PRD_NODE, @@ -135,12 +135,12 @@ class WritePRD(Action): async def _merge(self, new_requirement_doc, prd_doc, schema=CONFIG.prompt_schema) -> Document: if not CONFIG.project_name: CONFIG.project_name = Path(CONFIG.project_path).name - prompt = REFINE_PRD_TEMPLATE.format( + prompt = REFINED_TEMPLATE.format( requirements=new_requirement_doc.content, old_prd=prd_doc.content, project_name=CONFIG.project_name, ) - node = await REFINE_PRD_NODE.fill(context=prompt, llm=self.llm, schema=schema) + node = await REFINED_PRD_NODE.fill(context=prompt, llm=self.llm, schema=schema) prd_doc.content = node.instruct_content.model_dump_json() await self._rename_workspace(node) return prd_doc diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index 4830076e3..c65860822 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -186,7 +186,7 @@ REASON = ActionNode( key="reason", expected_type=str, instruction="Explain the reasoning process from question to answer", example="..." ) -REFINE_PRD_TEMPLATE = """ +REFINED_TEMPLATE = """ ### Project Name {project_name} @@ -216,7 +216,7 @@ NODES = [ ANYTHING_UNCLEAR, ] -REFINE_NODES = [ +REFINED_NODES = [ LANGUAGE, PROGRAMMING_LANGUAGE, REFINED_REQUIREMENTS, @@ -232,7 +232,7 @@ REFINE_NODES = [ ] WRITE_PRD_NODE = ActionNode.from_children("WritePRD", NODES) -REFINE_PRD_NODE = ActionNode.from_children("RefinePRD", REFINE_NODES) +REFINED_PRD_NODE = ActionNode.from_children("RefinedPRD", REFINED_NODES) WP_ISSUE_TYPE_NODE = ActionNode.from_children("WP_ISSUE_TYPE", [ISSUE_TYPE, REASON]) WP_IS_RELATIVE_NODE = ActionNode.from_children("WP_IS_RELATIVE", [IS_RELATIVE, REASON]) @@ -240,7 +240,7 @@ WP_IS_RELATIVE_NODE = ActionNode.from_children("WP_IS_RELATIVE", [IS_RELATIVE, R def main(): prompt = WRITE_PRD_NODE.compile(context="") logger.info(prompt) - prompt = REFINE_PRD_NODE.compile(context="") + prompt = REFINED_PRD_NODE.compile(context="") logger.info(prompt) diff --git a/metagpt/const.py b/metagpt/const.py index 8f0eeef23..fc3d54296 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -87,13 +87,13 @@ MESSAGE_ROUTE_TO_NONE = "" REQUIREMENT_FILENAME = "requirement.txt" BUGFIX_FILENAME = "bugfix.txt" PACKAGE_REQUIREMENTS_FILENAME = "requirements.txt" -PLAN_FILENAME = "plan.json" +CODE_PLAN_AND_CHANGE_FILENAME = "code_plan_and_change.json" DOCS_FILE_REPO = "docs" PRDS_FILE_REPO = "docs/prds" SYSTEM_DESIGN_FILE_REPO = "docs/system_design" TASK_FILE_REPO = "docs/tasks" -PLAN_FILE_REPO = "docs/plan" +CODE_PLAN_AND_CHANGE_FILE_REPO = "docs/code_plan_and_change" COMPETITIVE_ANALYSIS_FILE_REPO = "resources/competitive_analysis" DATA_API_DESIGN_FILE_REPO = "resources/data_api_design" SEQ_FLOW_FILE_REPO = "resources/seq_flow" diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 2803c8668..861c5435e 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -29,14 +29,17 @@ from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks from metagpt.actions.fix_bug import FixBug from metagpt.actions.project_management_an import REFINED_TASK_LIST, TASK_LIST from metagpt.actions.summarize_code import SummarizeCode -from metagpt.actions.write_code_plan_an import CODE_PLAN_CONTEXT, WriteCodePlan +from metagpt.actions.write_code_plan_and_change_an import ( + CODE_PLAN_AND_CHANGE_CONTEXT, + WriteCodePlanAndChange, +) from metagpt.actions.write_prd_an import REFINED_REQUIREMENT_POOL, REQUIREMENT_POOL from metagpt.config import CONFIG from metagpt.const import ( + CODE_PLAN_AND_CHANGE_FILE_REPO, + CODE_PLAN_AND_CHANGE_FILENAME, CODE_SUMMARIES_FILE_REPO, CODE_SUMMARIES_PDF_FILE_REPO, - PLAN_FILE_REPO, - PLAN_FILENAME, PRDS_FILE_REPO, SYSTEM_DESIGN_FILE_REPO, TASK_FILE_REPO, @@ -103,7 +106,7 @@ class Engineer(Role): m = json.loads(task_msg.content) return m.get(TASK_LIST.key) or m.get(REFINED_TASK_LIST.key) - async def _act_sp_with_cr(self, review=False, mode: Literal["normal", "plan"] = "normal") -> Set[str]: + async def _act_sp_with_cr(self, review=False, mode: Literal["normal", "guide"] = "normal") -> Set[str]: changed_files = set() src_file_repo = CONFIG.git_repo.new_file_repository(CONFIG.src_workspace) for todo in self.code_todos: @@ -122,8 +125,8 @@ class Engineer(Role): coding_context = await action.run() dependencies = {coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path} - if mode == "plan": - dependencies.add(os.path.join(PLAN_FILE_REPO, PLAN_FILENAME)) + if mode == "guide": + dependencies.add(os.path.join(CODE_PLAN_AND_CHANGE_FILE_REPO, CODE_PLAN_AND_CHANGE_FILENAME)) await src_file_repo.save( coding_context.filename, dependencies=dependencies, @@ -156,7 +159,7 @@ class Engineer(Role): async def _act_write_code(self): if CONFIG.inc: - await self._write_code_plan() + await self._write_code_plan_and_change() changed_files = await self._act_sp_with_cr(review=self.use_code_review) return Message( content="\n".join(changed_files), @@ -334,9 +337,9 @@ class Engineer(Role): """AgentStore uses this attribute to display to the user what actions the current role should take.""" return self.next_todo_action - async def _write_code_plan(self): - """Write code plan that guides subsequent WriteCode and WriteCodeReview""" - logger.info("Writing code plan..") + async def _write_code_plan_and_change(self): + """Write code plan and change that guides subsequent WriteCode and WriteCodeReview""" + logger.info("Writing code plan and change..") user_requirement = str(self.rc.memory.get_by_role("Human")[0]) pool_contents = [] @@ -356,16 +359,18 @@ class Engineer(Role): old_codes = await self.get_old_codes() - context = CODE_PLAN_CONTEXT.format( + context = CODE_PLAN_AND_CHANGE_CONTEXT.format( user_requirement=user_requirement, product_requirement_pools=product_requirement_pools, tasks=tasks, design=design, code=old_codes, ) - node = await WriteCodePlan().run(context=context) - plan = node.instruct_content.model_dump_json() - CONFIG.git_repo.new_file_repository(PLAN_FILE_REPO).save(filename=PLAN_FILENAME, content=plan) + node = await WriteCodePlanAndChange().run(context=context) + code_plan_and_change = node.instruct_content.model_dump_json() + CONFIG.git_repo.new_file_repository(CODE_PLAN_AND_CHANGE_FILE_REPO).save( + filename=CODE_PLAN_AND_CHANGE_FILENAME, content=code_plan_and_change + ) @staticmethod async def get_old_codes() -> str: diff --git a/tests/data/incremental_dev_project/mock.py b/tests/data/incremental_dev_project/mock.py index bb3d008a1..5c5191cf2 100644 --- a/tests/data/incremental_dev_project/mock.py +++ b/tests/data/incremental_dev_project/mock.py @@ -372,7 +372,7 @@ REFINED_TASKS_JSON = { "Anything UNCLEAR": "", } -PLAN_SAMPLE = { +CODE_PLAN_AND_CHANGE_SAMPLE = { "Plan": '\n1. Plan for gui.py: Develop the GUI using Tkinter to replace the command-line interface. Start by setting up the main window and event handling. Then, add widgets for displaying the game status, results, and feedback. Implement interactive elements for difficulty selection and visualize the guess history. Finally, create animations for guess feedback and ensure responsiveness across different screen sizes.\n```python\nclass GUI:\n- pass\n+ def __init__(self):\n+ self.setup_window()\n+\n+ def setup_window(self):\n+ # Initialize the main window using Tkinter\n+ pass\n+\n+ def bind_events(self):\n+ # Bind button clicks and other events\n+ pass\n+\n+ def update_feedback(self, message: str):\n+ # Update the feedback label with the given message\n+ pass\n+\n+ def update_attempts(self, attempts: int):\n+ # Update the attempts label with the number of attempts\n+ pass\n+\n+ def update_history(self, history: list):\n+ # Update the history view with the list of past guesses\n+ pass\n+\n+ def show_difficulty_selector(self):\n+ # Show buttons or a dropdown for difficulty selection\n+ pass\n+\n+ def animate_guess_result(self, correct: bool):\n+ # Trigger an animation for correct or incorrect guesses\n+ pass\n```\n\n2. Plan for main.py: Modify the main.py to initialize the GUI and start the event-driven game loop. Ensure that the GUI is the primary interface for user interaction.\n```python\nclass Main:\n def main(self):\n- user_interface = UI()\n- user_interface.start()\n+ graphical_user_interface = GUI()\n+ graphical_user_interface.setup_window()\n+ graphical_user_interface.bind_events()\n+ # Start the Tkinter main loop\n+ pass\n\n if __name__ == "__main__":\n main_instance = Main()\n main_instance.main()\n```\n\n3. Plan for ui.py: Refactor ui.py to work with the new GUI class. Remove command-line interactions and delegate display and input tasks to the GUI.\n```python\nclass UI:\n- def display_message(self, message: str):\n- print(message)\n+\n+ def display_message(self, message: str):\n+ # This method will now pass the message to the GUI to display\n+ pass\n\n- def get_user_input(self, prompt: str) -> str:\n- return input(prompt)\n+\n+ def get_user_input(self, prompt: str) -> str:\n+ # This method will now trigger the GUI to get user input\n+ pass\n\n- def show_attempts(self, attempts: int):\n- print(f"Number of attempts: {attempts}")\n+\n+ def show_attempts(self, attempts: int):\n+ # This method will now update the GUI with the number of attempts\n+ pass\n\n- def show_history(self, history: list):\n- print("Guess history:")\n- for guess in history:\n- print(guess)\n+\n+ def show_history(self, history: list):\n+ # This method will now update the GUI with the guess history\n+ pass\n```\n\n4. Plan for game.py: Ensure game.py remains mostly unchanged as it contains the core game logic. However, make minor adjustments if necessary to integrate with the new GUI.\n```python\nclass Game:\n # No changes required for now\n```\n' } diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index 39de2a595..fcd2ef666 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -10,7 +10,7 @@ from openai._models import BaseModel from metagpt.actions.action_node import ActionNode, dict_to_markdown from metagpt.actions.design_api import NEW_REQ_TEMPLATE -from metagpt.actions.design_api_an import REFINED_DESIGN_NODES +from metagpt.actions.design_api_an import REFINED_DESIGN_NODE from metagpt.llm import LLM from tests.data.incremental_dev_project.mock import ( DESIGN_SAMPLE, @@ -38,7 +38,7 @@ async def test_write_design_an(mocker): mocker.patch("metagpt.actions.design_api_an.REFINED_DESIGN_NODES.fill", return_value=root) prompt = NEW_REQ_TEMPLATE.format(old_design=DESIGN_SAMPLE, context=dict_to_markdown(REFINED_PRD_JSON)) - node = await REFINED_DESIGN_NODES.fill(prompt, llm) + node = await REFINED_DESIGN_NODE.fill(prompt, llm) assert "Refined Implementation Approach" in node.instruct_content.model_dump() assert "Refined File list" in node.instruct_content.model_dump() diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index 50dc47067..e230151da 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -10,7 +10,7 @@ from openai._models import BaseModel from metagpt.actions.action_node import ActionNode, dict_to_markdown from metagpt.actions.project_management import NEW_REQ_TEMPLATE -from metagpt.actions.project_management_an import REFINED_PM_NODES +from metagpt.actions.project_management_an import REFINED_PM_NODE from metagpt.llm import LLM from tests.data.incremental_dev_project.mock import ( REFINED_DESIGN_JSON, @@ -38,7 +38,7 @@ async def test_project_management_an(mocker): mocker.patch("metagpt.actions.project_management_an.REFINED_PM_NODES.fill", return_value=root) prompt = NEW_REQ_TEMPLATE.format(old_tasks=TASKS_SAMPLE, context=dict_to_markdown(REFINED_DESIGN_JSON)) - node = await REFINED_PM_NODES.fill(prompt, llm) + node = await REFINED_PM_NODE.fill(prompt, llm) assert "Refined Logic Analysis" in node.instruct_content.model_dump() assert "Refined Task list" in node.instruct_content.model_dump() diff --git a/tests/metagpt/actions/test_write_code_plan_an.py b/tests/metagpt/actions/test_write_code_plan_an.py index 0babab7c5..13bda7cc1 100644 --- a/tests/metagpt/actions/test_write_code_plan_an.py +++ b/tests/metagpt/actions/test_write_code_plan_an.py @@ -10,16 +10,17 @@ from openai._models import BaseModel from metagpt.actions.action_node import ActionNode from metagpt.actions.write_code import WriteCode -from metagpt.actions.write_code_plan_an import ( - CODE_PLAN_CONTEXT, - REFINED_CODE_TEMPLATE, - WriteCodePlan, +from metagpt.actions.write_code_plan_and_change_an import ( + CODE_PLAN_AND_CHANGE_CONTEXT, + REFINED_TEMPLATE, + WriteCodePlanAndChange, ) +from metagpt.actions.write_prd_an import REQUIREMENT_POOL from tests.data.incremental_dev_project.mock import ( + CODE_PLAN_AND_CHANGE_SAMPLE, DESIGN_SAMPLE, NEW_REQUIREMENT_SAMPLE, OLD_CODE_SAMPLE, - PLAN_SAMPLE, REFINED_CODE_INPUT_SAMPLE, REFINED_CODE_SAMPLE, REFINED_DESIGN_JSON, @@ -29,23 +30,23 @@ from tests.data.incremental_dev_project.mock import ( ) -def mock_plan(): - return PLAN_SAMPLE +def mock_code_plan_and_change(): + return CODE_PLAN_AND_CHANGE_SAMPLE @pytest.mark.asyncio async def test_write_code_plan_an(mocker): root = ActionNode.from_children( - "WriteCodePlan", [ActionNode(key="", expected_type=str, instruction="", example="")] + "WriteCodePlanAndChange", [ActionNode(key="", expected_type=str, instruction="", example="")] ) root.instruct_content = BaseModel() - root.instruct_content.model_dump = mock_plan - mocker.patch("metagpt.actions.write_code_plan_an.WriteCodePlan.run", return_value=root) + root.instruct_content.model_dump = mock_code_plan_and_change + mocker.patch("metagpt.actions.write_code_plan_an.WriteCodePlanAndChange.run", return_value=root) - write_code_plan = WriteCodePlan() - context = CODE_PLAN_CONTEXT.format( + write_code_plan = WriteCodePlanAndChange() + context = CODE_PLAN_AND_CHANGE_CONTEXT.format( user_requirement=NEW_REQUIREMENT_SAMPLE, - product_requirement_pools=REFINED_PRD_JSON.get("Refined Requirement Pool", ""), + product_requirement_pools=REFINED_PRD_JSON.get(REQUIREMENT_POOL.key), design=REFINED_DESIGN_JSON, tasks=REFINED_TASKS_JSON, code=OLD_CODE_SAMPLE, @@ -57,10 +58,10 @@ async def test_write_code_plan_an(mocker): @pytest.mark.asyncio async def test_refine_code(mocker): - mocker.patch("metagpt.actions.write_code.WriteCode.write_code", return_value=REFINED_CODE_SAMPLE) - prompt = REFINED_CODE_TEMPLATE.format( + mocker.patch("metagpt.actions.write_code.WriteCodePlanAndChange.write_code", return_value=REFINED_CODE_SAMPLE) + prompt = REFINED_TEMPLATE.format( user_requirement=NEW_REQUIREMENT_SAMPLE, - plan=PLAN_SAMPLE, + code_plan_and_change=CODE_PLAN_AND_CHANGE_SAMPLE, design=DESIGN_SAMPLE, tasks=TASKS_SAMPLE, code=REFINED_CODE_INPUT_SAMPLE, diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index 1fdaa75c2..806f88d36 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -9,7 +9,7 @@ import pytest from openai._models import BaseModel from metagpt.actions.action_node import ActionNode -from metagpt.actions.write_prd_an import REFINE_PRD_NODE, REFINE_PRD_TEMPLATE +from metagpt.actions.write_prd_an import REFINED_PRD_NODE, REFINED_TEMPLATE from metagpt.llm import LLM from tests.data.incremental_dev_project.mock import ( NEW_REQUIREMENT_SAMPLE, @@ -34,12 +34,12 @@ async def test_write_prd_an(mocker): root.instruct_content.model_dump = mock_refined_prd_json mocker.patch("metagpt.actions.write_prd_an.REFINE_PRD_NODE.fill", return_value=root) - prompt = REFINE_PRD_TEMPLATE.format( + prompt = REFINED_TEMPLATE.format( requirements=NEW_REQUIREMENT_SAMPLE, old_prd=PRD_SAMPLE, project_name="", ) - node = await REFINE_PRD_NODE.fill(prompt, llm) + node = await REFINED_PRD_NODE.fill(prompt, llm) assert "Refined Requirements" in node.instruct_content.model_dump() assert "Refined Product Goals" in node.instruct_content.model_dump() From b78cc3c1cab491a81bc59ddeff2f2ba96d3bc650 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Fri, 19 Jan 2024 11:49:35 +0800 Subject: [PATCH 414/637] feat: Maintain the original exceptions of OpenAI and HTTPX during exception handling. --- metagpt/utils/common.py | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 3295603b4..3102158c2 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -28,7 +28,7 @@ from typing import Any, List, Tuple, Union import aiofiles import loguru from pydantic_core import to_jsonable_python -from tenacity import RetryCallState, _utils +from tenacity import RetryCallState, RetryError, _utils from metagpt.const import MESSAGE_ROUTE_TO_ALL from metagpt.logs import logger @@ -510,10 +510,11 @@ def role_raise_decorator(func): # remove role newest observed msg to make it observed again self.rc.memory.delete(self.latest_observed_msg) # raise again to make it captured outside - last_error = e.last_attempt._exception - name = any_to_str(last_error) - if re.match(r"^openai\.", name) or re.match(r"^httpx\.", name): - raise last_error + if isinstance(e, RetryError): + last_error = e.last_attempt._exception + name = any_to_str(last_error) + if re.match(r"^openai\.", name) or re.match(r"^httpx\.", name): + raise last_error raise Exception(format_trackback_info(limit=None)) From cd919aa71bf9e1305edd9515025931acef6a9dfa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Fri, 19 Jan 2024 11:51:02 +0800 Subject: [PATCH 415/637] feat: +ver --- setup.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/setup.py b/setup.py index ca8bb3980..cc8112ba9 100644 --- a/setup.py +++ b/setup.py @@ -57,7 +57,7 @@ extras_require["dev"] = (["pylint~=3.0.3", "black~=23.3.0", "isort~=5.12.0", "pr setup( name="metagpt", - version="0.6.5", + version="0.6.6", description="The Multi-Agent Framework", long_description=long_description, long_description_content_type="text/markdown", From 1959743d0beed9bfeed7074ac346e6ba44efc2db Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 13:34:47 +0800 Subject: [PATCH 416/637] update write_code_plan_and_change_an.py and add it to _think and _act process --- .../actions/write_code_plan_and_change_an.py | 32 ++++- metagpt/const.py | 1 + metagpt/roles/engineer.py | 111 ++++++++++-------- metagpt/schema.py | 8 ++ .../actions/test_write_code_plan_an.py | 5 +- 5 files changed, 101 insertions(+), 56 deletions(-) diff --git a/metagpt/actions/write_code_plan_and_change_an.py b/metagpt/actions/write_code_plan_and_change_an.py index 1722855d2..151b2dc25 100644 --- a/metagpt/actions/write_code_plan_and_change_an.py +++ b/metagpt/actions/write_code_plan_and_change_an.py @@ -5,9 +5,14 @@ @Author : mannaandpoem @File : write_code_plan_and_change_an.py """ +import os + +from pydantic import Field from metagpt.actions.action import Action from metagpt.actions.action_node import ActionNode +from metagpt.config import CONFIG +from metagpt.schema import CodePlanAndChangeContext CODE_PLAN_AND_CHANGE = ActionNode( key="Code Plan And Change", @@ -106,10 +111,10 @@ def add_numbers(): CODE_PLAN_AND_CHANGE_CONTEXT = """ ## User New Requirements -{user_requirement} +{requirement} -## Product Requirement Pool -{product_requirement_pools} +## PRD +{prd} ## Design {design} @@ -179,7 +184,26 @@ WRITE_CODE_PLAN_AND_CHANGE_NODE = ActionNode.from_children("WriteCodePlanAndChan class WriteCodePlanAndChange(Action): - async def run(self, context): + name: str = "WriteCodePlanAndChange" + context: CodePlanAndChangeContext = Field(default_factory=CodePlanAndChangeContext) + + async def run(self, *args, **kwargs): self.llm.system_prompt = "You are a professional software engineer, your primary responsibility is to " "meticulously craft comprehensive incremental development plan and deliver detailed incremental change" + requirement = self.context.requirement_doc.content + prd = "\n".join([doc.content for doc in self.context.prd_docs]) + design = "\n".join([doc.content for doc in self.context.design_docs]) + tasks = "\n".join([doc.content for doc in self.context.task_docs]) + code_text = self.get_old_codes() + context = CODE_PLAN_AND_CHANGE_CONTEXT.format( + requirement=requirement, prd=prd, design=design, tasks=tasks, code=code_text + ) return await WRITE_CODE_PLAN_AND_CHANGE_NODE.fill(context=context, llm=self.llm, schema="json") + + @staticmethod + async def get_old_codes() -> str: + CONFIG.old_workspace = CONFIG.git_repo.workdir / os.path.basename(CONFIG.project_path) + old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) + old_codes = await old_file_repo.get_all() + codes = [f"----- {code.filename}\n```{code.content}```" for code in old_codes] + return "\n".join(codes) diff --git a/metagpt/const.py b/metagpt/const.py index fc3d54296..7011f9c6f 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -100,6 +100,7 @@ SEQ_FLOW_FILE_REPO = "resources/seq_flow" SYSTEM_DESIGN_PDF_FILE_REPO = "resources/system_design" PRD_PDF_FILE_REPO = "resources/prd" TASK_PDF_FILE_REPO = "resources/api_spec_and_tasks" +CODE_PLAN_AND_CHANGE_PDF_FILE_REPO = "resources/code_plan_and_change" TEST_CODES_FILE_REPO = "tests" TEST_OUTPUTS_FILE_REPO = "test_outputs" CODE_SUMMARIES_FILE_REPO = "docs/code_summaries" diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 861c5435e..ac6e3b720 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -29,24 +29,24 @@ from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks from metagpt.actions.fix_bug import FixBug from metagpt.actions.project_management_an import REFINED_TASK_LIST, TASK_LIST from metagpt.actions.summarize_code import SummarizeCode -from metagpt.actions.write_code_plan_and_change_an import ( - CODE_PLAN_AND_CHANGE_CONTEXT, - WriteCodePlanAndChange, -) -from metagpt.actions.write_prd_an import REFINED_REQUIREMENT_POOL, REQUIREMENT_POOL +from metagpt.actions.write_code_plan_and_change_an import WriteCodePlanAndChange from metagpt.config import CONFIG from metagpt.const import ( CODE_PLAN_AND_CHANGE_FILE_REPO, CODE_PLAN_AND_CHANGE_FILENAME, + CODE_PLAN_AND_CHANGE_PDF_FILE_REPO, CODE_SUMMARIES_FILE_REPO, CODE_SUMMARIES_PDF_FILE_REPO, + DOCS_FILE_REPO, PRDS_FILE_REPO, + REQUIREMENT_FILENAME, SYSTEM_DESIGN_FILE_REPO, TASK_FILE_REPO, ) from metagpt.logs import logger from metagpt.roles import Role from metagpt.schema import ( + CodePlanAndChangeContext, CodeSummarizeContext, CodingContext, Document, @@ -149,6 +149,9 @@ class Engineer(Role): """Determines the mode of action based on whether code review is used.""" if self.rc.todo is None: return None + if isinstance(self.rc.todo, WriteCodePlanAndChange): + self.next_todo_action = any_to_name(WriteCode) + return await self._act_code_plan_and_change() if isinstance(self.rc.todo, WriteCode): self.next_todo_action = any_to_name(SummarizeCode) return await self._act_write_code() @@ -212,6 +215,40 @@ class Engineer(Role): content=json.dumps(tasks), role=self.profile, cause_by=SummarizeCode, send_to=self, sent_from=self ) + async def _act_code_plan_and_change(self): + """Write code plan and change that guides subsequent WriteCode and WriteCodeReview""" + logger.info("Writing code plan and change..") + code_plan_and_change_file_repo = CONFIG.git_repo.new_file_repository(CODE_PLAN_AND_CHANGE_FILE_REPO) + code_plan_and_change_pdf_file_repo = CONFIG.git_repo.new_file_repository(CODE_PLAN_AND_CHANGE_PDF_FILE_REPO) + + node = await self.rc.todo.run() + code_plan_and_change = node.instruct_content.model_dump_json() + + dependencies = { + self.rc.todo.context.requirement_filename, + self.rc.todo.context.prd_filename, + self.rc.todo.context.design_filename, + self.rc.todo.context.task_filename, + } + + code_plan_and_change_filename = os.path.join(CODE_PLAN_AND_CHANGE_FILE_REPO, CODE_PLAN_AND_CHANGE_FILENAME) + await code_plan_and_change_file_repo.save( + filename=code_plan_and_change_filename, content=code_plan_and_change, dependencies=dependencies + ) + await code_plan_and_change_pdf_file_repo.save( + filename=Path(code_plan_and_change_filename).with_suffix(".md").name, + content=node.content, + dependencies=dependencies, + ) + + return Message( + content=code_plan_and_change, + role=self.profile, + cause_by=WriteCodePlanAndChange, + send_to=self, + sent_from=self, + ) + async def _is_pass(self, summary) -> (str, str): rsp = await self.llm.aask(msg=IS_PASS_PROMPT.format(context=summary), stream=False) logger.info(rsp) @@ -222,11 +259,16 @@ class Engineer(Role): async def _think(self) -> Action | None: if not CONFIG.src_workspace: CONFIG.src_workspace = CONFIG.git_repo.workdir / CONFIG.git_repo.workdir.name - write_code_filters = any_to_str_set([WriteTasks, SummarizeCode, FixBug]) + write_plan_and_change_filters = any_to_str_set([WriteTasks]) + write_code_filters = any_to_str_set([WriteTasks, WriteCodePlanAndChange, SummarizeCode, FixBug]) summarize_code_filters = any_to_str_set([WriteCode, WriteCodeReview]) if not self.rc.news: return None msg = self.rc.news[0] + if CONFIG.inc and msg.cause_by in write_plan_and_change_filters: + logger.debug(f"TODO WriteCodePlanAndChange:{msg.model_dump_json()}") + await self._new_code_plan_and_change_action() + return self.rc.todo if msg.cause_by in write_code_filters: logger.debug(f"TODO WriteCode:{msg.model_dump_json()}") await self._new_code_actions(bug_fix=msg.cause_by == any_to_str(FixBug)) @@ -332,50 +374,21 @@ class Engineer(Role): if self.summarize_todos: self.rc.todo = self.summarize_todos[0] + async def _new_code_plan_and_change_action(self): + """Create a WriteCodePlanAndChange action for subsequent to-do actions.""" + requirement_doc = await FileRepository.get_file(filename=REQUIREMENT_FILENAME, relative_path=DOCS_FILE_REPO) + prd_docs = await FileRepository.get_all_files(relative_path=PRDS_FILE_REPO) + design_docs = await FileRepository.get_all_files(relative_path=SYSTEM_DESIGN_FILE_REPO) + tasks_docs = await FileRepository.get_all_files(relative_path=TASK_FILE_REPO) + code_plan_and_change_context = CodePlanAndChangeContext( + requirement_doc=requirement_doc, + prd_docs=prd_docs, + design_docs=design_docs, + tasks_docs=tasks_docs, + ) + self.rc.todo = WriteCodePlanAndChange(context=code_plan_and_change_context, llm=self.llm) + @property def todo(self) -> str: """AgentStore uses this attribute to display to the user what actions the current role should take.""" return self.next_todo_action - - async def _write_code_plan_and_change(self): - """Write code plan and change that guides subsequent WriteCode and WriteCodeReview""" - logger.info("Writing code plan and change..") - - user_requirement = str(self.rc.memory.get_by_role("Human")[0]) - pool_contents = [] - prd = await FileRepository.get_all_files(relative_path=PRDS_FILE_REPO) - for doc in prd: - prd_json = json.loads(doc.content) - product_requirement_pool = prd_json.get(REFINED_REQUIREMENT_POOL.key) or prd_json.get(REQUIREMENT_POOL.key) - pool_contents.append(str(product_requirement_pool)) - - product_requirement_pools = "\n".join(pool_contents) - - design = await FileRepository.get_all_files(relative_path=SYSTEM_DESIGN_FILE_REPO) - design = "\n".join([doc.content for doc in design]) - - tasks = await FileRepository.get_all_files(relative_path=TASK_FILE_REPO) - tasks = "\n".join([doc.content for doc in tasks]) - - old_codes = await self.get_old_codes() - - context = CODE_PLAN_AND_CHANGE_CONTEXT.format( - user_requirement=user_requirement, - product_requirement_pools=product_requirement_pools, - tasks=tasks, - design=design, - code=old_codes, - ) - node = await WriteCodePlanAndChange().run(context=context) - code_plan_and_change = node.instruct_content.model_dump_json() - CONFIG.git_repo.new_file_repository(CODE_PLAN_AND_CHANGE_FILE_REPO).save( - filename=CODE_PLAN_AND_CHANGE_FILENAME, content=code_plan_and_change - ) - - @staticmethod - async def get_old_codes() -> str: - CONFIG.old_workspace = CONFIG.git_repo.workdir / os.path.basename(CONFIG.project_path) - old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) - old_codes = await old_file_repo.get_all() - codes = [f"----- {code.filename}\n```{code.content}```" for code in old_codes] - return "\n".join(codes) diff --git a/metagpt/schema.py b/metagpt/schema.py index e36bef395..5daaffdeb 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -451,3 +451,11 @@ class CodeSummarizeContext(BaseModel): class BugFixContext(BaseContext): filename: str = "" + + +class CodePlanAndChangeContext(BaseContext): + filename: str = "" + requirement_doc: Document + prd_docs: List[Document] + design_docs: List[Document] + task_docs: List[Document] diff --git a/tests/metagpt/actions/test_write_code_plan_an.py b/tests/metagpt/actions/test_write_code_plan_an.py index 13bda7cc1..44679e561 100644 --- a/tests/metagpt/actions/test_write_code_plan_an.py +++ b/tests/metagpt/actions/test_write_code_plan_an.py @@ -15,7 +15,6 @@ from metagpt.actions.write_code_plan_and_change_an import ( REFINED_TEMPLATE, WriteCodePlanAndChange, ) -from metagpt.actions.write_prd_an import REQUIREMENT_POOL from tests.data.incremental_dev_project.mock import ( CODE_PLAN_AND_CHANGE_SAMPLE, DESIGN_SAMPLE, @@ -45,8 +44,8 @@ async def test_write_code_plan_an(mocker): write_code_plan = WriteCodePlanAndChange() context = CODE_PLAN_AND_CHANGE_CONTEXT.format( - user_requirement=NEW_REQUIREMENT_SAMPLE, - product_requirement_pools=REFINED_PRD_JSON.get(REQUIREMENT_POOL.key), + requirement=NEW_REQUIREMENT_SAMPLE, + PRD=REFINED_PRD_JSON, design=REFINED_DESIGN_JSON, tasks=REFINED_TASKS_JSON, code=OLD_CODE_SAMPLE, From 81b59bfe0dd2d84df40781cffa61eab8d4053841 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 14:44:06 +0800 Subject: [PATCH 417/637] update CodePlanAndChangeContext --- metagpt/roles/engineer.py | 7 ++++--- metagpt/schema.py | 1 - 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index ac6e3b720..4d265f4df 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -125,6 +125,7 @@ class Engineer(Role): coding_context = await action.run() dependencies = {coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path} + # TODO: Add code plan and change file to context when _think if mode == "guide": dependencies.add(os.path.join(CODE_PLAN_AND_CHANGE_FILE_REPO, CODE_PLAN_AND_CHANGE_FILENAME)) await src_file_repo.save( @@ -161,8 +162,6 @@ class Engineer(Role): return None async def _act_write_code(self): - if CONFIG.inc: - await self._write_code_plan_and_change() changed_files = await self._act_sp_with_cr(review=self.use_code_review) return Message( content="\n".join(changed_files), @@ -376,10 +375,12 @@ class Engineer(Role): async def _new_code_plan_and_change_action(self): """Create a WriteCodePlanAndChange action for subsequent to-do actions.""" + # FIXME: The following code is not robust enough requirement_doc = await FileRepository.get_file(filename=REQUIREMENT_FILENAME, relative_path=DOCS_FILE_REPO) prd_docs = await FileRepository.get_all_files(relative_path=PRDS_FILE_REPO) design_docs = await FileRepository.get_all_files(relative_path=SYSTEM_DESIGN_FILE_REPO) - tasks_docs = await FileRepository.get_all_files(relative_path=TASK_FILE_REPO) + tasks_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) + tasks_docs = await tasks_file_repo.get_all() code_plan_and_change_context = CodePlanAndChangeContext( requirement_doc=requirement_doc, prd_docs=prd_docs, diff --git a/metagpt/schema.py b/metagpt/schema.py index 5daaffdeb..3fb934d93 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -454,7 +454,6 @@ class BugFixContext(BaseContext): class CodePlanAndChangeContext(BaseContext): - filename: str = "" requirement_doc: Document prd_docs: List[Document] design_docs: List[Document] From 69ad2f414757a0df131d102dd9effbe99016fae5 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 14:53:19 +0800 Subject: [PATCH 418/637] update mode from "guide" to "incremental" in get_codes function of write_code.py --- metagpt/actions/write_code.py | 12 +++++++----- metagpt/actions/write_code_plan_and_change_an.py | 2 +- metagpt/actions/write_code_review.py | 2 +- metagpt/roles/engineer.py | 4 ++-- 4 files changed, 11 insertions(+), 9 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 662524518..489c4ccb6 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -125,7 +125,9 @@ class WriteCode(Action): if bug_feedback: code_context = coding_context.code_doc.content elif code_plan_and_change: - code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename, mode="guide") + code_context = await self.get_codes( + coding_context.task_doc, exclude=self.context.filename, mode="incremental" + ) else: code_context = await self.get_codes(coding_context.task_doc, exclude=self.context.filename) @@ -161,14 +163,14 @@ class WriteCode(Action): return coding_context @staticmethod - async def get_codes(task_doc: Document, exclude: str, mode: Literal["normal", "guide"] = "normal") -> str: + async def get_codes(task_doc: Document, exclude: str, mode: Literal["normal", "incremental"] = "normal") -> str: """ Get code snippets based on different modes. Attributes: task_doc (Document): Document object of the task file. exclude (str): Specifies the filename to be excluded from the code snippets. - mode (str): Specifies the mode, either "normal" or "guide" (default is "normal"). + mode (str): Specifies the mode, either "normal" or "incremental" (default is "normal"). Returns: str: Code snippets. @@ -177,7 +179,7 @@ class WriteCode(Action): If mode is set to "normal", it returns code snippets for the regular coding phase, i.e., all the code generated before writing the current file. - If mode is set to "guide", it returns code snippets for generating the code plan and change, + If mode is set to "incremental", it returns code snippets for generating the code plan and change, building upon the existing code in the "normal" mode and adding code for the current file's older versions. """ if not task_doc: @@ -189,7 +191,7 @@ class WriteCode(Action): codes = [] src_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.src_workspace) - if mode == "guide": + if mode == "incremental": src_files = src_file_repo.all_files old_file_repo = CONFIG.git_repo.new_file_repository(relative_path=CONFIG.old_workspace) old_files = old_file_repo.all_files diff --git a/metagpt/actions/write_code_plan_and_change_an.py b/metagpt/actions/write_code_plan_and_change_an.py index 151b2dc25..db0b73554 100644 --- a/metagpt/actions/write_code_plan_and_change_an.py +++ b/metagpt/actions/write_code_plan_and_change_an.py @@ -194,7 +194,7 @@ class WriteCodePlanAndChange(Action): prd = "\n".join([doc.content for doc in self.context.prd_docs]) design = "\n".join([doc.content for doc in self.context.design_docs]) tasks = "\n".join([doc.content for doc in self.context.task_docs]) - code_text = self.get_old_codes() + code_text = await self.get_old_codes() context = CODE_PLAN_AND_CHANGE_CONTEXT.format( requirement=requirement, prd=prd, design=design, tasks=tasks, code=code_text ) diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index 6dbcb03fa..80f9dd36d 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -149,7 +149,7 @@ class WriteCodeReview(Action): filename=CODE_PLAN_AND_CHANGE_FILENAME, relative_path=CODE_PLAN_AND_CHANGE_FILE_REPO ) code_plan_and_change = code_plan_and_change_doc.content if code_plan_and_change_doc else "" - mode = "guide" if code_plan_and_change else "normal" + mode = "incremental" if code_plan_and_change else "normal" for i in range(k): format_example = FORMAT_EXAMPLE.format(filename=self.context.code_doc.filename) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 4d265f4df..44db2e8a0 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -106,7 +106,7 @@ class Engineer(Role): m = json.loads(task_msg.content) return m.get(TASK_LIST.key) or m.get(REFINED_TASK_LIST.key) - async def _act_sp_with_cr(self, review=False, mode: Literal["normal", "guide"] = "normal") -> Set[str]: + async def _act_sp_with_cr(self, review=False, mode: Literal["normal", "incremental"] = "normal") -> Set[str]: changed_files = set() src_file_repo = CONFIG.git_repo.new_file_repository(CONFIG.src_workspace) for todo in self.code_todos: @@ -126,7 +126,7 @@ class Engineer(Role): dependencies = {coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path} # TODO: Add code plan and change file to context when _think - if mode == "guide": + if mode == "incremental": dependencies.add(os.path.join(CODE_PLAN_AND_CHANGE_FILE_REPO, CODE_PLAN_AND_CHANGE_FILENAME)) await src_file_repo.save( coding_context.filename, From 134791ca35078e535677e817ec19cdda01ca9ef5 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 15:05:30 +0800 Subject: [PATCH 419/637] 1. update mode from "guide" to "incremental" in get_codes function of write_code.py 2. update _new_code_plan_and_change_action function --- metagpt/actions/write_code_plan_and_change_an.py | 2 +- metagpt/roles/engineer.py | 10 +++++----- metagpt/schema.py | 3 ++- 3 files changed, 8 insertions(+), 7 deletions(-) diff --git a/metagpt/actions/write_code_plan_and_change_an.py b/metagpt/actions/write_code_plan_and_change_an.py index db0b73554..8bf20e494 100644 --- a/metagpt/actions/write_code_plan_and_change_an.py +++ b/metagpt/actions/write_code_plan_and_change_an.py @@ -193,7 +193,7 @@ class WriteCodePlanAndChange(Action): requirement = self.context.requirement_doc.content prd = "\n".join([doc.content for doc in self.context.prd_docs]) design = "\n".join([doc.content for doc in self.context.design_docs]) - tasks = "\n".join([doc.content for doc in self.context.task_docs]) + tasks = "\n".join([doc.content for doc in self.context.tasks_docs]) code_text = await self.get_old_codes() context = CODE_PLAN_AND_CHANGE_CONTEXT.format( requirement=requirement, prd=prd, design=design, tasks=tasks, code=code_text diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 44db2e8a0..e1184dfb7 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -222,12 +222,12 @@ class Engineer(Role): node = await self.rc.todo.run() code_plan_and_change = node.instruct_content.model_dump_json() - + # FIXME: define a load function dependencies = { - self.rc.todo.context.requirement_filename, - self.rc.todo.context.prd_filename, - self.rc.todo.context.design_filename, - self.rc.todo.context.task_filename, + self.rc.todo.context.requirement_doc.filename, + self.rc.todo.context.prd_docs[0].filename, + self.rc.todo.context.design_docs[0].filename, + self.rc.todo.context.tasks_docs[0].filename, } code_plan_and_change_filename = os.path.join(CODE_PLAN_AND_CHANGE_FILE_REPO, CODE_PLAN_AND_CHANGE_FILENAME) diff --git a/metagpt/schema.py b/metagpt/schema.py index 3fb934d93..dd0e0a01e 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -454,7 +454,8 @@ class BugFixContext(BaseContext): class CodePlanAndChangeContext(BaseContext): + filename: str = "" requirement_doc: Document prd_docs: List[Document] design_docs: List[Document] - task_docs: List[Document] + tasks_docs: List[Document] From c6695a30236134873e65390e90dd38a9fd4d8d78 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Fri, 19 Jan 2024 15:37:28 +0800 Subject: [PATCH 420/637] fix: keep same return value in get_choice_function_arguments. --- metagpt/provider/openai_api.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 3edd89835..7bc4ee164 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -217,7 +217,7 @@ class OpenAILLM(BaseLLM): if code_value is None: raise ValueError(f"Parse code error for {arguments}") # arguments只有code的情况 - return {"language": language_value, "code": code_value} + return {"language": language_value or "python", "code": code_value} @handle_exception def get_choice_function_arguments(self, rsp: ChatCompletion) -> dict: @@ -241,7 +241,7 @@ class OpenAILLM(BaseLLM): f"Got JSONDecodeError for {message.tool_calls[0].function.arguments},\ we will use RegExp to parse code, \n {e}" ) - return {"language": "python", "code": self._parse_arguments(message.tool_calls[0].function.arguments)} + return self._parse_arguments(message.tool_calls[0].function.arguments) elif message.tool_calls is None and message.content is not None: # reponse is message return {"language": "markdown", "code": self.get_choice_text(rsp)} From 12b9f51abf289abe988ead22f49e3ecd574ddabe Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 16:00:59 +0800 Subject: [PATCH 421/637] update _watch in engineer.py --- metagpt/roles/engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index e1184dfb7..9bbd32649 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -96,7 +96,7 @@ class Engineer(Role): super().__init__(**kwargs) self._init_actions([WriteCode]) - self._watch([WriteTasks, SummarizeCode, WriteCode, WriteCodeReview, FixBug]) + self._watch([WriteTasks, SummarizeCode, WriteCode, WriteCodeReview, FixBug, WriteCodePlanAndChange]) self.code_todos = [] self.summarize_todos = [] self.next_todo_action = any_to_name(WriteCode) From 39c00848b9acb58eec0f740a45e99cc0cdbb50ff Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 19:39:59 +0800 Subject: [PATCH 422/637] update test case of ActionNode --- metagpt/roles/engineer.py | 4 +-- tests/metagpt/actions/test_design_api_an.py | 2 +- .../actions/test_project_management_an.py | 2 +- ... => test_write_code_plan_and_change_an.py} | 30 +++++++++---------- tests/metagpt/actions/test_write_prd_an.py | 4 +-- 5 files changed, 20 insertions(+), 22 deletions(-) rename tests/metagpt/actions/{test_write_code_plan_an.py => test_write_code_plan_and_change_an.py} (70%) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 9bbd32649..75393bc2b 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -379,8 +379,8 @@ class Engineer(Role): requirement_doc = await FileRepository.get_file(filename=REQUIREMENT_FILENAME, relative_path=DOCS_FILE_REPO) prd_docs = await FileRepository.get_all_files(relative_path=PRDS_FILE_REPO) design_docs = await FileRepository.get_all_files(relative_path=SYSTEM_DESIGN_FILE_REPO) - tasks_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) - tasks_docs = await tasks_file_repo.get_all() + tasks_docs = await FileRepository.get_all_files(relative_path=TASK_FILE_REPO) + code_plan_and_change_context = CodePlanAndChangeContext( requirement_doc=requirement_doc, prd_docs=prd_docs, diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index fcd2ef666..3d11f200d 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -35,7 +35,7 @@ async def test_write_design_an(mocker): ) root.instruct_content = BaseModel() root.instruct_content.model_dump = mock_refined_design_json - mocker.patch("metagpt.actions.design_api_an.REFINED_DESIGN_NODES.fill", return_value=root) + mocker.patch("metagpt.actions.design_api_an.REFINED_DESIGN_NODE.fill", return_value=root) prompt = NEW_REQ_TEMPLATE.format(old_design=DESIGN_SAMPLE, context=dict_to_markdown(REFINED_PRD_JSON)) node = await REFINED_DESIGN_NODE.fill(prompt, llm) diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index e230151da..aa759aec8 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -35,7 +35,7 @@ async def test_project_management_an(mocker): ) root.instruct_content = BaseModel() root.instruct_content.model_dump = mock_refined_tasks_json - mocker.patch("metagpt.actions.project_management_an.REFINED_PM_NODES.fill", return_value=root) + mocker.patch("metagpt.actions.project_management_an.REFINED_PM_NODE.fill", return_value=root) prompt = NEW_REQ_TEMPLATE.format(old_tasks=TASKS_SAMPLE, context=dict_to_markdown(REFINED_DESIGN_JSON)) node = await REFINED_PM_NODE.fill(prompt, llm) diff --git a/tests/metagpt/actions/test_write_code_plan_an.py b/tests/metagpt/actions/test_write_code_plan_and_change_an.py similarity index 70% rename from tests/metagpt/actions/test_write_code_plan_an.py rename to tests/metagpt/actions/test_write_code_plan_and_change_an.py index 44679e561..33114dfcf 100644 --- a/tests/metagpt/actions/test_write_code_plan_an.py +++ b/tests/metagpt/actions/test_write_code_plan_and_change_an.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_write_code_plan_an.py +@File : test_write_code_plan_and_change_an.py """ import pytest from openai._models import BaseModel @@ -11,20 +11,16 @@ from openai._models import BaseModel from metagpt.actions.action_node import ActionNode from metagpt.actions.write_code import WriteCode from metagpt.actions.write_code_plan_and_change_an import ( - CODE_PLAN_AND_CHANGE_CONTEXT, REFINED_TEMPLATE, WriteCodePlanAndChange, ) +from metagpt.schema import CodePlanAndChangeContext, Document from tests.data.incremental_dev_project.mock import ( CODE_PLAN_AND_CHANGE_SAMPLE, DESIGN_SAMPLE, NEW_REQUIREMENT_SAMPLE, - OLD_CODE_SAMPLE, REFINED_CODE_INPUT_SAMPLE, REFINED_CODE_SAMPLE, - REFINED_DESIGN_JSON, - REFINED_PRD_JSON, - REFINED_TASKS_JSON, TASKS_SAMPLE, ) @@ -34,23 +30,25 @@ def mock_code_plan_and_change(): @pytest.mark.asyncio -async def test_write_code_plan_an(mocker): +async def test_write_code_plan_and_change_an(mocker): root = ActionNode.from_children( "WriteCodePlanAndChange", [ActionNode(key="", expected_type=str, instruction="", example="")] ) root.instruct_content = BaseModel() root.instruct_content.model_dump = mock_code_plan_and_change - mocker.patch("metagpt.actions.write_code_plan_an.WriteCodePlanAndChange.run", return_value=root) + mocker.patch("metagpt.actions.write_code_plan_and_change_an.WriteCodePlanAndChange.run", return_value=root) - write_code_plan = WriteCodePlanAndChange() - context = CODE_PLAN_AND_CHANGE_CONTEXT.format( - requirement=NEW_REQUIREMENT_SAMPLE, - PRD=REFINED_PRD_JSON, - design=REFINED_DESIGN_JSON, - tasks=REFINED_TASKS_JSON, - code=OLD_CODE_SAMPLE, + requirement_doc = Document() + prd_docs = [Document()] + design_docs = [Document()] + tasks_docs = [Document()] + code_plan_and_change_context = CodePlanAndChangeContext( + requirement_doc=requirement_doc, + prd_docs=prd_docs, + design_docs=design_docs, + tasks_docs=tasks_docs, ) - node = await write_code_plan.run(context=context) + node = await WriteCodePlanAndChange(context=code_plan_and_change_context).run() assert "Plan" in node.instruct_content.model_dump() diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index 806f88d36..3bfb4d3be 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -29,10 +29,10 @@ def mock_refined_prd_json(): @pytest.mark.asyncio async def test_write_prd_an(mocker): - root = ActionNode.from_children("RefinePRD", [ActionNode(key="", expected_type=str, instruction="", example="")]) + root = ActionNode.from_children("RefinedPRD", [ActionNode(key="", expected_type=str, instruction="", example="")]) root.instruct_content = BaseModel() root.instruct_content.model_dump = mock_refined_prd_json - mocker.patch("metagpt.actions.write_prd_an.REFINE_PRD_NODE.fill", return_value=root) + mocker.patch("metagpt.actions.write_prd_an.REFINED_PRD_NODE.fill", return_value=root) prompt = REFINED_TEMPLATE.format( requirements=NEW_REQUIREMENT_SAMPLE, From 4bb9c006c282db05ced4cdf1ebfa0db72927c192 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 19 Jan 2024 19:39:59 +0800 Subject: [PATCH 423/637] update test case of ActionNode --- metagpt/roles/engineer.py | 7 ++--- tests/metagpt/actions/test_design_api_an.py | 2 +- .../actions/test_project_management_an.py | 2 +- ... => test_write_code_plan_and_change_an.py} | 30 +++++++++---------- tests/metagpt/actions/test_write_prd_an.py | 4 +-- 5 files changed, 20 insertions(+), 25 deletions(-) rename tests/metagpt/actions/{test_write_code_plan_an.py => test_write_code_plan_and_change_an.py} (70%) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 9bbd32649..b1588f9ef 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -125,7 +125,6 @@ class Engineer(Role): coding_context = await action.run() dependencies = {coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path} - # TODO: Add code plan and change file to context when _think if mode == "incremental": dependencies.add(os.path.join(CODE_PLAN_AND_CHANGE_FILE_REPO, CODE_PLAN_AND_CHANGE_FILENAME)) await src_file_repo.save( @@ -222,7 +221,6 @@ class Engineer(Role): node = await self.rc.todo.run() code_plan_and_change = node.instruct_content.model_dump_json() - # FIXME: define a load function dependencies = { self.rc.todo.context.requirement_doc.filename, self.rc.todo.context.prd_docs[0].filename, @@ -375,12 +373,11 @@ class Engineer(Role): async def _new_code_plan_and_change_action(self): """Create a WriteCodePlanAndChange action for subsequent to-do actions.""" - # FIXME: The following code is not robust enough requirement_doc = await FileRepository.get_file(filename=REQUIREMENT_FILENAME, relative_path=DOCS_FILE_REPO) prd_docs = await FileRepository.get_all_files(relative_path=PRDS_FILE_REPO) design_docs = await FileRepository.get_all_files(relative_path=SYSTEM_DESIGN_FILE_REPO) - tasks_file_repo = CONFIG.git_repo.new_file_repository(TASK_FILE_REPO) - tasks_docs = await tasks_file_repo.get_all() + tasks_docs = await FileRepository.get_all_files(relative_path=TASK_FILE_REPO) + code_plan_and_change_context = CodePlanAndChangeContext( requirement_doc=requirement_doc, prd_docs=prd_docs, diff --git a/tests/metagpt/actions/test_design_api_an.py b/tests/metagpt/actions/test_design_api_an.py index fcd2ef666..3d11f200d 100644 --- a/tests/metagpt/actions/test_design_api_an.py +++ b/tests/metagpt/actions/test_design_api_an.py @@ -35,7 +35,7 @@ async def test_write_design_an(mocker): ) root.instruct_content = BaseModel() root.instruct_content.model_dump = mock_refined_design_json - mocker.patch("metagpt.actions.design_api_an.REFINED_DESIGN_NODES.fill", return_value=root) + mocker.patch("metagpt.actions.design_api_an.REFINED_DESIGN_NODE.fill", return_value=root) prompt = NEW_REQ_TEMPLATE.format(old_design=DESIGN_SAMPLE, context=dict_to_markdown(REFINED_PRD_JSON)) node = await REFINED_DESIGN_NODE.fill(prompt, llm) diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index e230151da..aa759aec8 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -35,7 +35,7 @@ async def test_project_management_an(mocker): ) root.instruct_content = BaseModel() root.instruct_content.model_dump = mock_refined_tasks_json - mocker.patch("metagpt.actions.project_management_an.REFINED_PM_NODES.fill", return_value=root) + mocker.patch("metagpt.actions.project_management_an.REFINED_PM_NODE.fill", return_value=root) prompt = NEW_REQ_TEMPLATE.format(old_tasks=TASKS_SAMPLE, context=dict_to_markdown(REFINED_DESIGN_JSON)) node = await REFINED_PM_NODE.fill(prompt, llm) diff --git a/tests/metagpt/actions/test_write_code_plan_an.py b/tests/metagpt/actions/test_write_code_plan_and_change_an.py similarity index 70% rename from tests/metagpt/actions/test_write_code_plan_an.py rename to tests/metagpt/actions/test_write_code_plan_and_change_an.py index 44679e561..33114dfcf 100644 --- a/tests/metagpt/actions/test_write_code_plan_an.py +++ b/tests/metagpt/actions/test_write_code_plan_and_change_an.py @@ -3,7 +3,7 @@ """ @Time : 2024/01/03 @Author : mannaandpoem -@File : test_write_code_plan_an.py +@File : test_write_code_plan_and_change_an.py """ import pytest from openai._models import BaseModel @@ -11,20 +11,16 @@ from openai._models import BaseModel from metagpt.actions.action_node import ActionNode from metagpt.actions.write_code import WriteCode from metagpt.actions.write_code_plan_and_change_an import ( - CODE_PLAN_AND_CHANGE_CONTEXT, REFINED_TEMPLATE, WriteCodePlanAndChange, ) +from metagpt.schema import CodePlanAndChangeContext, Document from tests.data.incremental_dev_project.mock import ( CODE_PLAN_AND_CHANGE_SAMPLE, DESIGN_SAMPLE, NEW_REQUIREMENT_SAMPLE, - OLD_CODE_SAMPLE, REFINED_CODE_INPUT_SAMPLE, REFINED_CODE_SAMPLE, - REFINED_DESIGN_JSON, - REFINED_PRD_JSON, - REFINED_TASKS_JSON, TASKS_SAMPLE, ) @@ -34,23 +30,25 @@ def mock_code_plan_and_change(): @pytest.mark.asyncio -async def test_write_code_plan_an(mocker): +async def test_write_code_plan_and_change_an(mocker): root = ActionNode.from_children( "WriteCodePlanAndChange", [ActionNode(key="", expected_type=str, instruction="", example="")] ) root.instruct_content = BaseModel() root.instruct_content.model_dump = mock_code_plan_and_change - mocker.patch("metagpt.actions.write_code_plan_an.WriteCodePlanAndChange.run", return_value=root) + mocker.patch("metagpt.actions.write_code_plan_and_change_an.WriteCodePlanAndChange.run", return_value=root) - write_code_plan = WriteCodePlanAndChange() - context = CODE_PLAN_AND_CHANGE_CONTEXT.format( - requirement=NEW_REQUIREMENT_SAMPLE, - PRD=REFINED_PRD_JSON, - design=REFINED_DESIGN_JSON, - tasks=REFINED_TASKS_JSON, - code=OLD_CODE_SAMPLE, + requirement_doc = Document() + prd_docs = [Document()] + design_docs = [Document()] + tasks_docs = [Document()] + code_plan_and_change_context = CodePlanAndChangeContext( + requirement_doc=requirement_doc, + prd_docs=prd_docs, + design_docs=design_docs, + tasks_docs=tasks_docs, ) - node = await write_code_plan.run(context=context) + node = await WriteCodePlanAndChange(context=code_plan_and_change_context).run() assert "Plan" in node.instruct_content.model_dump() diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index 806f88d36..3bfb4d3be 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -29,10 +29,10 @@ def mock_refined_prd_json(): @pytest.mark.asyncio async def test_write_prd_an(mocker): - root = ActionNode.from_children("RefinePRD", [ActionNode(key="", expected_type=str, instruction="", example="")]) + root = ActionNode.from_children("RefinedPRD", [ActionNode(key="", expected_type=str, instruction="", example="")]) root.instruct_content = BaseModel() root.instruct_content.model_dump = mock_refined_prd_json - mocker.patch("metagpt.actions.write_prd_an.REFINE_PRD_NODE.fill", return_value=root) + mocker.patch("metagpt.actions.write_prd_an.REFINED_PRD_NODE.fill", return_value=root) prompt = REFINED_TEMPLATE.format( requirements=NEW_REQUIREMENT_SAMPLE, From c4a60d89e0061e7896af2f02ad076f7568e778cb Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 19 Jan 2024 22:29:37 +0800 Subject: [PATCH 424/637] make tool yaml from class or func docstring --- metagpt/tools/libs/__init__.py | 4 +- metagpt/tools/libs/data_preprocess.py | 280 +++++++++++++++--- metagpt/tools/libs/feature_engineering.py | 3 +- .../{web_scrapping.py => web_scraping.py} | 0 .../schemas/data_preprocess/OrdinalEncode.yml | 46 +++ .../schemas/data_preprocess/RobustScale.yml | 47 +++ .../scrape_web_playwright.yml | 0 metagpt/tools/tool_convert.py | 85 ++++++ metagpt/tools/tool_registry.py | 41 ++- 9 files changed, 449 insertions(+), 57 deletions(-) rename metagpt/tools/libs/{web_scrapping.py => web_scraping.py} (100%) create mode 100644 metagpt/tools/schemas/data_preprocess/OrdinalEncode.yml create mode 100644 metagpt/tools/schemas/data_preprocess/RobustScale.yml rename metagpt/tools/schemas/{web_scrapping => web_scraping}/scrape_web_playwright.yml (100%) create mode 100644 metagpt/tools/tool_convert.py diff --git a/metagpt/tools/libs/__init__.py b/metagpt/tools/libs/__init__.py index 442f57149..c9767c1e5 100644 --- a/metagpt/tools/libs/__init__.py +++ b/metagpt/tools/libs/__init__.py @@ -9,7 +9,7 @@ from metagpt.tools.libs import ( feature_engineering, sd_engine, gpt_v_generator, - web_scrapping, + web_scraping, ) -_ = data_preprocess, feature_engineering, sd_engine, gpt_v_generator, web_scrapping # Avoid pre-commit error +_ = data_preprocess, feature_engineering, sd_engine, gpt_v_generator, web_scraping # Avoid pre-commit error diff --git a/metagpt/tools/libs/data_preprocess.py b/metagpt/tools/libs/data_preprocess.py index 3891f9df0..0480e71a7 100644 --- a/metagpt/tools/libs/data_preprocess.py +++ b/metagpt/tools/libs/data_preprocess.py @@ -26,31 +26,64 @@ class MLProcess(object): def transform(self, df): raise NotImplementedError - def fit_transform(self, df): + def fit_transform(self, df) -> pd.DataFrame: + """ + Fit and transform the input DataFrame. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ self.fit(df) return self.transform(df) @register_tool(tool_type=TOOL_TYPE) class FillMissingValue(MLProcess): - def __init__( - self, - features: list, - strategy: str = "mean", - fill_value=None, - ): + """ + Completing missing values with simple strategies. + """ + + def __init__(self, features: list, strategy: str = "mean", fill_value=None): + """ + Initialize self. + + Args: + features (list): Columns to be processed. + strategy (str, optional): The imputation strategy, notice 'mean' and 'median' can only + be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'. + fill_value (int, optional): Fill_value is used to replace all occurrences of missing_values. + Defaults to None. + """ self.features = features self.strategy = strategy self.fill_value = fill_value self.si = None def fit(self, df: pd.DataFrame): + """ + Fit the FillMissingValue model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ if len(self.features) == 0: return self.si = SimpleImputer(strategy=self.strategy, fill_value=self.fill_value) self.si.fit(df[self.features]) - def transform(self, df: pd.DataFrame): + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ if len(self.features) == 0: return df new_df = df.copy() @@ -60,18 +93,40 @@ class FillMissingValue(MLProcess): @register_tool(tool_type=TOOL_TYPE) class MinMaxScale(MLProcess): - def __init__( - self, - features: list, - ): + """ + Transform features by scaling each feature to a range, which is (0, 1). + """ + + def __init__(self, features: list): + """ + Initialize self. + + Args: + features (list): Columns to be processed. + """ self.features = features self.mms = None def fit(self, df: pd.DataFrame): + """ + Fit the MinMaxScale model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ self.mms = MinMaxScaler() self.mms.fit(df[self.features]) - def transform(self, df: pd.DataFrame): + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ new_df = df.copy() new_df[self.features] = self.mms.transform(new_df[self.features]) return new_df @@ -79,18 +134,40 @@ class MinMaxScale(MLProcess): @register_tool(tool_type=TOOL_TYPE) class StandardScale(MLProcess): - def __init__( - self, - features: list, - ): + """ + Standardize features by removing the mean and scaling to unit variance. + """ + + def __init__(self, features: list): + """ + Initialize self. + + Args: + features (list): Columns to be processed. + """ self.features = features self.ss = None def fit(self, df: pd.DataFrame): + """ + Fit the StandardScale model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ self.ss = StandardScaler() self.ss.fit(df[self.features]) - def transform(self, df: pd.DataFrame): + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ new_df = df.copy() new_df[self.features] = self.ss.transform(new_df[self.features]) return new_df @@ -98,18 +175,40 @@ class StandardScale(MLProcess): @register_tool(tool_type=TOOL_TYPE) class MaxAbsScale(MLProcess): - def __init__( - self, - features: list, - ): + """ + Scale each feature by its maximum absolute value. + """ + + def __init__(self, features: list): + """ + Initialize self. + + Args: + features (list): Columns to be processed. + """ self.features = features self.mas = None def fit(self, df: pd.DataFrame): + """ + Fit the MaxAbsScale model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ self.mas = MaxAbsScaler() self.mas.fit(df[self.features]) - def transform(self, df: pd.DataFrame): + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ new_df = df.copy() new_df[self.features] = self.mas.transform(new_df[self.features]) return new_df @@ -117,18 +216,40 @@ class MaxAbsScale(MLProcess): @register_tool(tool_type=TOOL_TYPE) class RobustScale(MLProcess): - def __init__( - self, - features: list, - ): + """ + Apply the RobustScaler to scale features using statistics that are robust to outliers. + """ + + def __init__(self, features: list): + """ + Initialize the RobustScale instance with feature names. + + Args: + features (list): List of feature names to be scaled. + """ self.features = features self.rs = None def fit(self, df: pd.DataFrame): + """ + Compute the median and IQR for scaling. + + Args: + df (pd.DataFrame): Dataframe containing the features. + """ self.rs = RobustScaler() self.rs.fit(df[self.features]) def transform(self, df: pd.DataFrame): + """ + Scale features using the previously computed median and IQR. + + Args: + df (pd.DataFrame): Dataframe containing the features to be scaled. + + Returns: + pd.DataFrame: A new dataframe with scaled features. + """ new_df = df.copy() new_df[self.features] = self.rs.transform(new_df[self.features]) return new_df @@ -136,18 +257,40 @@ class RobustScale(MLProcess): @register_tool(tool_type=TOOL_TYPE) class OrdinalEncode(MLProcess): - def __init__( - self, - features: list, - ): + """ + Encode categorical features as ordinal integers. + """ + + def __init__(self, features: list): + """ + Initialize the OrdinalEncode instance with feature names. + + Args: + features (list): List of categorical feature names to be encoded. + """ self.features = features self.oe = None def fit(self, df: pd.DataFrame): + """ + Learn the ordinal encodings for the features. + + Args: + df (pd.DataFrame): Dataframe containing the categorical features. + """ self.oe = OrdinalEncoder() self.oe.fit(df[self.features]) def transform(self, df: pd.DataFrame): + """ + Convert the categorical features to ordinal integers. + + Args: + df (pd.DataFrame): Dataframe containing the categorical features to be encoded. + + Returns: + pd.DataFrame: A new dataframe with the encoded features. + """ new_df = df.copy() new_df[self.features] = self.oe.transform(new_df[self.features]) return new_df @@ -155,18 +298,40 @@ class OrdinalEncode(MLProcess): @register_tool(tool_type=TOOL_TYPE) class OneHotEncode(MLProcess): - def __init__( - self, - features: list, - ): + """ + Apply one-hot encoding to specified categorical columns, the original columns will be dropped. + """ + + def __init__(self, features: list): + """ + Initialize self. + + Args: + features (list): Categorical columns to be one-hot encoded and dropped. + """ self.features = features self.ohe = None def fit(self, df: pd.DataFrame): + """ + Fit the OneHotEncoding model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ self.ohe = OneHotEncoder(handle_unknown="ignore", sparse=False) self.ohe.fit(df[self.features]) - def transform(self, df: pd.DataFrame): + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ ts_data = self.ohe.transform(df[self.features]) new_columns = self.ohe.get_feature_names_out(self.features) ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) @@ -177,21 +342,43 @@ class OneHotEncode(MLProcess): @register_tool(tool_type=TOOL_TYPE) class LabelEncode(MLProcess): - def __init__( - self, - features: list, - ): + """ + Apply label encoding to specified categorical columns in-place. + """ + + def __init__(self, features: list): + """ + Initialize self. + + Args: + features (list): Categorical columns to be label encoded. + """ self.features = features self.le_encoders = [] def fit(self, df: pd.DataFrame): + """ + Fit the LabelEncode model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ if len(self.features) == 0: return for col in self.features: le = LabelEncoder().fit(df[col].astype(str).unique().tolist() + ["unknown"]) self.le_encoders.append(le) - def transform(self, df: pd.DataFrame): + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ if len(self.features) == 0: return df new_df = df.copy() @@ -204,8 +391,17 @@ class LabelEncode(MLProcess): return new_df -@register_tool(tool_type=TOOL_TYPE) def get_column_info(df: pd.DataFrame) -> dict: + """ + Analyzes a DataFrame and categorizes its columns based on data types. + + Args: + df (pd.DataFrame): The DataFrame to be analyzed. + + Returns: + dict: A dictionary with four keys ('Category', 'Numeric', 'Datetime', 'Others'). + Each key corresponds to a list of column names belonging to that category. + """ column_info = { "Category": [], "Numeric": [], diff --git a/metagpt/tools/libs/feature_engineering.py b/metagpt/tools/libs/feature_engineering.py index 308150f9b..79e1c1b07 100644 --- a/metagpt/tools/libs/feature_engineering.py +++ b/metagpt/tools/libs/feature_engineering.py @@ -184,7 +184,7 @@ class SplitBins(MLProcess): return new_df -@register_tool(tool_type=TOOL_TYPE) +# @register_tool(tool_type=TOOL_TYPE) class ExtractTimeComps(MLProcess): def __init__(self, time_col: str, time_comps: list): self.time_col = time_col @@ -242,6 +242,7 @@ class GeneralSelection(MLProcess): # skip for now because lgb is needed +# @register_tool(tool_type=TOOL_TYPE) class TreeBasedSelection(MLProcess): def __init__(self, label_col: str, task_type: str): self.label_col = label_col diff --git a/metagpt/tools/libs/web_scrapping.py b/metagpt/tools/libs/web_scraping.py similarity index 100% rename from metagpt/tools/libs/web_scrapping.py rename to metagpt/tools/libs/web_scraping.py diff --git a/metagpt/tools/schemas/data_preprocess/OrdinalEncode.yml b/metagpt/tools/schemas/data_preprocess/OrdinalEncode.yml new file mode 100644 index 000000000..79ebaf37c --- /dev/null +++ b/metagpt/tools/schemas/data_preprocess/OrdinalEncode.yml @@ -0,0 +1,46 @@ +OrdinalEncode: + type: class + description: Encode categorical features as ordinal integers. + methods: + __init__: + description: 'Initialize the OrdinalEncode instance with feature names. ' + parameters: + properties: + features: + type: list + description: List of categorical feature names to be encoded. + required: + - features + fit: + description: 'Learn the ordinal encodings for the features. ' + parameters: + properties: + df: + type: pd.DataFrame + description: Dataframe containing the categorical features. + required: + - df + fit_transform: + description: 'Fit and transform the input DataFrame. ' + parameters: + properties: + df: + type: pd.DataFrame + description: The input DataFrame. + required: + - df + returns: + - type: pd.DataFrame + description: The transformed DataFrame. + transform: + description: 'Convert the categorical features to ordinal integers. ' + parameters: + properties: + df: + type: pd.DataFrame + description: Dataframe containing the categorical features to be encoded. + required: + - df + returns: + - type: pd.DataFrame + description: A new dataframe with the encoded features. diff --git a/metagpt/tools/schemas/data_preprocess/RobustScale.yml b/metagpt/tools/schemas/data_preprocess/RobustScale.yml new file mode 100644 index 000000000..6d5dfaf3a --- /dev/null +++ b/metagpt/tools/schemas/data_preprocess/RobustScale.yml @@ -0,0 +1,47 @@ +RobustScale: + type: class + description: Apply the RobustScaler to scale features using statistics that are + robust to outliers. + methods: + __init__: + description: 'Initialize the RobustScale instance with feature names. ' + parameters: + properties: + features: + type: list + description: List of feature names to be scaled. + required: + - features + fit: + description: 'Compute the median and IQR for scaling. ' + parameters: + properties: + df: + type: pd.DataFrame + description: Dataframe containing the features. + required: + - df + fit_transform: + description: 'Fit and transform the input DataFrame. ' + parameters: + properties: + df: + type: pd.DataFrame + description: The input DataFrame. + required: + - df + returns: + - type: pd.DataFrame + description: The transformed DataFrame. + transform: + description: 'Scale features using the previously computed median and IQR. ' + parameters: + properties: + df: + type: pd.DataFrame + description: Dataframe containing the features to be scaled. + required: + - df + returns: + - type: pd.DataFrame + description: A new dataframe with scaled features. diff --git a/metagpt/tools/schemas/web_scrapping/scrape_web_playwright.yml b/metagpt/tools/schemas/web_scraping/scrape_web_playwright.yml similarity index 100% rename from metagpt/tools/schemas/web_scrapping/scrape_web_playwright.yml rename to metagpt/tools/schemas/web_scraping/scrape_web_playwright.yml diff --git a/metagpt/tools/tool_convert.py b/metagpt/tools/tool_convert.py new file mode 100644 index 000000000..c2ea33085 --- /dev/null +++ b/metagpt/tools/tool_convert.py @@ -0,0 +1,85 @@ +import inspect +import re + + +def remove_spaces(text): + return re.sub(r"\s+", " ", text) + + +def convert_code_to_tool_schema(obj, include: list[str] = []): + docstring = inspect.getdoc(obj) + assert docstring, "no docstring found for the objects, skip registering" + + if inspect.isclass(obj): + schema = {"type": "class", "description": remove_spaces(docstring), "methods": {}} + for name, method in inspect.getmembers(obj, inspect.isfunction): + if include and name not in include: + continue + method_doc = inspect.getdoc(method) + if method_doc: + schema["methods"][name] = docstring_to_schema(method_doc) + + elif inspect.isfunction(obj): + schema = { + "type": "function", + **docstring_to_schema(docstring), + } + + schema = {obj.__name__: schema} + + return schema + + +def docstring_to_schema(docstring: str): + if docstring is None: + return {} + + # 匹配简介部分 + description_match = re.search(r"^(.*?)(?:Args:|Returns:|Raises:|$)", docstring, re.DOTALL) + description = remove_spaces(description_match.group(1)) if description_match else "" + + # 匹配Args部分 + args_match = re.search(r"Args:\s*(.*?)(?:Returns:|Raises:|$)", docstring, re.DOTALL) + _args = args_match.group(1).strip() if args_match else "" + # variable_pattern = re.compile(r"(\w+)\s*\((.*?)\):\s*(.*)") + variable_pattern = re.compile( + r"(\w+)\s*\((.*?)\):\s*(.*?)(?=\n\s*\w+\s*\(|\Z)", re.DOTALL + ) # (?=\n\w+\s*\(|\Z) isb to assert that what follows is either the start of the next parameter (indicated by a newline, some word characters, and an opening parenthesis) or the end of the string (\Z). + + params = variable_pattern.findall(_args) + parameter_schema = {"properties": {}, "required": []} + for param in params: + param_name, param_type, param_desc = param + # check required or optional + if "optional" in param_type: + param_type = param_type.replace(", optional", "") + else: + parameter_schema["required"].append(param_name) + # type and desc + param_dict = {"type": param_type, "description": remove_spaces(param_desc)} + # match Default for optional args + default_val = re.search(r"Defaults to (.+?)\.", param_desc) + if default_val: + param_dict["default"] = default_val.group(1) + # match Enum + enum_val = re.search(r"Enum: \[(.+?)\]", param_desc) + if enum_val: + param_dict["enum"] = [e.strip() for e in enum_val.group(1).split(",")] + # add to parameter schema + parameter_schema["properties"].update({param_name: param_dict}) + + # 匹配Returns部分 + returns_match = re.search(r"Returns:\s*(.*?)(?:Raises:|$)", docstring, re.DOTALL) + returns = returns_match.group(1).strip() if returns_match else "" + return_pattern = re.compile(r"^(.*)\s*:\s*(.*)$") + returns = return_pattern.findall(returns) + + # 构建YAML字典 + schema = { + "description": description, + "parameters": parameter_schema, + } + if returns: + schema["returns"] = [{"type": ret[0], "description": remove_spaces(ret[1])} for ret in returns] + + return schema diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index 52ad25ce4..fbdfb3cfd 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -14,6 +14,7 @@ import yaml from metagpt.const import TOOL_SCHEMA_PATH from metagpt.logs import logger +from metagpt.tools.tool_convert import convert_code_to_tool_schema from metagpt.tools.tool_data_type import Tool, ToolSchema, ToolType @@ -34,7 +35,9 @@ class ToolRegistry: schema_path=None, tool_code="", tool_type="other", - make_schema_if_not_exists=False, + tool_source_object=None, + include_functions=[], + make_schema_if_not_exists=True, ): if self.has_tool(tool_name): return @@ -44,14 +47,16 @@ class ToolRegistry: if not os.path.exists(schema_path): if make_schema_if_not_exists: logger.warning(f"no schema found, will make schema at {schema_path}") - make_schema(tool_code, schema_path) + schema_dict = make_schema(tool_source_object, include_functions, schema_path) else: logger.warning(f"no schema found at assumed schema_path {schema_path}, skip registering {tool_name}") return - - with open(schema_path, "r", encoding="utf-8") as f: - schema_dict = yaml.safe_load(f) - schemas = schema_dict.get(tool_name) or list(schema_dict.values())[0] + else: + with open(schema_path, "r", encoding="utf-8") as f: + schema_dict = yaml.safe_load(f) + if not schema_dict: + return + schemas = schema_dict.get(tool_name) or list(schema_dict.values())[0] schemas["tool_path"] = tool_path # corresponding code file path of the tool try: ToolSchema(**schemas) # validation @@ -94,7 +99,7 @@ def register_tool_type(cls): return cls -def register_tool(tool_name="", tool_type="other", schema_path=None): +def register_tool(tool_name="", tool_type="other", schema_path=None, **kwargs): """register a tool to registry""" def decorator(cls, tool_name=tool_name): @@ -112,15 +117,27 @@ def register_tool(tool_name="", tool_type="other", schema_path=None): schema_path=schema_path, tool_code=source_code, tool_type=tool_type, + tool_source_object=cls, + **kwargs, ) return cls return decorator -def make_schema(tool_code, path): +def make_schema(tool_source_object, include, path): os.makedirs(os.path.dirname(path), exist_ok=True) # Create the necessary directories - schema = {} # an empty schema for now - with open(path, "w", encoding="utf-8") as f: - yaml.dump(schema, f) - return path + try: + schema = convert_code_to_tool_schema(tool_source_object, include=include) + with open(path, "w", encoding="utf-8") as f: + yaml.dump(schema, f, sort_keys=False) + # import json + # with open(str(path).replace("yml", "json"), "w", encoding="utf-8") as f: + # json.dump(schema, f, ensure_ascii=False, indent=4) + logger.info(f"schema made at {path}") + except Exception as e: + schema = {} + logger.error("Fail to make schema") + print(e) + + return schema From 2ccfe3112362824acdbcfd362285a19463751006 Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 19 Jan 2024 22:32:43 +0800 Subject: [PATCH 425/637] unittest for tool convert --- tests/metagpt/tools/test_tool_convert.py | 158 +++++++++++++++++++++++ 1 file changed, 158 insertions(+) create mode 100644 tests/metagpt/tools/test_tool_convert.py diff --git a/tests/metagpt/tools/test_tool_convert.py b/tests/metagpt/tools/test_tool_convert.py new file mode 100644 index 000000000..1dad997bd --- /dev/null +++ b/tests/metagpt/tools/test_tool_convert.py @@ -0,0 +1,158 @@ +import pandas as pd + +from metagpt.tools.tool_convert import convert_code_to_tool_schema, docstring_to_schema + + +def test_docstring_to_schema(): + docstring = """ + Some test desc. + + Args: + features (list): Columns to be processed. + strategy (str, optional): The imputation strategy, notice 'mean' and 'median' can only be + used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'. + fill_value (int, optional): Fill_value is used to replace all occurrences of missing_values. + Defaults to None. + Returns: + pd.DataFrame: The transformed DataFrame. + """ + expected = { + "description": " Some test desc. ", + "parameters": { + "properties": { + "features": {"type": "list", "description": "Columns to be processed."}, + "strategy": { + "type": "str", + "description": "The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.", + "default": "'mean'", + "enum": ["'mean'", "'median'", "'most_frequent'", "'constant'"], + }, + "fill_value": { + "type": "int", + "description": "Fill_value is used to replace all occurrences of missing_values. Defaults to None.", + "default": "None", + }, + }, + "required": ["features"], + }, + "returns": [{"type": "pd.DataFrame", "description": "The transformed DataFrame."}], + } + schema = docstring_to_schema(docstring) + assert schema == expected + + +class DummyClass: + """ + Completing missing values with simple strategies. + """ + + def __init__(self, features: list, strategy: str = "mean", fill_value=None): + """ + Initialize self. + + Args: + features (list): Columns to be processed. + strategy (str, optional): The imputation strategy, notice 'mean' and 'median' can only + be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'. + fill_value (int, optional): Fill_value is used to replace all occurrences of missing_values. + Defaults to None. + """ + pass + + def fit(self, df: pd.DataFrame): + """ + Fit the FillMissingValue model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ + pass + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ + pass + + +def dummy_fn(df: pd.DataFrame) -> dict: + """ + Analyzes a DataFrame and categorizes its columns based on data types. + + Args: + df (pd.DataFrame): The DataFrame to be analyzed. + + Returns: + dict: A dictionary with four keys ('Category', 'Numeric', 'Datetime', 'Others'). + Each key corresponds to a list of column names belonging to that category. + """ + pass + + +def test_convert_code_to_tool_schema_class(): + expected = { + "DummyClass": { + "type": "class", + "description": "Completing missing values with simple strategies.", + "methods": { + "__init__": { + "description": "Initialize self. ", + "parameters": { + "properties": { + "features": {"type": "list", "description": "Columns to be processed."}, + "strategy": { + "type": "str", + "description": "The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.", + "default": "'mean'", + "enum": ["'mean'", "'median'", "'most_frequent'", "'constant'"], + }, + "fill_value": { + "type": "int", + "description": "Fill_value is used to replace all occurrences of missing_values. Defaults to None.", + "default": "None", + }, + }, + "required": ["features"], + }, + }, + "fit": { + "description": "Fit the FillMissingValue model. ", + "parameters": { + "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, + "required": ["df"], + }, + }, + "transform": { + "description": "Transform the input DataFrame with the fitted model. ", + "parameters": { + "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, + "required": ["df"], + }, + "returns": [{"type": "pd.DataFrame", "description": "The transformed DataFrame."}], + }, + }, + } + } + schema = convert_code_to_tool_schema(DummyClass) + assert schema == expected + + +def test_convert_code_to_tool_schema_function(): + expected = { + "dummy_fn": { + "type": "function", + "description": "Analyzes a DataFrame and categorizes its columns based on data types. ", + "parameters": { + "properties": {"df": {"type": "pd.DataFrame", "description": "The DataFrame to be analyzed."}}, + "required": ["df"], + }, + } + } + schema = convert_code_to_tool_schema(dummy_fn) + assert schema == expected From 540542834ebafb0043503a7860e5b382d46b47cf Mon Sep 17 00:00:00 2001 From: yzlin Date: Sat, 20 Jan 2024 21:06:48 +0800 Subject: [PATCH 426/637] allow select tool at role initialization & restructure writecodewithtools --- metagpt/actions/write_analysis_code.py | 137 +++++++++++--------- metagpt/prompts/ml_engineer.py | 10 +- metagpt/roles/code_interpreter.py | 14 +- metagpt/roles/ml_engineer.py | 2 +- metagpt/roles/role.py | 2 +- metagpt/tools/tool_registry.py | 37 ++++-- tests/metagpt/roles/run_code_interpreter.py | 13 +- tests/metagpt/tools/test_tool_registry.py | 2 +- 8 files changed, 127 insertions(+), 90 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index cf806a986..c6e504b9e 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -22,7 +22,8 @@ from metagpt.prompts.ml_engineer import ( TOOL_USAGE_PROMPT, ) from metagpt.schema import Message, Plan -from metagpt.tools.tool_registry import TOOL_REGISTRY +from metagpt.tools import TOOL_REGISTRY +from metagpt.tools.tool_registry import validate_tool_names from metagpt.utils.common import create_func_config, remove_comments @@ -90,30 +91,29 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): class WriteCodeWithTools(BaseWriteAnalysisCode): """Write code with help of local available tools. Choose tools first, then generate code to use the tools""" - available_tools: dict = {} + # selected tools to choose from, listed by their names. En empty list means selection from all tools. + selected_tools: list[str] = [] - def __init__(self, **kwargs): - super().__init__(**kwargs) - - def _parse_recommend_tools(self, recommend_tools: list) -> dict: + def _get_tools_by_type(self, tool_type: str) -> dict: """ - Parses and validates a list of recommended tools, and retrieves their schema from registry. + Retreive tools by tool type from registry, but filtered by pre-selected tool list Args: - recommend_tools (list): A list of recommended tools. + tool_type (str): Tool type to retrieve from the registry Returns: - dict: A dict of valid tool schemas. + dict: A dict of tool name to Tool object, representing available tools under the type """ - valid_tools = [] - for tool_name in recommend_tools: - if TOOL_REGISTRY.has_tool(tool_name): - valid_tools.append(TOOL_REGISTRY.get_tool(tool_name)) + candidate_tools = TOOL_REGISTRY.get_tools_by_type(tool_type) + if self.selected_tools: + candidate_tools = { + tool_name: candidate_tools[tool_name] + for tool_name in self.selected_tools + if tool_name in candidate_tools + } + return candidate_tools - tool_catalog = {tool.name: tool.schemas for tool in valid_tools} - return tool_catalog - - async def _tool_recommendation( + async def _recommend_tool( self, task: str, code_steps: str, @@ -128,7 +128,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): available_tools (dict): the available tools description Returns: - list: recommended tools for the specified task + dict: schemas of recommended tools for the specified task """ prompt = TOOL_RECOMMENDATION_PROMPT.format( current_task=task, @@ -138,42 +138,62 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): tool_config = create_func_config(SELECT_FUNCTION_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) recommend_tools = rsp["recommend_tools"] - return recommend_tools + logger.info(f"Recommended tools: \n{recommend_tools}") + + # Parses and validates the recommended tools, for LLM might hallucinate and recommend non-existing tools + valid_tools = validate_tool_names(recommend_tools, return_tool_object=True) + + tool_schemas = {tool.name: tool.schemas for tool in valid_tools} + + return tool_schemas + + async def _prepare_tools(self, plan: Plan) -> Tuple[dict, str]: + """Prepare tool schemas and usage instructions according to current task + + Args: + plan (Plan): The overall plan containing task information. + + Returns: + Tuple[dict, str]: A tool schemas ({tool_name: tool_schema_dict}) and a usage prompt for the type of tools selected + """ + # find tool type from task type through exact match, can extend to retrieval in the future + tool_type = plan.current_task.task_type + + # prepare tool-type-specific instruction + tool_type_usage_prompt = ( + TOOL_REGISTRY.get_tool_type(tool_type).usage_prompt if TOOL_REGISTRY.has_tool_type(tool_type) else "" + ) + + # prepare schemas of available tools + tool_schemas = {} + available_tools = self._get_tools_by_type(tool_type) + if available_tools: + available_tools = {tool_name: tool.schemas["description"] for tool_name, tool in available_tools.items()} + code_steps = plan.current_task.code_steps + tool_schemas = await self._recommend_tool(plan.current_task.instruction, code_steps, available_tools) + + return tool_schemas, tool_type_usage_prompt async def run( self, context: List[Message], - plan: Plan = None, + plan: Plan, **kwargs, ) -> str: - tool_type = ( - plan.current_task.task_type - ) # find tool type from task type through exact match, can extend to retrieval in the future - available_tools = TOOL_REGISTRY.get_tools_by_type(tool_type) - special_prompt = ( - TOOL_REGISTRY.get_tool_type(tool_type).usage_prompt if TOOL_REGISTRY.has_tool_type(tool_type) else "" + # prepare tool schemas and tool-type-specific instruction + tool_schemas, tool_type_usage_prompt = await self._prepare_tools(plan=plan) + + # form a complete tool usage instruction and include it as a message in context + tools_instruction = TOOL_USAGE_PROMPT.format( + tool_schemas=tool_schemas, tool_type_usage_prompt=tool_type_usage_prompt ) - code_steps = plan.current_task.code_steps - - tool_catalog = {} - - if available_tools: - available_tools = {tool_name: tool.schemas["description"] for tool_name, tool in available_tools.items()} - - recommend_tools = await self._tool_recommendation( - plan.current_task.instruction, code_steps, available_tools - ) - tool_catalog = self._parse_recommend_tools(recommend_tools) - logger.info(f"Recommended tools: \n{recommend_tools}") - - tools_instruction = TOOL_USAGE_PROMPT.format(special_prompt=special_prompt, tool_catalog=tool_catalog) - context.append(Message(content=tools_instruction, role="user")) + # prepare prompt & LLM call prompt = self.process_msg(context) - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) + return rsp @@ -185,36 +205,25 @@ class WriteCodeWithToolsML(WriteCodeWithTools): column_info: str = "", **kwargs, ) -> Tuple[List[Message], str]: - tool_type = ( - plan.current_task.task_type - ) # find tool type from task type through exact match, can extend to retrieval in the future - available_tools = TOOL_REGISTRY.get_tools_by_type(tool_type) - special_prompt = ( - TOOL_REGISTRY.get_tool_type(tool_type).usage_prompt if TOOL_REGISTRY.has_tool_type(tool_type) else "" - ) - code_steps = plan.current_task.code_steps + # prepare tool schemas and tool-type-specific instruction + tool_schemas, tool_type_usage_prompt = await self._prepare_tools(plan=plan) + # ML-specific variables to be used in prompt + code_steps = plan.current_task.code_steps finished_tasks = plan.get_finished_tasks() code_context = [remove_comments(task.code) for task in finished_tasks] code_context = "\n\n".join(code_context) - if available_tools: - available_tools = {tool_name: tool.schemas["description"] for tool_name, tool in available_tools.items()} - - recommend_tools = await self._tool_recommendation( - plan.current_task.instruction, code_steps, available_tools - ) - tool_catalog = self._parse_recommend_tools(recommend_tools) - logger.info(f"Recommended tools: \n{recommend_tools}") - + # prepare prompt depending on tool availability & LLM call + if tool_schemas: prompt = ML_TOOL_USAGE_PROMPT.format( user_requirement=plan.goal, history_code=code_context, current_task=plan.current_task.instruction, column_info=column_info, - special_prompt=special_prompt, + tool_type_usage_prompt=tool_type_usage_prompt, code_steps=code_steps, - tool_catalog=tool_catalog, + tool_schemas=tool_schemas, ) else: @@ -223,13 +232,15 @@ class WriteCodeWithToolsML(WriteCodeWithTools): history_code=code_context, current_task=plan.current_task.instruction, column_info=column_info, - special_prompt=special_prompt, + tool_type_usage_prompt=tool_type_usage_prompt, code_steps=code_steps, ) - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) + + # Extra output to be used for potential debugging context = [Message(content=prompt, role="user")] + return context, rsp diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_engineer.py index 3fd895e6e..ac95e14bd 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_engineer.py @@ -161,7 +161,7 @@ Latest data info after previous tasks: # Task Write complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc. -Specifically, {special_prompt} +Specifically, {tool_type_usage_prompt} # Code Steps: Strictly follow steps below when you writing code if it's convenient. @@ -192,7 +192,7 @@ model.fit(train, y_train) TOOL_USAGE_PROMPT = """ # Instruction Write complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc. -Specifically, {special_prompt} +Specifically, {tool_type_usage_prompt} # Capabilities - You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class. @@ -200,7 +200,7 @@ Specifically, {special_prompt} # Available Tools (can be empty): Each Class tool is described in JSON format. When you call a tool, import the tool first. -{tool_catalog} +{tool_schemas} # Constraints: - Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. @@ -225,7 +225,7 @@ Latest data info after previous tasks: # Task Write complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc. -Specifically, {special_prompt} +Specifically, {tool_type_usage_prompt} # Code Steps: Strictly follow steps below when you writing code if it's convenient. @@ -237,7 +237,7 @@ Strictly follow steps below when you writing code if it's convenient. # Available Tools: Each Class tool is described in JSON format. When you call a tool, import the tool from its path first. -{tool_catalog} +{tool_schemas} # Output Example: when current task is "do data preprocess, like fill missing value, handle outliers, etc.", and their are two steps in 'Code Steps', the code be like: diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index f972e72e2..11ede6068 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -19,6 +19,7 @@ class CodeInterpreter(Role): make_udfs: bool = False # whether to save user-defined functions use_code_steps: bool = False execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) + tools: list[str] = [] def __init__( self, @@ -27,13 +28,20 @@ class CodeInterpreter(Role): goal="", auto_run=True, use_tools=False, - make_udfs=False, + tools=[], **kwargs, ): super().__init__( - name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools, make_udfs=make_udfs, **kwargs + name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools, tools=tools, **kwargs ) self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run, use_tools=use_tools) + if use_tools and tools: + from metagpt.tools.tool_registry import ( + validate_tool_names, # import upon use + ) + + self.tools = validate_tool_names(tools) + logger.info(f"will only use {self.tools} as tools") @property def working_memory(self): @@ -92,7 +100,7 @@ class CodeInterpreter(Role): return code["code"], result, success async def _write_code(self): - todo = WriteCodeByGenerate() if not self.use_tools else WriteCodeWithTools() + todo = WriteCodeByGenerate() if not self.use_tools else WriteCodeWithTools(selected_tools=self.tools) logger.info(f"ready to {todo.name}") context = self.planner.get_useful_memories() diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 6b671f9c2..d1a22b9d3 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -27,7 +27,7 @@ class MLEngineer(CodeInterpreter): column_info = await self._update_data_columns() logger.info("Write code with tools") - tool_context, code = await WriteCodeWithToolsML().run( + tool_context, code = await WriteCodeWithToolsML(selected_tools=self.tools).run( context=[], # context assembled inside the Action plan=self.planner.plan, column_info=column_info, diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index a2f2f2e9d..21e48a127 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -477,7 +477,7 @@ class Role(SerializationMixin, is_polymorphic_base=True): else: # update plan according to user's feedback and to take on changed tasks - await self.planner.update_plan(review) + await self.planner.update_plan() completed_plan_memory = self.planner.get_useful_memories() # completed plan as a outcome diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index fbdfb3cfd..c064a19de 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -11,6 +11,7 @@ import re from collections import defaultdict import yaml +from pydantic import BaseModel from metagpt.const import TOOL_SCHEMA_PATH from metagpt.logs import logger @@ -18,11 +19,10 @@ from metagpt.tools.tool_convert import convert_code_to_tool_schema from metagpt.tools.tool_data_type import Tool, ToolSchema, ToolType -class ToolRegistry: - def __init__(self): - self.tools = {} - self.tool_types = {} - self.tools_by_types = defaultdict(dict) # two-layer k-v, {tool_type: {tool_name: {...}, ...}, ...} +class ToolRegistry(BaseModel): + tools: dict = {} + tool_types: dict = {} + tools_by_types: dict = defaultdict(dict) # two-layer k-v, {tool_type: {tool_name: {...}, ...}, ...} def register_tool_type(self, tool_type: ToolType): self.tool_types[tool_type.name] = tool_type @@ -70,22 +70,22 @@ class ToolRegistry: self.tools_by_types[tool_type][tool_name] = tool logger.info(f"{tool_name} registered") - def has_tool(self, key): + def has_tool(self, key: str) -> Tool: return key in self.tools - def get_tool(self, key): + def get_tool(self, key) -> Tool: return self.tools.get(key) - def get_tools_by_type(self, key): - return self.tools_by_types.get(key) + def get_tools_by_type(self, key) -> dict[str, Tool]: + return self.tools_by_types.get(key, {}) - def has_tool_type(self, key): + def has_tool_type(self, key) -> bool: return key in self.tool_types - def get_tool_type(self, key): + def get_tool_type(self, key) -> ToolType: return self.tool_types.get(key) - def get_tool_types(self): + def get_tool_types(self) -> dict[str, ToolType]: return self.tool_types @@ -141,3 +141,16 @@ def make_schema(tool_source_object, include, path): print(e) return schema + + +def validate_tool_names(tools: list[str], return_tool_object=False) -> list[str]: + valid_tools = [] + for tool_name in tools: + if not TOOL_REGISTRY.has_tool(tool_name): + logger.warning( + f"Specified tool {tool_name} not found and was skipped. Check if you have registered it properly" + ) + else: + valid_tool = TOOL_REGISTRY.get_tool(tool_name) if return_tool_object else tool_name + valid_tools.append(valid_tool) + return valid_tools diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py index 539b20286..766a25998 100644 --- a/tests/metagpt/roles/run_code_interpreter.py +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -10,7 +10,7 @@ from metagpt.utils.recovery_util import load_history, save_history async def run_code_interpreter( - role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir + role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir, tools ): """ The main function to run the MLEngineer with optional history loading. @@ -25,7 +25,9 @@ async def run_code_interpreter( """ if role_class == "ci": - role = CodeInterpreter(goal=requirement, auto_run=auto_run, use_tools=use_tools, make_udfs=make_udfs) + role = CodeInterpreter( + goal=requirement, auto_run=auto_run, use_tools=use_tools, make_udfs=make_udfs, tools=tools + ) else: role = MLEngineer( goal=requirement, @@ -33,7 +35,7 @@ async def run_code_interpreter( use_tools=use_tools, use_code_steps=use_code_steps, make_udfs=make_udfs, - use_udfs=use_udfs, + tools=tools, ) if save_dir: @@ -73,6 +75,8 @@ if __name__ == "__main__": use_tools = True make_udfs = False use_udfs = False + tools = [] + # tools = ["FillMissingValue", "CatCross", "non_existing_test"] async def main( role_class: str = role_class, @@ -83,9 +87,10 @@ if __name__ == "__main__": make_udfs: bool = make_udfs, use_udfs: bool = use_udfs, save_dir: str = save_dir, + tools=tools, ): await run_code_interpreter( - role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir + role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir, tools ) fire.Fire(main) diff --git a/tests/metagpt/tools/test_tool_registry.py b/tests/metagpt/tools/test_tool_registry.py index 582c368a8..c24122e39 100644 --- a/tests/metagpt/tools/test_tool_registry.py +++ b/tests/metagpt/tools/test_tool_registry.py @@ -98,4 +98,4 @@ def test_get_tools_by_type(tool_registry, schema_yaml): # Test case for when the tool type does not exist def test_get_tools_by_nonexistent_type(tool_registry): tools_by_type = tool_registry.get_tools_by_type("NonexistentType") - assert tools_by_type is None + assert not tools_by_type From 9661c3c6810dac0c177b161f9804fd9cef40c04e Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 22 Jan 2024 09:15:17 +0800 Subject: [PATCH 427/637] update IMAGE2WEBPAGE_PROMPT for gpt_v_generator tool --- metagpt/prompts/tool_types.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/prompts/tool_types.py b/metagpt/prompts/tool_types.py index 718eefd51..42d9c1ece 100644 --- a/metagpt/prompts/tool_types.py +++ b/metagpt/prompts/tool_types.py @@ -42,5 +42,5 @@ The current task is about evaluating a model, please note the following: IMAGE2WEBPAGE_PROMPT = """ The current task is about converting image into webpage code. please note the following: - Single-Step Code Generation: Execute the entire code generation process in a single step, encompassing HTML, CSS, and JavaScript. Avoid fragmenting the code generation into multiple separate steps to maintain consistency and simplify the development workflow. -- Save webpages: Be sure to use the save method inside Vision. +- Save webpages: Be sure to use the save method provided. """ From 8084fca1d0642076e1e73145ca03be22e64eeeaa Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 22 Jan 2024 10:18:25 +0800 Subject: [PATCH 428/637] change default value of language_value. --- metagpt/provider/openai_api.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 7bc4ee164..72af5f40a 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -204,7 +204,7 @@ class OpenAILLM(BaseLLM): # 匹配language language_pattern = re.compile(r'[\"\']?language[\"\']?\s*:\s*["\']([^"\']+?)["\']', re.DOTALL) language_match = language_pattern.search(arguments) - language_value = language_match.group(1) if language_match else None + language_value = language_match.group(1) if language_match else "python" # 匹配code code_pattern = r'(["\'`]{3}|["\'`])([\s\S]*?)\1' @@ -217,7 +217,7 @@ class OpenAILLM(BaseLLM): if code_value is None: raise ValueError(f"Parse code error for {arguments}") # arguments只有code的情况 - return {"language": language_value or "python", "code": code_value} + return {"language": language_value, "code": code_value} @handle_exception def get_choice_function_arguments(self, rsp: ChatCompletion) -> dict: From 8b84e269a1e803b852e7d8e319f8579e6b10de4b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Mon, 22 Jan 2024 10:30:38 +0800 Subject: [PATCH 429/637] feat: remove error print --- metagpt/document_store/faiss_store.py | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/metagpt/document_store/faiss_store.py b/metagpt/document_store/faiss_store.py index 54585dcfc..2359917d5 100644 --- a/metagpt/document_store/faiss_store.py +++ b/metagpt/document_store/faiss_store.py @@ -37,11 +37,7 @@ class FaissStore(LocalStore): return FAISS.load_local(self.raw_data_path.parent, self.embedding, self.fname) def _write(self, docs, metadatas): - try: - store = FAISS.from_texts(docs, self.embedding, metadatas=metadatas) - except Exception as e: - logger.error(f"Failed to write. error: {e}") - raise e + store = FAISS.from_texts(docs, self.embedding, metadatas=metadatas) return store def persist(self): From bda2e06d368e1c80306d441a7ce80da57d5d62f3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Mon, 22 Jan 2024 10:50:32 +0800 Subject: [PATCH 430/637] feat: +note --- metagpt/roles/role.py | 13 ++++++++++--- 1 file changed, 10 insertions(+), 3 deletions(-) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index c363d332c..28d2fe693 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -523,19 +523,26 @@ class Role(SerializationMixin, ContextMixin, BaseModel): return not self.rc.news and not self.rc.todo and self.rc.msg_buffer.empty() async def think(self) -> Action: - """The exported `think` function""" - await self._observe() + """ + Export SDK API, used by AgentStore RPC. + The exported `think` function + """ + await self._observe() # For compatibility with the old version of the Agent. await self._think() return self.rc.todo async def act(self) -> ActionOutput: - """The exported `act` function""" + """ + Export SDK API, used by AgentStore RPC. + The exported `act` function + """ msg = await self._act() return ActionOutput(content=msg.content, instruct_content=msg.instruct_content) @property def action_description(self) -> str: """ + Export SDK API, used by AgentStore RPC. AgentStore uses this attribute to display to the user what actions the current role should take. """ if self.rc.todo: From 525d94317cdcfb191280b2ac8485edc500e2b8f6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Mon, 22 Jan 2024 10:52:36 +0800 Subject: [PATCH 431/637] feat: +note --- metagpt/roles/role.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 28d2fe693..53798f385 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -524,7 +524,7 @@ class Role(SerializationMixin, ContextMixin, BaseModel): async def think(self) -> Action: """ - Export SDK API, used by AgentStore RPC. + Export SDK API, used by AgentStore RPC and Agent. The exported `think` function """ await self._observe() # For compatibility with the old version of the Agent. @@ -533,7 +533,7 @@ class Role(SerializationMixin, ContextMixin, BaseModel): async def act(self) -> ActionOutput: """ - Export SDK API, used by AgentStore RPC. + Export SDK API, used by AgentStore RPC and Agent. The exported `act` function """ msg = await self._act() @@ -542,7 +542,7 @@ class Role(SerializationMixin, ContextMixin, BaseModel): @property def action_description(self) -> str: """ - Export SDK API, used by AgentStore RPC. + Export SDK API, used by AgentStore RPC and Agent. AgentStore uses this attribute to display to the user what actions the current role should take. """ if self.rc.todo: From f15b772d77d76a92a8214cf2b5d4f071de9d8a10 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Mon, 22 Jan 2024 10:54:15 +0800 Subject: [PATCH 432/637] feat: +note --- metagpt/roles/role.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 53798f385..dba14bb72 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -524,7 +524,7 @@ class Role(SerializationMixin, ContextMixin, BaseModel): async def think(self) -> Action: """ - Export SDK API, used by AgentStore RPC and Agent. + Export SDK API, used by AgentStore RPC. The exported `think` function """ await self._observe() # For compatibility with the old version of the Agent. @@ -533,7 +533,7 @@ class Role(SerializationMixin, ContextMixin, BaseModel): async def act(self) -> ActionOutput: """ - Export SDK API, used by AgentStore RPC and Agent. + Export SDK API, used by AgentStore RPC. The exported `act` function """ msg = await self._act() From 85465fe500121f65b9c38312034926b7754a0a17 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Mon, 22 Jan 2024 11:04:03 +0800 Subject: [PATCH 433/637] feat: +note --- metagpt/roles/role.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index dba14bb72..3747072f1 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -544,6 +544,8 @@ class Role(SerializationMixin, ContextMixin, BaseModel): """ Export SDK API, used by AgentStore RPC and Agent. AgentStore uses this attribute to display to the user what actions the current role should take. + `Role` provides the default property, and this property should be overridden by children classes if necessary, + as demonstrated by the `Engineer` class. """ if self.rc.todo: if self.rc.todo.desc: From 0f245f530ece957a460773d7ec74cd158cb47ae7 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 22 Jan 2024 13:57:52 +0800 Subject: [PATCH 434/637] logging --- metagpt/tools/tool_convert.py | 2 +- metagpt/tools/tool_registry.py | 3 +-- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/metagpt/tools/tool_convert.py b/metagpt/tools/tool_convert.py index c2ea33085..fdb69bfb3 100644 --- a/metagpt/tools/tool_convert.py +++ b/metagpt/tools/tool_convert.py @@ -44,7 +44,7 @@ def docstring_to_schema(docstring: str): # variable_pattern = re.compile(r"(\w+)\s*\((.*?)\):\s*(.*)") variable_pattern = re.compile( r"(\w+)\s*\((.*?)\):\s*(.*?)(?=\n\s*\w+\s*\(|\Z)", re.DOTALL - ) # (?=\n\w+\s*\(|\Z) isb to assert that what follows is either the start of the next parameter (indicated by a newline, some word characters, and an opening parenthesis) or the end of the string (\Z). + ) # (?=\n\w+\s*\(|\Z) is to assert that what follows is either the start of the next parameter (indicated by a newline, some word characters, and an opening parenthesis) or the end of the string (\Z). params = variable_pattern.findall(_args) parameter_schema = {"properties": {}, "required": []} diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index c064a19de..d16defa0a 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -137,8 +137,7 @@ def make_schema(tool_source_object, include, path): logger.info(f"schema made at {path}") except Exception as e: schema = {} - logger.error("Fail to make schema") - print(e) + logger.error(f"Fail to make schema: {e}") return schema From 5ddaaaa3471e096b5ea02f2e2b8a4cc34f50332a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 22 Jan 2024 14:56:44 +0800 Subject: [PATCH 435/637] add test: test_get_choice_function_arguments_for_aask_code. --- tests/metagpt/provider/test_openai.py | 139 +++++++++----------------- 1 file changed, 48 insertions(+), 91 deletions(-) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index 8de29c11b..7af2f6892 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -1,104 +1,21 @@ from unittest.mock import Mock import pytest +from openai.types.chat import ( + ChatCompletion, + ChatCompletionMessage, + ChatCompletionMessageToolCall, +) +from openai.types.chat.chat_completion import Choice +from openai.types.chat.chat_completion_message_tool_call import Function from metagpt.config import CONFIG +from metagpt.logs import logger from metagpt.provider.openai_api import OpenAILLM -from metagpt.schema import UserMessage CONFIG.openai_proxy = None -@pytest.mark.asyncio -async def test_aask_code(): - llm = OpenAILLM() - msg = [{"role": "user", "content": "Write a python hello world code."}] - rsp = await llm.aask_code(msg) # -> {'language': 'python', 'code': "print('Hello, World!')"} - assert "language" in rsp - assert "code" in rsp - assert len(rsp["code"]) > 0 - - -@pytest.mark.asyncio -async def test_aask_code_str(): - llm = OpenAILLM() - msg = "Write a python hello world code." - rsp = await llm.aask_code(msg) # -> {'language': 'python', 'code': "print('Hello, World!')"} - assert "language" in rsp - assert "code" in rsp - assert len(rsp["code"]) > 0 - - -@pytest.mark.asyncio -async def test_aask_code_Message(): - llm = OpenAILLM() - msg = UserMessage("Write a python hello world code.") - rsp = await llm.aask_code(msg) # -> {'language': 'python', 'code': "print('Hello, World!')"} - assert "language" in rsp - assert "code" in rsp - assert len(rsp["code"]) > 0 - - -def test_ask_code(): - llm = OpenAIGPTAPI() - msg = [{"role": "user", "content": "Write a python hello world code."}] - rsp = llm.ask_code(msg) # -> {'language': 'python', 'code': "print('Hello, World!')"} - assert "language" in rsp - assert "code" in rsp - assert len(rsp["code"]) > 0 - - -def test_ask_code_str(): - llm = OpenAIGPTAPI() - msg = "Write a python hello world code." - rsp = llm.ask_code(msg) # -> {'language': 'python', 'code': "print('Hello, World!')"} - assert "language" in rsp - assert "code" in rsp - assert len(rsp["code"]) > 0 - - -def test_ask_code_Message(): - llm = OpenAIGPTAPI() - msg = UserMessage("Write a python hello world code.") - rsp = llm.ask_code(msg) # -> {'language': 'python', 'code': "print('Hello, World!')"} - assert "language" in rsp - assert "code" in rsp - assert len(rsp["code"]) > 0 - - -def test_ask_code_list_Message(): - llm = OpenAIGPTAPI() - msg = [UserMessage("a=[1,2,5,10,-10]"), UserMessage("写出求a中最大值的代码python")] - rsp = llm.ask_code(msg) # -> {'language': 'python', 'code': 'max_value = max(a)\nmax_value'} - assert "language" in rsp - assert "code" in rsp - assert len(rsp["code"]) > 0 - - -def test_ask_code_list_str(): - llm = OpenAIGPTAPI() - msg = ["a=[1,2,5,10,-10]", "写出求a中最大值的代码python"] - rsp = llm.ask_code(msg) # -> {'language': 'python', 'code': 'max_value = max(a)\nmax_value'} - print(rsp) - assert "language" in rsp - assert "code" in rsp - assert len(rsp["code"]) > 0 - - -@pytest.mark.asyncio -async def test_ask_code_steps2(): - llm = OpenAIGPTAPI() - msg = ["step by setp 生成代码: Step 1. 先生成随机数组a, Step 2. 求a中最大值, Step 3. 绘制数据a的直方图"] - rsp = await llm.aask_code(msg) # -> {'language': 'python', 'code': 'max_value = max(a)\nmax_value'} - print(rsp) - assert "language" in rsp - assert "code" in rsp - assert len(rsp["code"]) > 0 - assert "Step 1" in rsp["code"] - assert "Step 2" in rsp["code"] - assert "Step 3" in rsp["code"] - - class TestOpenAI: @pytest.fixture def config(self): @@ -146,6 +63,32 @@ class TestOpenAI: openai_api_type="azure", ) + @pytest.fixture + def tool_calls_rsp(self): + function_rsps = [ + Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), + Function(arguments='{\n"language": "python",\n"code": ```print("hello world")```', name="execute"), + Function(arguments='{\n"language": "python",\n"code": \'print("hello world")\'', name="execute"), + Function(arguments='{\n"language": \'python\',\n"code": "print(\'hello world\')"', name="execute"), + Function(arguments='\nprint("hello world")\\n', name="execute"), + ] + tool_calls = [ + ChatCompletionMessageToolCall(type="function", id=f"call_{i}", function=f) + for i, f in enumerate(function_rsps) + ] + messages = [ChatCompletionMessage(content=None, role="assistant", tool_calls=[t]) for t in tool_calls] + # 添加一个纯文本响应 + messages.append( + ChatCompletionMessage(content="Completed a python code for hello world!", role="assistant", tool_calls=None) + ) + choices = [ + Choice(finish_reason="tool_calls", logprobs=None, index=i, message=msg) for i, msg in enumerate(messages) + ] + return [ + ChatCompletion(id=str(i), choices=[c], created=i, model="gpt-4", object="chat.completion") + for i, c in enumerate(choices) + ] + def test_make_client_kwargs_without_proxy(self, config): instance = OpenAILLM() instance.config = config @@ -171,3 +114,17 @@ class TestOpenAI: instance.config = config_azure_proxy kwargs = instance._make_client_kwargs() assert "http_client" in kwargs + + def test_get_choice_function_arguments_for_aask_code(self, tool_calls_rsp): + instance = OpenAILLM() + for i, rsp in enumerate(tool_calls_rsp): + code = instance.get_choice_function_arguments(rsp) + logger.info(f"\ntest get function call arguments {i}: {code}") + assert "code" in code + assert "language" in code + assert "hello world" in code["code"] + + if "Completed a python code for hello world!" == code["code"]: + code["language"] == "markdown" + else: + code["language"] == "python" From b4e09341b354d12d419977eb5907d310eb3d226a Mon Sep 17 00:00:00 2001 From: Arnaud Gelas Date: Thu, 18 Jan 2024 20:47:33 +0100 Subject: [PATCH 436/637] Stop generating unit test for non python files When trying to create a simple HelloWorld with test, metagpt creates test for README.md --- metagpt/roles/qa_engineer.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/metagpt/roles/qa_engineer.py b/metagpt/roles/qa_engineer.py index 0e323893e..45a1c7715 100644 --- a/metagpt/roles/qa_engineer.py +++ b/metagpt/roles/qa_engineer.py @@ -65,6 +65,8 @@ class QaEngineer(Role): code_doc = await src_file_repo.get(filename) if not code_doc: continue + if not code_doc.filename.endswith(".py"): + continue test_doc = await tests_file_repo.get("test_" + code_doc.filename) if not test_doc: test_doc = Document( From 9b3987ff296ef1439e1f1259d4d603db9dfeb61c Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 22 Jan 2024 14:58:06 +0800 Subject: [PATCH 437/637] add docstring parser --- metagpt/tools/tool_convert.py | 41 +++++---------- metagpt/utils/parse_docstring.py | 87 ++++++++++++++++++++++++++++++++ 2 files changed, 101 insertions(+), 27 deletions(-) create mode 100644 metagpt/utils/parse_docstring.py diff --git a/metagpt/tools/tool_convert.py b/metagpt/tools/tool_convert.py index fdb69bfb3..b8377e67a 100644 --- a/metagpt/tools/tool_convert.py +++ b/metagpt/tools/tool_convert.py @@ -1,9 +1,6 @@ import inspect -import re - -def remove_spaces(text): - return re.sub(r"\s+", " ", text) +from metagpt.utils.parse_docstring import GoogleDocstringParser, remove_spaces def convert_code_to_tool_schema(obj, include: list[str] = []): @@ -34,45 +31,35 @@ def docstring_to_schema(docstring: str): if docstring is None: return {} + parser = GoogleDocstringParser(docstring=docstring) + # 匹配简介部分 - description_match = re.search(r"^(.*?)(?:Args:|Returns:|Raises:|$)", docstring, re.DOTALL) - description = remove_spaces(description_match.group(1)) if description_match else "" + description = parser.parse_desc() # 匹配Args部分 - args_match = re.search(r"Args:\s*(.*?)(?:Returns:|Raises:|$)", docstring, re.DOTALL) - _args = args_match.group(1).strip() if args_match else "" - # variable_pattern = re.compile(r"(\w+)\s*\((.*?)\):\s*(.*)") - variable_pattern = re.compile( - r"(\w+)\s*\((.*?)\):\s*(.*?)(?=\n\s*\w+\s*\(|\Z)", re.DOTALL - ) # (?=\n\w+\s*\(|\Z) is to assert that what follows is either the start of the next parameter (indicated by a newline, some word characters, and an opening parenthesis) or the end of the string (\Z). - - params = variable_pattern.findall(_args) + params = parser.parse_params() parameter_schema = {"properties": {}, "required": []} for param in params: param_name, param_type, param_desc = param # check required or optional - if "optional" in param_type: - param_type = param_type.replace(", optional", "") - else: + is_optional, param_type = parser.check_and_parse_optional(param_type) + if not is_optional: parameter_schema["required"].append(param_name) # type and desc param_dict = {"type": param_type, "description": remove_spaces(param_desc)} # match Default for optional args - default_val = re.search(r"Defaults to (.+?)\.", param_desc) - if default_val: - param_dict["default"] = default_val.group(1) + has_default_val, default_val = parser.check_and_parse_default_value(param_desc) + if has_default_val: + param_dict["default"] = default_val # match Enum - enum_val = re.search(r"Enum: \[(.+?)\]", param_desc) - if enum_val: - param_dict["enum"] = [e.strip() for e in enum_val.group(1).split(",")] + has_enum, enum_vals = parser.check_and_parse_enum(param_desc) + if has_enum: + param_dict["enum"] = enum_vals # add to parameter schema parameter_schema["properties"].update({param_name: param_dict}) # 匹配Returns部分 - returns_match = re.search(r"Returns:\s*(.*?)(?:Raises:|$)", docstring, re.DOTALL) - returns = returns_match.group(1).strip() if returns_match else "" - return_pattern = re.compile(r"^(.*)\s*:\s*(.*)$") - returns = return_pattern.findall(returns) + returns = parser.parse_returns() # 构建YAML字典 schema = { diff --git a/metagpt/utils/parse_docstring.py b/metagpt/utils/parse_docstring.py new file mode 100644 index 000000000..970257676 --- /dev/null +++ b/metagpt/utils/parse_docstring.py @@ -0,0 +1,87 @@ +import re +from typing import Tuple + +from pydantic import BaseModel + + +def remove_spaces(text): + return re.sub(r"\s+", " ", text) + + +class DocstringParser(BaseModel): + docstring: str + + def parse_desc(self) -> str: + """Parse and return the description from the docstring.""" + + def parse_params(self) -> list[Tuple[str, str, str]]: + """Parse and return the parameters from the docstring. + + Returns: + list[Tuple[str, str, str]]: A list of input paramter info. Each info is a triple of (param name, param type, param description) + """ + + def parse_returns(self) -> list[Tuple[str, str]]: + """Parse and return the return information from the docstring. + + Returns: + list[Tuple[str, str, str]]: A list of output info. Each info is a tuple of (return type, return description) + """ + + @staticmethod + def check_and_parse_optional(param_type: str) -> Tuple[bool, str]: + """Check if a parameter is optional and return a processed param_type rid of the optionality info if so""" + + @staticmethod + def check_and_parse_default_value(param_desc: str) -> Tuple[bool, str]: + """Check if a parameter has a default value and return the default value if so""" + + @staticmethod + def check_and_parse_enum(param_desc: str) -> Tuple[bool, str]: + """Check if a parameter description includes an enum and return enum values if so""" + + +class reSTDocstringParser(DocstringParser): + """A parser for reStructuredText (reST) docstring""" + + +class GoogleDocstringParser(DocstringParser): + """A parser for Google-stype docstring""" + + docstring: str + + def parse_desc(self) -> str: + description_match = re.search(r"^(.*?)(?:Args:|Returns:|Raises:|$)", self.docstring, re.DOTALL) + description = remove_spaces(description_match.group(1)) if description_match else "" + return description + + def parse_params(self) -> list[Tuple[str, str, str]]: + args_match = re.search(r"Args:\s*(.*?)(?:Returns:|Raises:|$)", self.docstring, re.DOTALL) + _args = args_match.group(1).strip() if args_match else "" + # variable_pattern = re.compile(r"(\w+)\s*\((.*?)\):\s*(.*)") + variable_pattern = re.compile( + r"(\w+)\s*\((.*?)\):\s*(.*?)(?=\n\s*\w+\s*\(|\Z)", re.DOTALL + ) # (?=\n\w+\s*\(|\Z) is to assert that what follows is either the start of the next parameter (indicated by a newline, some word characters, and an opening parenthesis) or the end of the string (\Z). + params = variable_pattern.findall(_args) + return params + + def parse_returns(self) -> list[Tuple[str, str]]: + returns_match = re.search(r"Returns:\s*(.*?)(?:Raises:|$)", self.docstring, re.DOTALL) + returns = returns_match.group(1).strip() if returns_match else "" + return_pattern = re.compile(r"^(.*)\s*:\s*(.*)$") + returns = return_pattern.findall(returns) + return returns + + @staticmethod + def check_and_parse_optional(param_type: str) -> Tuple[bool, str]: + return "optional" in param_type, param_type.replace(", optional", "") + + @staticmethod + def check_and_parse_default_value(param_desc: str) -> Tuple[bool, str]: + default_val = re.search(r"Defaults to (.+?)\.", param_desc) + return (True, default_val.group(1)) if default_val else (False, "") + + @staticmethod + def check_and_parse_enum(param_desc: str) -> Tuple[bool, str]: + enum_val = re.search(r"Enum: \[(.+?)\]", param_desc) + return (True, [e.strip() for e in enum_val.group(1).split(",")]) if enum_val else (False, []) From 33e13b677b908899cfe79e7b3a62cb1e5c1e0f62 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 22 Jan 2024 15:01:44 +0800 Subject: [PATCH 438/637] typo --- metagpt/utils/parse_docstring.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/utils/parse_docstring.py b/metagpt/utils/parse_docstring.py index 970257676..8a017e1f7 100644 --- a/metagpt/utils/parse_docstring.py +++ b/metagpt/utils/parse_docstring.py @@ -22,10 +22,10 @@ class DocstringParser(BaseModel): """ def parse_returns(self) -> list[Tuple[str, str]]: - """Parse and return the return information from the docstring. + """Parse and return the output information from the docstring. Returns: - list[Tuple[str, str, str]]: A list of output info. Each info is a tuple of (return type, return description) + list[Tuple[str, str]]: A list of output info. Each info is a tuple of (return type, return description) """ @staticmethod From 6a9bd4a3914b565e98b04752d11cc58ee1f4afe2 Mon Sep 17 00:00:00 2001 From: Arnaud Gelas Date: Thu, 18 Jan 2024 20:48:35 +0100 Subject: [PATCH 439/637] Do not try installing requirements if there are none Do not try running pip install -r requirements.txt if the file does not exist or is empty. It avoids seeing an error in the log. --- metagpt/actions/run_code.py | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/run_code.py b/metagpt/actions/run_code.py index 30b06f1a6..4a26e5137 100644 --- a/metagpt/actions/run_code.py +++ b/metagpt/actions/run_code.py @@ -16,6 +16,7 @@ class. """ import subprocess +from pathlib import Path from typing import Tuple from pydantic import Field @@ -152,11 +153,23 @@ class RunCode(Action): return subprocess.run(cmd, check=check, cwd=cwd, env=env) @staticmethod - def _install_dependencies(working_directory, env): + def _install_requirements(working_directory, env): + file_path = Path(working_directory) / "requirements.txt" + if not file_path.exists(): + return + if file_path.stat().st_size == 0: + return install_command = ["python", "-m", "pip", "install", "-r", "requirements.txt"] logger.info(" ".join(install_command)) RunCode._install_via_subprocess(install_command, check=True, cwd=working_directory, env=env) + @staticmethod + def _install_pytest(working_directory, env): install_pytest_command = ["python", "-m", "pip", "install", "pytest"] logger.info(" ".join(install_pytest_command)) RunCode._install_via_subprocess(install_pytest_command, check=True, cwd=working_directory, env=env) + + @staticmethod + def _install_dependencies(working_directory, env): + RunCode._install_requirements(working_directory, env) + RunCode._install_pytest(working_directory, env) From 6cb2910d144c56ccd2ef84c223cd9125cbf22a62 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 22 Jan 2024 15:29:28 +0800 Subject: [PATCH 440/637] fix: now present the results of failure and success code in different ways. --- metagpt/actions/execute_code.py | 32 ++++++++++++++++---------------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 5b6cba57d..851794b91 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -15,14 +15,13 @@ import nbformat from nbclient import NotebookClient from nbclient.exceptions import CellTimeoutError, DeadKernelError from nbformat import NotebookNode -from nbformat.v4 import new_code_cell, new_output, new_markdown_cell -from rich.console import Console -from rich.syntax import Syntax +from nbformat.v4 import new_code_cell, new_markdown_cell, new_output +from rich.box import MINIMAL +from rich.console import Console, Group +from rich.live import Live from rich.markdown import Markdown from rich.panel import Panel -from rich.box import MINIMAL -from rich.live import Live -from rich.console import Group +from rich.syntax import Syntax from metagpt.actions import Action from metagpt.logs import logger @@ -229,7 +228,7 @@ class ExecutePyCode(ExecuteCode, Action): # code success outputs = self.parse_outputs(self.nb.cells[-1].outputs) return truncate(remove_escape_and_color_codes(outputs), is_success=success) - elif language == 'markdown': + elif language == "markdown": # markdown self.add_markdown_cell(code) return code, True @@ -238,26 +237,27 @@ class ExecutePyCode(ExecuteCode, Action): def truncate(result: str, keep_len: int = 2000, is_success: bool = True): - desc = f"Executed code {'successfully' if is_success else 'failed, please reflect the cause of bug and then debug'}" + """执行失败的代码, 展示result后keep_len个字符; 执行成功的代码, 展示result前keep_len个字符。""" + desc = f"Executed code {'successfully. ' if is_success else 'failed, please reflect the cause of bug and then debug. '}" if is_success: - desc += f"Truncated to show only {keep_len} characters\n" + desc += f"Truncated to show only first {keep_len} characters\n" else: - desc += "Show complete information for you." + desc += f"Truncated to show only last {keep_len} characters\n" if result.startswith(desc): result = result[len(desc) :] if len(result) > keep_len: - result = result[-keep_len:] if not is_success else result + result = result[-keep_len:] if not is_success else result[:keep_len] if not result: - result = 'No output about your code. Only when importing packages it is normal case. Recap and go ahead.' + result = "No output about your code. Only when importing packages it is normal case. Recap and go ahead." return result, False if result.strip().startswith(" Date: Mon, 22 Jan 2024 15:36:25 +0800 Subject: [PATCH 441/637] add test. --- tests/metagpt/actions/test_execute_code.py | 50 +++++++++------------- 1 file changed, 20 insertions(+), 30 deletions(-) diff --git a/tests/metagpt/actions/test_execute_code.py b/tests/metagpt/actions/test_execute_code.py index 904cc3c58..ecddccf6f 100644 --- a/tests/metagpt/actions/test_execute_code.py +++ b/tests/metagpt/actions/test_execute_code.py @@ -52,42 +52,21 @@ async def test_plotting_code(): # 显示图形 plt.show() + plt.close() """ output = await pi.run(code) assert output[1] is True -@pytest.mark.asyncio -async def test_plotting_bug(): - code = """ - import matplotlib.pyplot as plt - import seaborn as sns - import pandas as pd - from sklearn.datasets import load_iris - # Load the Iris dataset - iris_data = load_iris() - # Convert the loaded Iris dataset into a DataFrame for easier manipulation - iris_df = pd.DataFrame(iris_data['data'], columns=iris_data['feature_names']) - # Add a column for the target - iris_df['species'] = pd.Categorical.from_codes(iris_data['target'], iris_data['target_names']) - # Set the style of seaborn - sns.set(style='whitegrid') - # Create a pairplot of the iris dataset - plt.figure(figsize=(10, 8)) - pairplot = sns.pairplot(iris_df, hue='species') - # Show the plot - plt.show() - """ - pi = ExecutePyCode() - output = await pi.run(code) - assert output[1] is True - - def test_truncate(): - output = "hello world" - assert truncate(output) == output - output = "hello world" - assert truncate(output, 5) == "Truncated to show only the last 5 characters\nworld" + # 代码执行成功 + output, is_success = truncate("hello world", 5, True) + assert "Truncated to show only first 5 characters\nhello" in output + assert is_success + # 代码执行失败 + output, is_success = truncate("hello world", 5, False) + assert "Truncated to show only last 5 characters\nworld" in output + assert not is_success @pytest.mark.asyncio @@ -97,3 +76,14 @@ async def test_run_with_timeout(): message, success = await pi.run(code) assert not success assert message.startswith("Cell execution timed out") + + +@pytest.mark.asyncio +async def test_run_code_text(): + pi = ExecutePyCode() + message, success = await pi.run(code='print("This is a code!")', language="python") + assert success + assert message == "This is a code!\n" + message, success = await pi.run(code="# This is a code!", language="markdown") + assert success + assert message == "# This is a code!" From f3d295787eeaeeb14b17424a32b21b48470b4adc Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 22 Jan 2024 15:37:00 +0800 Subject: [PATCH 442/637] resolve zhipu 2.0.1 --- .../metagpt/provider/zhipuai/test_zhipu_model_api.py | 11 +---------- 1 file changed, 1 insertion(+), 10 deletions(-) diff --git a/tests/metagpt/provider/zhipuai/test_zhipu_model_api.py b/tests/metagpt/provider/zhipuai/test_zhipu_model_api.py index abaafb402..15673c51c 100644 --- a/tests/metagpt/provider/zhipuai/test_zhipu_model_api.py +++ b/tests/metagpt/provider/zhipuai/test_zhipu_model_api.py @@ -6,8 +6,6 @@ from typing import Any, Tuple import pytest import zhipuai -from zhipuai.model_api.api import InvokeType -from zhipuai.utils.http_client import headers as zhipuai_default_headers from metagpt.provider.zhipuai.zhipu_model_api import ZhiPuModelAPI @@ -23,14 +21,7 @@ async def mock_requestor_arequest(self, **kwargs) -> Tuple[Any, Any, str]: @pytest.mark.asyncio async def test_zhipu_model_api(mocker): - header = ZhiPuModelAPI.get_header() - zhipuai_default_headers.update({"Authorization": api_key}) - assert header == zhipuai_default_headers - - ZhiPuModelAPI.get_sse_header() - # assert len(sse_header["Authorization"]) == 191 - - url_prefix, url_suffix = ZhiPuModelAPI.split_zhipu_api_url(InvokeType.SYNC, kwargs={"model": "chatglm_turbo"}) + url_prefix, url_suffix = ZhiPuModelAPI(api_key=api_key).split_zhipu_api_url() assert url_prefix == "https://open.bigmodel.cn/api" assert url_suffix == "/paas/v4/chat/completions" From 47af1967b46370da055c42cf39ea4f0a70ba542b Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Mon, 22 Jan 2024 15:45:19 +0800 Subject: [PATCH 443/637] fix pybrowsers not found --- setup.py | 1 + 1 file changed, 1 insertion(+) diff --git a/setup.py b/setup.py index b3fd62178..1ba08c636 100644 --- a/setup.py +++ b/setup.py @@ -48,6 +48,7 @@ extras_require["test"] = [ "grpcio-status==1.48.2", "mock==5.1.0", "pylint==3.0.3", + "pybrowsers", ] extras_require["pyppeteer"] = [ From 1793a5fec64cdb624b4b425f7c6798ea7a5627af Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 22 Jan 2024 16:09:23 +0800 Subject: [PATCH 444/637] update function_rsps. --- tests/metagpt/provider/test_openai.py | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index 7af2f6892..2e5799475 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -66,11 +66,17 @@ class TestOpenAI: @pytest.fixture def tool_calls_rsp(self): function_rsps = [ - Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), - Function(arguments='{\n"language": "python",\n"code": ```print("hello world")```', name="execute"), - Function(arguments='{\n"language": "python",\n"code": \'print("hello world")\'', name="execute"), - Function(arguments='{\n"language": \'python\',\n"code": "print(\'hello world\')"', name="execute"), + Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"}', name="execute"), + Function(arguments='{\n"language": "python",\n"code": \'print("hello world")\'}', name="execute"), + Function(arguments='{\n"language": \'python\',\n"code": "print(\'hello world\')"}', name="execute"), + Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"}', name="execute"), + Function(arguments='{\n"language": "python",\n"code": ```print("hello world")```}', name="execute"), + Function(arguments='{\n"language": "python",\n"code": """print("hello world")"""}', name="execute"), Function(arguments='\nprint("hello world")\\n', name="execute"), + # only `{` in arguments + Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), + # no `{`, `}` in arguments + Function(arguments='\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), ] tool_calls = [ ChatCompletionMessageToolCall(type="function", id=f"call_{i}", function=f) From 64a296a29d321e4d05c1b0473a073dc05ee2bb1e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 22 Jan 2024 16:14:59 +0800 Subject: [PATCH 445/637] update logger warning for JSONDecodeError. --- metagpt/provider/openai_api.py | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 72af5f40a..3358b3aad 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -237,9 +237,13 @@ class OpenAILLM(BaseLLM): try: return json.loads(message.tool_calls[0].function.arguments, strict=False) except json.decoder.JSONDecodeError as e: - logger.debug( - f"Got JSONDecodeError for {message.tool_calls[0].function.arguments},\ - we will use RegExp to parse code, \n {e}" + logger.warning( + "\n".join( + [ + (f"Got JSONDecodeError for \n{'--'*40} \n{message.tool_calls[0].function.arguments}"), + (f"{'--'*40}\nwe will use RegExp to parse code. JSONDecodeError is: {e}"), + ] + ) ) return self._parse_arguments(message.tool_calls[0].function.arguments) elif message.tool_calls is None and message.content is not None: From e8b3e6762b494d1b2117a4cfac930e95dd1f6528 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Mon, 22 Jan 2024 17:13:20 +0800 Subject: [PATCH 446/637] feat: replace global CONTEXT with Config() fixbug: unit test --- metagpt/actions/write_code_review.py | 2 +- metagpt/config2.py | 1 + metagpt/context.py | 4 ---- metagpt/context_mixin.py | 6 +++--- metagpt/learn/skill_loader.py | 7 ++++--- metagpt/llm.py | 9 +++++---- metagpt/roles/assistant.py | 3 +-- metagpt/startup.py | 6 ++++-- metagpt/team.py | 17 ++++++++++++----- tests/data/audio/hello.mp3 | Bin 0 -> 31391 bytes tests/metagpt/roles/test_engineer.py | 2 +- .../metagpt/serialize_deserialize/test_team.py | 4 ++++ tests/metagpt/test_context.py | 7 ++++--- tests/metagpt/test_environment.py | 7 +++---- tests/metagpt/test_role.py | 2 +- 15 files changed, 44 insertions(+), 33 deletions(-) create mode 100644 tests/data/audio/hello.mp3 diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index ec56afc61..8fe2cc5b5 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -161,7 +161,7 @@ class WriteCodeReview(Action): format_example=format_example, ) len1 = len(iterative_code) if iterative_code else 0 - len2 = len(self.context.code_doc.content) if self.context.code_doc.content else 0 + len2 = len(self.i_context.code_doc.content) if self.i_context.code_doc.content else 0 logger.info( f"Code review and rewrite {self.i_context.code_doc.filename}: {i + 1}/{k} | len(iterative_code)={len1}, " f"len(self.i_context.code_doc.content)={len2}" diff --git a/metagpt/config2.py b/metagpt/config2.py index 92dd98bad..5a556cc52 100644 --- a/metagpt/config2.py +++ b/metagpt/config2.py @@ -38,6 +38,7 @@ class CLIParams(BaseModel): if self.project_path: self.inc = True self.project_name = self.project_name or Path(self.project_path).name + return self class Config(CLIParams, YamlModel): diff --git a/metagpt/context.py b/metagpt/context.py index 8e9749d66..3dfd52d58 100644 --- a/metagpt/context.py +++ b/metagpt/context.py @@ -95,7 +95,3 @@ class Context(BaseModel): if llm.cost_manager is None: llm.cost_manager = self.cost_manager return llm - - -# Global context, not in Env -CONTEXT = Context() diff --git a/metagpt/context_mixin.py b/metagpt/context_mixin.py index 1d239d2e4..bdf2d0734 100644 --- a/metagpt/context_mixin.py +++ b/metagpt/context_mixin.py @@ -10,7 +10,7 @@ from typing import Optional from pydantic import BaseModel, ConfigDict, Field from metagpt.config2 import Config -from metagpt.context import CONTEXT, Context +from metagpt.context import Context from metagpt.provider.base_llm import BaseLLM @@ -34,7 +34,7 @@ class ContextMixin(BaseModel): def __init__( self, - context: Optional[Context] = CONTEXT, + context: Optional[Context] = None, config: Optional[Config] = None, llm: Optional[BaseLLM] = None, **kwargs, @@ -81,7 +81,7 @@ class ContextMixin(BaseModel): """Role context: role context > context""" if self.private_context: return self.private_context - return CONTEXT + return Context() @context.setter def context(self, context: Context) -> None: diff --git a/metagpt/learn/skill_loader.py b/metagpt/learn/skill_loader.py index ddcd7ccba..bcf28bb87 100644 --- a/metagpt/learn/skill_loader.py +++ b/metagpt/learn/skill_loader.py @@ -13,7 +13,7 @@ import aiofiles import yaml from pydantic import BaseModel, Field -from metagpt.context import CONTEXT, Context +from metagpt.context import Context class Example(BaseModel): @@ -73,14 +73,15 @@ class SkillsDeclaration(BaseModel): skill_data = yaml.safe_load(data) return SkillsDeclaration(**skill_data) - def get_skill_list(self, entity_name: str = "Assistant", context: Context = CONTEXT) -> Dict: + def get_skill_list(self, entity_name: str = "Assistant", context: Context = None) -> Dict: """Return the skill name based on the skill description.""" entity = self.entities.get(entity_name) if not entity: return {} # List of skills that the agent chooses to activate. - agent_skills = context.kwargs.agent_skills + ctx = context or Context() + agent_skills = ctx.kwargs.agent_skills if not agent_skills: return {} diff --git a/metagpt/llm.py b/metagpt/llm.py index 30ced25d2..a3fc5613a 100644 --- a/metagpt/llm.py +++ b/metagpt/llm.py @@ -8,12 +8,13 @@ from typing import Optional from metagpt.configs.llm_config import LLMConfig -from metagpt.context import CONTEXT +from metagpt.context import Context from metagpt.provider.base_llm import BaseLLM -def LLM(llm_config: Optional[LLMConfig] = None) -> BaseLLM: +def LLM(llm_config: Optional[LLMConfig] = None, context: Context = None) -> BaseLLM: """get the default llm provider if name is None""" + ctx = context or Context() if llm_config is not None: - CONTEXT.llm_with_cost_manager_from_llm_config(llm_config) - return CONTEXT.llm() + ctx.llm_with_cost_manager_from_llm_config(llm_config) + return ctx.llm() diff --git a/metagpt/roles/assistant.py b/metagpt/roles/assistant.py index 2e9ec9bf7..2774bd9b6 100644 --- a/metagpt/roles/assistant.py +++ b/metagpt/roles/assistant.py @@ -22,7 +22,6 @@ from pydantic import Field from metagpt.actions.skill_action import ArgumentsParingAction, SkillAction from metagpt.actions.talk_action import TalkAction -from metagpt.context import CONTEXT from metagpt.learn.skill_loader import SkillsDeclaration from metagpt.logs import logger from metagpt.memory.brain_memory import BrainMemory @@ -48,7 +47,7 @@ class Assistant(Role): def __init__(self, **kwargs): super().__init__(**kwargs) - language = kwargs.get("language") or self.context.kwargs.language or CONTEXT.kwargs.language + language = kwargs.get("language") or self.context.kwargs.language self.constraints = self.constraints.format(language=language) async def think(self) -> bool: diff --git a/metagpt/startup.py b/metagpt/startup.py index 771cde80c..000b3c5d4 100644 --- a/metagpt/startup.py +++ b/metagpt/startup.py @@ -8,6 +8,7 @@ import typer from metagpt.config2 import config from metagpt.const import CONFIG_ROOT, METAGPT_ROOT +from metagpt.context import Context app = typer.Typer(add_completion=False, pretty_exceptions_show_locals=False) @@ -37,9 +38,10 @@ def generate_repo( from metagpt.team import Team config.update_via_cli(project_path, project_name, inc, reqa_file, max_auto_summarize_code) + ctx = Context(config=config) if not recover_path: - company = Team() + company = Team(context=ctx) company.hire( [ ProductManager(), @@ -58,7 +60,7 @@ def generate_repo( if not stg_path.exists() or not str(stg_path).endswith("team"): raise FileNotFoundError(f"{recover_path} not exists or not endswith `team`") - company = Team.deserialize(stg_path=stg_path) + company = Team.deserialize(stg_path=stg_path, context=ctx) idea = company.idea company.invest(investment) diff --git a/metagpt/team.py b/metagpt/team.py index aec72970b..35f987b57 100644 --- a/metagpt/team.py +++ b/metagpt/team.py @@ -10,12 +10,13 @@ import warnings from pathlib import Path -from typing import Any +from typing import Any, Optional from pydantic import BaseModel, ConfigDict, Field from metagpt.actions import UserRequirement from metagpt.const import MESSAGE_ROUTE_TO_ALL, SERDESER_PATH +from metagpt.context import Context from metagpt.environment import Environment from metagpt.logs import logger from metagpt.roles import Role @@ -36,12 +37,17 @@ class Team(BaseModel): model_config = ConfigDict(arbitrary_types_allowed=True) - env: Environment = Field(default_factory=Environment) + env: Optional[Environment] = None investment: float = Field(default=10.0) idea: str = Field(default="") - def __init__(self, **data: Any): + def __init__(self, context: Context = None, **data: Any): super(Team, self).__init__(**data) + ctx = context or Context() + if not self.env: + self.env = Environment(context=ctx) + else: + self.env.context = ctx # The `env` object is allocated by deserialization if "roles" in data: self.hire(data["roles"]) if "env_desc" in data: @@ -54,7 +60,7 @@ class Team(BaseModel): write_json_file(team_info_path, self.model_dump()) @classmethod - def deserialize(cls, stg_path: Path) -> "Team": + def deserialize(cls, stg_path: Path, context: Context = None) -> "Team": """stg_path = ./storage/team""" # recover team_info team_info_path = stg_path.joinpath("team.json") @@ -64,7 +70,8 @@ class Team(BaseModel): ) team_info: dict = read_json_file(team_info_path) - team = Team(**team_info) + ctx = context or Context() + team = Team(**team_info, context=ctx) return team def hire(self, roles: list[Role]): diff --git a/tests/data/audio/hello.mp3 b/tests/data/audio/hello.mp3 new file mode 100644 index 0000000000000000000000000000000000000000..7b3aab0a439e9eb4e6c32c64964aa0a68674771d GIT binary patch literal 31391 zcmeFYcT^Kw-0wZ9Bq2b60FzKmLcoB4lY$gc6G{vakR}2mC$tFI5G>e}0HFn?8xS>A z0UHP^*iJ%6f+C<|Js`rd9~3>-bI$9z&w8HquJ!(N?|SdOe?8wnW|!HsX4YOapS^!& ze7)I7;9psXg@*e4EDZnvA&lK6c6N2*FrAnX^gqu2$KjvvQ|kX$^FOM-TgBUc)_z_G zSOUP_G{Df%kVGQUe#+k7p8ZqHmoNAJDWOof_NO*%*dP{*xBk@b-MbI{RBmo=@lRD& zR@VMhOG`_8d;8U&>h0~-{?y&Occ1*!+}zx|pZfgy^FROm^Y5L1K4kpQhjelNxAC8% zBToHCb#L0?)p}q4v-`ix|DJ*Wo`L@(Gw_oMy#VavR>Dn#O**T0dsiV~8fy@@$uYQy7>uTg*Po=X=uI(FVXpTms|pMc`D(Tw3Mo-O5|r zlB=h4gze|mo4lmIpSn{0YTw}UI0*-6GMqU&f+ybLn%1(9p0i!JKHKRa-DF;1zee`@ zJpC3Mx_XLK26pB*mUC?xu0)EpDtDHmrc%m8r>F1)SA)fgy2aN5cjlkrpS4t6w!G97 z#gB>>v=|#AFAkhB3lnU6Homl)1T=i#T#UVORCMRmm;e@od$<)QxfZhKH0H(Cq1>l- z>zz_4z>=(Mu_}V<$}cL*v!uWcZXZ8Gh4e?#-DXL#@aeVhh6-zkw^Kh;bDG+e{ueil za4E&AeFn0czQM1&K)WsX?wqY3h8Pxu^>N#iFFe;7qgEZ&6toRS2uvF7uk5^mp*zAK zL=Jvs3T56C835XdcDeb=iz3$L2RyU5Hyib`E@x6<0bSI`z9MgHgwBEIr4pmGgY~9y zn=@m?{o7|x@dssAUq?;WRBnEky3i#JV511UlBw}VE2r-@ zok4e}=8}@}8R^h`FNz({v-8p=N%&i^`E(Hqtd(>>mnmXr!df2-I72yA;KG!Ne5$4@ zAwfcZiB?O(T(gSpW1DR`qyl$~H{OhUM9+^O*gQNzMHmn4V7VRj4EjM$K4qVX1}SJt zNJJw~m6TRPs^z!w&W3F9lloLSH-`CbIy(7~LC9qK;x*Qy$terR%5rz{XyyD=o!EQQ zH3P|TEs0lhhtk|Fw`Bq7Y-WQnFNE@DqVpLQtt^`A@hB0*0(vN6CWe6|P-^y121~&8 z#cFwf#P-i$uFL5e^E-7mrM=F66bIu1cPK%qFSeK?&%GvqYmFA*d(W>I%N8v?CS1#j z-ooR}Buy68aPy&|DRjrH&U;J%4|N}neCH*N_9kcQlKs`0)Xv%!bC zDaoJ4$V==^9e;AH=cU@g@(~4dtCtdLdC3QbM8h{|QXy7)T`0YvhRKdxvRoDJ@=bS> zJtONg^~bInW>HXv%_39U)v~&WC88v#;p~b#iCJoDhmJ8aJYh!E_R4-si_7cr2dq#{ z1H=PzAvd1MimzeQebxdJe&q^go~XxDC3>SCWN+=bMY|6DXq~?06>X@HIa?)xVYE0( zO=&hlKJHkg2Zl2#$gwU&9D5np5p8=?Ep_h-E;Grv95e`B!VHE^KYiuZR?Mhk8k*ddws{&>(L93EKM>?8&Z z?7X~yrctF;1coERD>Y+G^C0JRT-fuA98~}VA;93mCV$1XF_VWW2}kT&p%xdCi46}9 zN2^;ohK-Ck8HH%6pp3s^?YLj#NID7+y0Ft~tg}NRnHo*#&dM%EJKk(_Y$b4^XU20) z)6DIJRKrx6s!^;*Eu!2R53on&$>gB1ttvUPlmZu+67%g}5Q9BWYPh(SFt{YBjdl~1 zd3oyg{-VK{WBy)W`}+~y4K=d60o3A2z{o-uS=Ly(dT)^F)#C#K3tkZI;q{2x-pA!` zWy||sgwjvO4X$_C^;>1c{2ouE3Co81cq$$+0dfiM<(F-%x}*RJ9F_NuP7}#Z9QPn-qtAt&x5-%j z6?n%6j;qs-n^pUp`d#z{>O*?H>PJA>B&KwckzP^RGSOx}l>;Z-rLsjH+Pw`PFp5aNYtLvAFEPXBZE+| zv(qTA%`c{{Tzd73)uCD05-uF3LF6>FmDlm=G+UCK6@g4H0@ojCdug5Q(?-(-&coa90d)K773lOTK_TS!@iK6JQ4w z*s>6(V?O}>qi-3PH|vF#OMqsS0aF4e!eMavsrBYziTBsd{*^Yl&r(QCLZTs0-4}6Q z1-g-E?2R5?%o!)1T9c^f+i}(J)4JXFA0&}eFvE=t%bMT2H}QF`i=ob9XZ?oFZ%?25 ztBFJjD_Hk#yk*Q#YmeH--06eeLdC^gdpMqrk<3^jQKLfHKjSnP#Zby32@IHoQoXH1Fz} zEDF-76^URTgMsHzl2U;2OkV=%fr0@re8T>2f$<$%UeL2lTPx1LP>2Us%JsjP{3X#D z^_iYoF$%;zhf~v$rbK;SD0MBu6kZnMa9b*LLlVqW2fKTbXmIOgegSb((>;)>6(D=P zU|#(z>7qEKM%tb1V*(n9DQXTDQMj5bOhR67eol{y%e1bk(bQ4vRARD~cT$2xjPw?9 z=Zu^#DUjFthP=6Tls~?-e!&zbAUwxB)LsZQ?G-+{6syPc&PE}#VZm<%Ha#&d-U8>X zDvJ&*St&CUh{XWK#SqI=hN@`9pBLfa*z{Z;Y7(Qgb(GzAz9Nd9xnWa+vpQNf-z$W6J2 zrcBGe`pz|#-u3J5=O4FhVZJ#R=}GE~EDMZz%{*29a;qM1_EgiY-~NVy&n_tdAc@YB z>#N@-3*5!58d`Gw3fk~J(PKk7LQUr4J$Ycfkzu_??1w~BLPSVJBTPaU;Sqc|9{k5R2wmqLm~Q2i1n#9{(QMa_F(rfpsE3XoBza<4chV>N1XEDVhXkbEKrZHMUhC~9;m zXmpS-y=hh-^3RU7fj)bxVSux>-fO&)~jQUo@2oyg7$U1bLX8GMZPaJ%#NpF;};?7{6pIPC)oZ&#>th4 zB=Srmz9%mxH0_aF{GC$0+MF#e$+;IFj$IK}SgM8VD%Z5_{Bo8hRBe1lF1a#y<*4m< zXa28F^0OxV4;SXzPTB5P%!!C-9=?AOOzxB;HjqqB2{W*O1Y&0T zgG2g&w5VNY|3Qpq`y5-p+3U~5))(_XdbnQb%pfZX!WgURGny_$X$APq5EKNcsQ_x! zq5bi>UatBXET{WlC~TFiGF@W5=N(`WrJ;NMItq(Z1*u^L{DyG|1K~}DP$$NMK+8Fh{D zKzWa*3h*F7N9%120rlNfkYs)^T>L z*8&P2;wr#-SaO$hGl;jy+fVHy=XQ>KTY*vE9xr5S_dcmF|vYn=y3BNBSZDtjGPhg zReHzVgw`esH$7y0b8b9|@N@K6o#Gk+rSRnEtkIO`G?a`iYQI{n+EmAFK4oz0&^~ zHe2%Kug6{fYYzRf+iLwEf4A&j`gDHo)7jbGZm%FZcMU}5N}*M-Zb&_}!nN6_3wov= zhGHb)kh@v~Jp@id4DJ#H?4Sq_phcm;W0A>EryZc|<8(e6RZ+0$pVs~UUh0$pxu#e` z=nit|R=ew?rEJQbiDV`#&gHhquS~7}UL!6sm;d58cgN!t+`2!&p+Ahj5yzVQHoe-A#4kYE7SkHH z!1-ZE#7p7*1_+7-!s*DewBDH9&ZICLo~VbygOe|y{XL|z_uDRo;=nF& zva;B;67tPPfq>r{RNo>g2tf|15j`jcC{pYA;q!5thN)88VlO5fvMw#aEkn(~?IG?o z4WWaAMma6E7Bn2lZ{+#E1xr zTxW{bh?@-Mlc%3FZua)*U&)PVG+>ybajkl`#H8c>R)cy|T)YA%!f4S!1U#yBu@*j$ z!|Ph${it3hgicT-3K5XOR|HT?gh@iU^lqwmkEpd9@+iVEJ$68YYoH%kh+E7GHeU9k z!MS`XyH%GxX5umY$^Cga3)Gly(mvwo>F_;!G)0WoNDp`93HK!qi=*{12Lpvc0mY<3 z_eedzEw|6Tzjd+VNbgQP^m>I0b7V6ZU{Y*4NX`)k7wF=9q$IB9 z9qBgItefI|y-`l8n$J~_3KVT93reP~E{={zfxZulYL!7viC{z$!AM-&-&m69Wn1P+ zDRIG-5~3mGxZA39O<(fL=yR^WHbv82 zJUxT91Q0S$?f^v|Kq?k z)5aAZm_p2g$dF0VK83pBZ!d*MK6J}3}PB4NG34PI#Xze z6@&?9MQ1~g&7hB-OpFG8ZLX+ZTz9y!Idt)ZZNINB=wEtjE>2+^Y-{^N+DK?O4__oNLyBu0#NQ6{J6Qd{8SoipVhdGcz8RlyBG-k9V+`ilS^ZSN)&$8d zD&C;C!qgON1QBXqkfLDt!E{cvr!pDvL`VM$bgTKp%l7hnGO%dW?U?Bll8k z%-?qY>jF9X<-_Rtd*Sb|Zu3~#8xpyz4`rHua*fmVq#^}*DikYO33&iIklb@CGmy~j zvWwpXt$`heEY!`A2~Z313vV!gm)(R`NT`sZ+6O{%#SlVT%iNp?K^}NxAr-H{VzfHg zZWUI-%PZ*KlNB#JR^w(av?2*9WF5&kinjpb+e!J>NFfAqkR}1%UIBl!9Y*UGo(v1{ zmfXf24Hz%M&(nmaQ32%4TTzt4uP*qgFd=Awr!P|6wi(&)Gh&sY&0G`U6n~2mGji2K z@!V5;0Fw`&>bxR)ahSjd;V4OVJdV@Ul+|3$h|gAb$2og*G?x??Lql5ojB!=SC6JcO zlc9@RSWrOX410602kWfxbh{y@<;)O+wMFO^+biZ2Z&UMn`3hH4qIdv(-gWuUN~ro5wj_wM~6EqnmzWGI{1uefou*!@;W=r9(?s1lVcvLsu1iX-qUy0pQB#DZ_-JB^WA>_uod^3GC z2{va8>$fGNx7*8OLixP|R^;nu|hG6Su8)LiE35Zz{3f>3@=<5{b^Xa8A5kbjUi5Pj3#C$Tx z9uqWu5p<{@B(Z!~mmp~vWc=&Oa$w)wg*}A=Q9<@3xm}#}vJ%3=k@hUNE%de>-I7?N zvR~J=bV=pt4Vx}*gs-mDB;1yvNE)Q4_Kvj!zBbCerdCs^pNN0Zc)34;TTVZ(vq;>( z<3)7q4%ReEhs1g&6>`-YwB7ok!aTe_;E?zu{24vzUnpDzuEX^fE&UuYP@$#-?rB4p@7aJy5`bexD#cB0fg-U*ybAM-3ex2FDkDI#)5_Q@*%GfT{I#ize33ZTv$d3mXO>8U zUl0|!KsQWiVd&iF4fD3OV#Ur$5@yIa3=MBUUJ))crb$#V!l2Q%$_I9b@g$6=%G4^H z0;H;-*1elKQ-Y{-TziGhBxGXZ4`eSafgkt2EIoB0oIZVio#tyuL+uaO(aqW2N3Yx( zmeTVsuS;Q5Du&ZeRSdQ_oP~G!&#{U_kF;PKl6z5@U-Za2wYwWSEZP)}q@2_*6>DC9 zuQ*U!dERNjf_3%AXS`$7_v&BEO_uF$85?_gHs4f0AuLlm6lgLt=vAqZTQ!plags&NmlTBkRG$h6yma z+;|-9xdEhLn$jWVEh`9+0^V7fqDmXv5w~OdaY$Lt@+K@F>C2xnjm}b^L#VgM_MRZI zW?$mY5a{qCtEb}QmG4s|(|L#>a$_}#n@7kd=w|G(qJ4snbXW+xzfib^_%Vki8|)6aJ*cUZK6zc_z8zDY{AKqBmv8bz_w=$Fk0HNvxqc5h z+-~V%m(LC1bSNk7O}yTN)16P#CVgJBzZHz#TJu3WQ7b?DPxsNa=MFf&CdHJ$JFB-p z3v{ql0ydR6ki@SrH%FhKJ6DATU|7HLnw0laXH!AK=4fU_EJx*Q+rp&)<(!v(ZLMv$o_A_?QoRHvRV_jW_F~QVYTGB|= z0||NZylk(N^4gIHH!}wjYm1MfN`Qe1ViZio8n(_uEhByckulO9P5wk4UDr*D>icX$p%o9h!br zF$&oU%gV1=jZw1Y%HlfvFoJ|_Wzp(-y|E}GviJrrR&E}5O5~Q~eX9_Vh!D7I4;rjS zMZES8lmtqJy3{QMcqqx$wQ4+I{628>sNs$0E7m&vzO=o3HFc!;W?kXc%gD~qw0kr9 z?n-sXOig#aZd%ciP56UH<1P^iOGDaq3&;|UK@eZ*fwDU*q-lA+USi;$l|6w z)rfL8wI|M?aAg2NPqE%sKhOfz3m}g9RHrbTr4RV+9+gu-;s?9q*(!o2OVV$_71?+~ zhydbGO6%#d`g#_xckwEP#p(yZY&%MI@DK&2@(G5f2%MVpUdrTDknVd;4UAAGoH~$b z+~Fz}Hi&uze5uOq+|CHwVd2NMC&0PTMwJ{x#_D2B&=|rogT57IjvEjkvLGiJ>7IF><@xcDK*TMfdp^V7>YFfGX^9SN&BbgoJ{NuVbM1Aj z)Vn_dlJ6|ey!0(FyryC}SsX!&!^^Np<`ry{UYF^^x-)-FUf(LW>~a4W3e&Lba$O7k zg)O(`y!Z!Cle=`rkQedJH#fp|-Wj`TqwKGrAnv_>de`owHaiUqjWc?UsG*v4?-j;AgcB#*bGM>S+(_}xPCv@q$iTTO|0v92;8ZZlL_dcP5emg(A>rdg4o z#A~q1GTQ}xlIs;Qz|7Qb41p;hN5?U_L|YtCO`Uq-AJK={>M8d;qgc;fmkz!dGz<&! zPb)JE!#F&yX7&{3kN5wWEJVfXIJ1rg?hdKuci9ytB!-xeOX?ckajOcEXl&$R-weYo z%kj`C1?U7Wcv%S;xmvkVP=JH|mv`kG03Q67gqtI;sHfbiblk8!3obv5>8|v|D{4q3 zwir0)XSnRjmeX-DBe6Tr*KqmB;G_~^lLg)xC+!}sek=8aY9O+5``T=Ms*Q zS8kcfi!)6liJr*33EQ8i3gw=C&rf&lrbUp9ZbaV-t$i8VQ}{he_2%4qgeTE7^;f&k z`Zu3^0WnzX>dLhPA9j^S-@N|&h2Xfg2ZK?XcM0dfU9_~9-|uhow;{j%d)3DsMH8?! zQSm!BZ(MV5!0xKEkQd;?{x{$4-yF67H8HpAdjS$U3{jI!iBWFLY3_QkQ)!lED9LbC z3i4}mC|60z`b;@w4C+fF<*i}LQ@S_M$9idB^X<-3#F>Ik?CD@t%P)7a7#ubo|HgXzub$-Oah#ga9o z0nM14j}U5iRWRG+7@Sj-F~tFr0C~0gNZLi~VII-f z`}%2o-I6Gb@Wmtgp2t+)9MS!<@)EkN4Ko7|@@5R_Le!MS%OiQ0(9IInP06sTrX?D} zXZS5yaTxjOiVY*ymgtrP_eLgxT(%Lh$p*Q(=7~rn%~X)y+rd2yVG^TL6E&xZ@wb&s zCvGU|Zql}P+_GfZqgBTex9POr|y+wn*YZI+Rn~MbG6v~Pn zjqNt8%NA2^hS*$bVed*Bn7S0T0GX<9{L&wI-%7AzB5#@V;J;9KocZHV1ByXFU4Zd+7ZmCKqu z1!b2Kk4Q67n!)l4mROy@$oRMwQOmQ%W{XJ34zU`WmFX-* z2O^6UO&09b-kzR1JJ$flA{UDHz3KKWHhyX&6r7>q#iNZBT6^Xm^-ccv6j+^<{{mS# zK@fNIWVv#46!@6tpi45UE3h|(HSISN0h!Hd4>=|RIxH7bT>S99A&F5B6xzER9+nMR zC?D)J_5pO<*H2x2yYmY-V?L8ti_3z^GL2TuSZyJM8`>0zVH*rRRX?}6&9-*Wb<@2o zyj8`fskh?_&VC9-p)dVf6dmztR2rLVtRz=8gSE;jFt*&s9m}jBK_me=RW02B3--@c zFS{YKci50H>%NiG!!laDbE$s)ZKLiU92m!yri8n6tnA+A zaYHvH%WW<7Nynm%y=C6Uks8|0*f}g3V{~N^A z(?MWb4?!^G3@{Z8J!m(+oH~Yv?~#nEluPh1#Z5u=c3H4Qpa=KFcPF?h^j9k;N46c-F}3jmu(8(LP82#G7_H-( zEmhzx+yuzZ)ZbNc<5Y}uq4iZVr*8q>=3H%T8xQ@yKu=pn%rteyn{mo|!ffeHh!Wyq zyenSh@VLMgXl=5Ad+n%Q+ltGh)G<9pqw(<6Uil91O#f%ldhJ;80a{RRDk)u zlk`u?D?zSx?US4MHzky?!|<`&Ju3d(56FnYK{W%@yp3?WpiY&-K#*wP|apw44m-O$}Qa;c~R zo%g^{twvi8HrHisHW7AO>_x+2M@9m`OyuOx!F1Ejs zTs+i3QS1Dr>I6s&?o#=SdzYZD4JMu;&)F6sqFuD)4qLvqkZ{j=jaDH;ES}RbVd# zfsq7QFrdyYrrI4k5SP+j^Fd8i%Q6WBZGSpjL+p-Aw{z9rlDUO-K1=K&nk#TP=ZGEJ z+)TPmGtdJkk9Z@o=qd@{-Hs!9oDd>h3P`*pfT$DW2}d|={Gvi$fAldt|J-`4_Ygja zc(dqHO%~@U_U12G@;zN(xY~iyV;Cpmz71|-y7c9%oRg_hzBQQ{UE#2qpj~l8VY0B6 zP&AyhW4iv85gd$(PT&;~2s$)!w$E*@^$v&kJC7w?-!*B}5D{GSj4P{Cq;!u1t#Ap# z+q@Kw%&T{>htzsdkJL_a1xjKgErRcCRm3d-%4Ly{FKzN8)ok1hE4AN!j2v1UJTlYj zse+dZB+-=xID(*vqOPF))uW%^em3N_J$=7pt?L+U}u)>mHy?w(%GIC;IogcgK6=fevy&EB}z%b}rSy0B`ZVBKkd z-mZV_|2H$?Kd}N?byKX+_3BFa5A~oy?vQK9m!97-Ds=+zTpf(;?crLB?jNEGx|bqy zF(?6DKa^0fFvfxOOjL(qJL(5E9ueWxLn_u_L$-8?T? zK0d>e9lW{EhpwxeiNBx+adQW{Jt@3gFS*mtIIz8l8hk{~7uJ%PRa-w`%L4hK3}*~c z49_92;5~A*U^FjSm&R0XiqLf!$W>^)2@YWxcutp7K5y0FvvIQ)Q4{TH2Fa6a@AmM0 z?$)5kk2*|PvmoFqL7A5AFajC@NsW33o$j7mL%%dq=x?nL%aYikqeiMot$uhxFL73wbWHx&B@SAqx1OTrVJb)w<8BtA2v<1 zvnayB+(8)7IH&IvYmGKVAL&;(#P;+%S)oNtM7%eOp+W3J8EfoBjqw{9j?GdI5G%}h zGt>j~e+FaYh|DP}r8Q_^(OUHrX8i=QU`cE>!H6>C{atjfY}3o*B?KA-v8LAsFDrF<%aikii|ck2ANysE5(!h{VA($%ZFD6`RjhRUnUnRNenu%)M}as> zEa)O>2ZIGaz(pDw7{uMC^rKcPH;D$6qcUgZY9Jr9fiaYEHQT`3neV_!-k=gKkt;Fk zFr}hq@rh8;zN@}HT=60kB~e5Hl^*=qb~o98JdOwo?c;*Fa&+XT28`4fg--Z00`@oj z?JuRGyuzJu*J_I%8wBxafW0MaT=>Tn+pk1h5znl*SZHa>ioLrGraVXjKdmkVw?AK= znT>><+Jrq^(TX=JNQpS|CZ95YHLdX4`xhq1gZm%;IF7w_VQmw9&FK}nor8+Whgp>OBJq^>lRq24RNZCk*S&CMWun<@`1SFga%g9B$?~y33dtbxCIxv?9`=yY zG4Y;ja!XUS$#LVRF3D5*#_uf;!t*w6v)KF^pWgqbJ?yXAljRfryMz9U9Dcul_8b4#HSjhnx*}Kg90QX&r!SEYid7IddfD z?S{-E_R6(0ikf|p`QmgSey^)?&gj-a^zb@Qm$L^JZ23i;)ubNohXB@y$AW>> z1Nwwqz(Y0g#dxO}R#FDsk(2b|=IV6gB4OM%h?+5MZA`Sli`>{rleZD^UQE1EfXl0D z%Vf48IP|=RjO^|S4LX9~FLJ3)7#?Z&sTL|9I~%GrfnEm2_*-0rIX)xE0gDeY-PG{+ zFG${Ql>pv+CA^Y+rlBHe;P)BBW+XM%s%+3Rc4Md#8GIfy3Ta@1(VMu+J72b1FKcOFm)b2n3i85ABVun3kdAG-YPIVBWnTWQR-|yp@zD zE^XYcC#c!g;ReRbQcdj0w|iC!=_@a1wbp%a#&kZixTr~LP!C~i(b?5E>N!pCgqE#sL(>m_8!gd4zo`S=mMeNDNZ{bI}^X+7H4mPAR4FZHcJi;CL{_;WV?a}F)o|9RPejw?EQS7d~^XB4$l}gd$9X4+DD!Iv$%4Uy^MX{AI{z{X| z=xd_+Z#or4e7)F)=lgZ{wx+(We`8l)zLlqiIp2St-5pj8z8Z`0HisJtS$B?3;`MaK z9EfBaLbEs%~Yz7IzFs0ab^yFrtK~SU*2G2p*MpmK^ zcxfXWs}0+A*BlZ#_UFAIW3RaCoxpp7z)giAkzAJKDJdd)Lbax>HvY-U+WY&e4U2Hi zh||W7We;nz?!9YRqiakM{3Zz19TSzug$Q zY4BX&!*2VzUV47d+*HU2EoVFi=I{RcWJ)q^;?CD7}hXcmQgl_?)taQ4<7WCr&F%wcUCYjy;~ zTGB^=C`eDJ1>OVEfndmA^6h8P8p*K>8k`vqzN~#gP16FwoV*}@8Y@=RV_h5%HBH=( zD_!d_a_tvqlZ>7h?JMY75s^R;9$p7M;NI~qwZaQce|O&0I_hb>4WoFM%b_IDD6zpW zk0ANxWU%v0Tx#>4(DFTYv!esYFJnjcR=nP~w`7B*bjMi3k&FG$j)^4DQBjcFrjUpX zf~YCsy4_gwFAX2%Z)Sz-LU->puP3Tbzw*U5b_8(pWx8LmAk{P;&V5F`eG@J9MC*e_ z?o!U^^WyDhJEJu_E|vEj#w?3kK)T{$uMvShEh8l4$Yfek=UNW@TzK3p$d-pCSA)%K z-p6K5CFo2(+YJ)r!nXQ+P&Ii`$ILM}+w=!4h&GO3!6LIWj$7F6x^n*EzEy8vlAhq< zx8f5g&^gJfLE67i7|#6Whp;eM2sJQP$4mcCiLGz+@%JrT!@#BdCYt;@1fM&i%-I zkRQc7F8Kqhm)vK1AwNP>yl%)rQUN*2j38^^0b~qVKykUZ9TCpe_-G!-9va-(!>|Oj z@kW5cA{t8m($ka)B7`x4&PaTeVxx+B!q=970;l+K-9wZ&-2!Xh`-*vjQCDR#OBCEz zG}iC40vr`3+{w!?RY{oTnT=-p%#QC=_?U~cUCyJ(uJjY{$JXAp{2nf*S&76yOG<=80;Ohb1cD5z1U0ZbKi;nRhC2hRk) zjHtumiUI8Sva6OGCN@NPTIxC#}()mKGgUwog``fJjOs*d9em zvGPMsWo9$XLW~POb%(0lTFk7ErzuPD8u^tURmEwB=aQC#A54SQk&i*otmMOEJ>)8B zv3yVnI_7Hf=AD&r=}Q^tag1=qph!q|YddnJFuBx0$sG-P4p=qoHKgA3!Y3l<$x)Fd zNN}2Lp_LrhM>bJ#BEO+BGgAH+cNBP&cRQW!J^zviVo=5B5>OgfsB5lo|D&LS(I7w z%S}D+n|?gMbaB1WP2Z*)mK8BoZ8vXhZUtZ}F-8r}5i(ZTQVrKNI(%6&ho55h8xsDK zqm;lLE=gaUmRrU*5W#G9l`;G^lOX_wtqmGKIkHQm7J^eWg>7w!!hYTwPY6mDW(H5- zo#*)qxM~HSI_0O;6XArYR{LIwEJ{B1ce-y4S8ejD$B1+cN5L?LbPBcagDVNePn?d8 zwOu(zCOjl44JSt-=H$KhfErHE^zGY{As?-_>3xf^Z1ZpaflT6HZlOqkKVwaG4raR} z0GKYq+3nLqDkiLQU;65b8 sQz}F%2=kwIxcTl#HpI52a%>v;7Pk!6#@5c9(5U& zo+}se<@AS_`IA@jsvBG1d>h)phXF-MK1>OeVTwCY7!&8jrWYWfx}=BD+m2k!_%?kY z@#b-3;tr6x(br=21zImfs=McC1SgS6gq^5sRoSQ5hkQf{t+!lqHfN2ZQqw(8-@P+F z1s0P~6qD6UF!MXS-%m}~R<$9H1j43+sn;DO+?urU4v3Oe*uojT7zFbHws$t9znXP- zyz}Z{?N;rjeGc*8dX|2x*tTY}zB%fTO&ca!W~1%WB(v>n&TKU{?w;7YB7B4Y)-gs^0tPTbioPT8BHc`3+%wQ4?vpG-bP$M$eRm}0XNeTOEt;bnRZ5Ex zyj+g`bWuJgICC$-H{atv_OQ2Aq{N6O&1`XO&Z>u#vSK)u#V;^>Bx>M}q}JF_wja(2 zL)9UZbYvpBI!i|t75DRhOHOHQke9_qgs#$#bmfRL1ABNec_F2UnTT*@K++y0&2RS1 zHtRfwXmq2Ic4O@+195u)9MN;PFEv=)!pjRRL#u~0^w@hA^m@(uG|3enSEFq{bH12B zeO~x6Zs#Alt|K#n2SlzdYD%!&RyYCXI#|MHxWj&0Edx!zH_LjN&D>%9@{dE$yn7NI zmg!_f`-tqb8DJgS)%^)2m;N9^rZ&3b(R%_=aVcp(Br>x|bJGO;u$k+2uB=fwt>B8X z^7ZvWAw8lKYNPBT(C2S>Qn98mtJnbs_LwOJ10^oW%rC{#Vru6+Zhm1u*S;9-mG`T-8pz~dkp2(VFt(j^4hL{p)e$M4@7~t{|LE_ z*3@eLniDZAtAmI3obQAtCi1QyNZkAVQ`)KThU+hO9e-^kJb&lO;a4eE6@%iK9P(nk z|GD8U4^DK4rjw$X;)c`r&uub7FKSvohg935y@r_mW}-!Cofi>}RKY_uIu|WQVPkEb zL{L9~S;^OLdx|#eT!Yq8jyA3(XY)yuu3{b4Wps$rsvj|>*N=Vh*cL1butiQX`i&@V zHNX{&h6TAu0Jwyaa*Qtt{3SkfEtyTHET6AiqVGH)>VqJ=ZdIk-9TjBx@N3qot9=_i z30{tV0HQMZYeiut#2hd^=E{`_XDW+`W`V3=3L#hyk?cfS*ZLn7(tV>%GOY{r(<;r6 zBOKg~@0~twa_ebY!RIH-aIfHAWR9a@83Q7C&S85fuu1NGC(CThDUJVNo@rq5wm7+p z6m)bK?&9PFO3!LL(-=Zj6H3NN5M82ty84GP9YH8;3{I#a=-0e3NH}*z*G8Jht7r~e z@V2qz<_48~%%`N-$HxV@1_yC!79m!<++J1y+q<(WC2&6jw*QvmFU8q$vtIKGav!|O z5*fLe*g4b;TpwRI6B?Ws_xe!Pc;IH-j>nJA;O3WI_sM39>0M8 z{I>H!W^>Z^CkI0&CF>nuB-THTv!~rQt#}<8zsmU5iO4t?nydkQaDLVD>vuOf4f>2;F~^6rAfv_OI58HH;)PXZX=myuFMaf#pcGN!`b8U znY++ZFrGR<%FI&qO=IU{WkofU>i&QxP6V?1=Hu1HB|}+plDs{oB0jHU*)42wz-)vybShh8lFHOtWs0p~PQvC-zXS&jS2*4^8|kEmcr6?*ISsx*7M z(CB~!g|{>PMv3q0=uVSUeg8s1gK)W|W4h)Kwt)$+miKoGI6ZjvFJvF}7^M5MGrQok za|?JjZ*$JE`^jfF9It8^cKx(`I`Qg-*E>FaU*o7eE&go#GVx=^FV|BVTk;=d#o^jy z-NZYk)p5l+?ucTE{i5Q+6o))%6pAE`IWm+(hb2q<5z?a_@Yu{7L_1m97TD45UtVU{ zlBik4L%f`RM2&1rFS$I$-%X~(ZkNdkd*lzX?{j#ziA#=6k|COlIs=3ePHPpOa1@^S zgGqrfbXO?1pIfSPRtqO%=%n616ZzXqSD(`JgL(5|E^3whH@td} z-gBO~tF#qYm77cbf7&_Epr#gn-LF(aNFan_K*S^zLkPH1z<{Vp=nynW5fC+$AWcO; z1l+n4N(dN`E?^5%1Vp8&*tQ`QDM3+Cz)F*?*syG0`+M1U?!0r(y`Szm_uQE~bHA;% zX4d+!X4Y^0%kw<_N*P*OW`{gLh2lkSq3uwSr2y7My>5$vVro8kRP9r&>SI?bFzni` zkmkUsei(H5@r6F&(PJ`BfLoic(5ToyZ8ac)F(ka=9+`pjkUo=bjH;-LfzI*`>iLWwV^=8qXmc|Ra zopTviT&Q4V*EVnuE`Z+0dkNYqwHQUW=D{<-2whv%xaNVg4V|qr2k(v?V)$o#0>R8d zMndi(kOQHC#u=kvti+MA7}@iz9g7~Bm}*$^RbgnYb}gyBUCxg6YQYM+181VKn7sG! zsky!>uw!T$b4QL^z+KBAn3RG=PD-+XsC=a5j3u}bNeP_|rK!i+06jJIzyuDTE^ezP z-u9$-t5X*xyRs$JulzG;)ZY3twHrKtU*-F4uJYokpm+WLAg9C8Fes1E&ji`gF)}o$ z`$bpFcS_)A>tOCDj!SHWaWC}_Q!uA(I#4}NiU5N=9LaS=Mnor2YPDz`|8Uz{rEJx3 zeY#p^WcAw)x9|=(Kn9c=+2+d7Jfm~C0w%AEuZjEI-mFHG9;G-FnPVhI9zh!7T1t``cIC&)FOH8KZ(w(hg+ z*7Lm9@zUV-#_22nUC=lPb4by`>wNUpAaiT5Z)1rECW3CKzd3QxBKAXr`g~r*$8y)z zTUyiWPCQN6)3xchuo&;@#+Olo;g`GC+zWW()O7al;D>KoP69w6bK5mz8-}!wN~s7G zZ8V!FXBqApN$n#+q(XYgp9J$_;RD`$qu=ew?Re=b{Xax!{E$fKXOVH5hQFIUe4c7~ z%L*S(AOoD=nb8{ECk%1f(==@>`(hfrpCxp=N zVeXYgoGwsDZUDPqkC?%?vSh5T*;(t#0=ATu7Ttr)TnVTtn#@!arGlh8OEytL1n-3d z7@@}1df@6eLDZafOZddGPxJn|@3NBGyZDt>;{wTDXB9*cr{&yR$mB^pWM9DA4SKY54qdv2X;q-TLdx6z1 zf*9JHCNLqoQ2CZ@?6BIQh{|$=1f@n(tZW0rC1u$muhQbPR?*syXc09`JjR^kG+W?E z8qF5rGb9kg38d&iG6rB|*!l9!@R2;Da6+HuK3y>wqFQAgR>j#3TO8gJv5o&EyyRi4g%D0bOr75qTG~O)R8CMqnF6dX<7G=bhotTc3Ta_^+5LfQKbJKp~e!55)w5EnE;>3vC9Av)_T{VmSywOBlUf zS)c<)gU-$O0{NWtU=u9Q{uCk&OaqAEE<`%>g4h~d0mRXPtO!uN6jop==7A|)KQ*+X zOix55WMto@*B+-Q8~VVk2ksw7@RS~RGpl0?Yl91vG1(GO@Csb?2NBxr_y91PK#$g7 z>f6%^w)8@Uk%kAtIXvV2bWpFZA_ipyQ&S8f9Ng5@`vH4H`|_bFsOzJh)xi9Ai_YfQ zR_;?m$9B>2{!`=hBiE12detN%AkDP}pvuz~<=XwC9%7Q?E9{mS_6P0FgbQU=+I7D` z7Yw*%&IK#8b5)#q=)HZrO7{Y}p%95OL>fCyuQVQw`W=U_$W6>{3b0xYBT%!FD?`t{ ziYk~pP5WusYyHTWPvFidApHRy0kFNRUs(B)Yw#V!7IsVy67a`v$MqQdKQKX-v7U=V zFyZZ}&aNdtBIAYcad?*Ja*F)wn|r2NYuo{H{qiy?s%&v8j* z%QM6Lk_<15KKyfyQR-QjSDXDJQvR+K9>Bh)Xc2V&;j5F;bz1Ukfq1*6`>R7>?3jbr z@6+Q~*B#jR&l_*zeh*AJH@N)LTIvmw^r6mPaUSY!$)VNL1#oy*##5M8St5-tX~R^D zlVde2i*;MGDDW6L*^FCkuGN}lX#r6c7kJ7Z7=O)<-x_wZWmQae3+%2WWL;Pu6q1&4 z8a6FSSXPu()0)>eS&<{j<5-EKD(lE73Y9)_n~p((i&yOE#!E{KIvv4`7Ouc#Vee+u zREy0d>NnGcyn)sPv3_Q0aXwL}#p=X`FwJ@%Y=LD~L9mAb)O-i4&)^Uz5VF zl#(N%W@%Y?g78LH6!NEMb;56BrqmYR@WQX_ll$RVX+4AovxRO5*g#20snME z37oxb8wtV-GLjp&%o~{LZV@WxGF(Xc1It|PJm5f%xHO~4j9)#TWofPca7DWaXvJq9 zXWjK5Dr6X~b)P?yBm&JaS8L|MQihzoF5O){$U6HLR9lCxY z<*oSv24{13#wM7%rrzoJ^Z&p`OKVTiJz#I1HD zDrwUv0*~!rT)MI%{%%<58JvPJ!OepOPF)h z79i%~Vplg|(XM*T3lB@ExS&2w;DFRp#Jhp&=srPeaUWJJoljHdSgvQUt;hARkHUQ@hv^x}tF(mm z7^CuCtGt^RtY?Ufx0}qz{h}XZenteA*RjrTIq}e=RLeTZ{J;u88}N?b@sEp^5&-%J z3kQ_ADUzhiQX5e*K@ftiMcz^hsc5bGwj8P%^wEl+YV%>=!^GYBz&Nm2cEIR7f@@PD zv+sE8b{vkZcL7zE!iK4AE|=6p^IxHh7B`~A8f-@+UCRtNI^Y5Es%X#DLToe_Xo}Hc zca0W$3TyqjQz){I$I=ZN#O*?dGY(U@dyUs*;fy;dPkzX!yi=O~=L_CkPRIVvtjw6M zH2aG+937jIp?P#Tq||G-UvFI4*2`}q@YPK-52TdW{NdO)23Jyi50CzX4C=jjCm4($ zvQf5N4)NYmHs9>yD{A`v+QEOd5C8tl{STj^FdX6pUpHZsoNe|La~O9}jste{SB5|0 z8@Fsz!xR&k_GEiZR7)ZTI%UcaZ2PJrEFTJ0sg} z6w2aY)P{^`>|tmBFN5-RopOtCR|&M#w{pYY`h=6af||h=w~f$sUS`mW!M<;;%l7#P z9X8}LR-IEO^q!CCft|390h77rn5{c;D6Rv4q&<+L%A%{(0}#!~Aylj;oEVtvVv(22 z=Ym~QlSPF^AtI0(e+$X3)ot$2Vl~Ryb+xfQNBJhv^b+Mn=;w3BjXqqf182SCA00fj z4R-HobTSL)a-M0cO4(}uE%v~tD}nE9ww2#JR_-g3Uie$kIG*{AsKWXTkDdAIA@Fbpi8cr6mIWGv#a2Utc}+IQGzvf+$?7hg2h48;MT8hQu|n z+@MOtJdWc#&q;NS!ho9d)Z@YRPKB*~h9i#i4n${Vm6%B(y zOB!avdyR`%0gzsaxXZO$D5cs)aDX$j?o9A>%_*qEnszt7?h&cc=}rPiAL24o*FES& z7Obx#L8fN*$Hdj)-IYn|)@zp*Qb6O&5aP?&uxZg3C=`%Cu20`CPGIZntDIf(`9l33 zXFkPW4Gc)LmSQsWxp*vDWTHV=E+*!Evvig6GUDaUAb}SSM0P8RaXp2U;2fCYoS3a} z>F!<@5q)KRj(YNZ@|ss%yRUTWG5L^iL&mo8UwP;0UlI>(D7!K=F@Ae@`us-zt1qu_ zKR0?dd^_g5>crcM^jdl}xo_oeQ`xqvlrvWK7f&`VSUe4?{^MUv_&2@(f8r2yoZrH^xcxOCz{1**HYne>pJdY?H-VJf5PT{d=UiHxcB zjEPJJ0SS&VK@$VJq**?LMJ&-|8FNqqs)We8`FzmNmzO(g75=Q2GTZlt6)}qA`r@%%%O-Ei=+_rUYURP_z^Vux(AM z?P&l&8HAQp$X>S)-|NA!(E8<;L869%hp$$f09+ddID^oDR5XyR0K$qRoN6Vi1q7k1 z7&wKHo-5JRWH5~kaS#k_&5PY6C|y88g`s5Zem14cSt~>vj)q_E%23o~>gk{$bV&6` z8u)Zc}|1nfYc zmcGtE_H6Q0&Sd|3`7)|l+U{3dy87#&$NH@y$NcjL%R@hHdVJ*VlWP?tvFQa?r&bP5 zY1*s9w>a~i68z&+DgQ^%_i zon+Z^M{3ujyI8$>Jps-|ar?KBuO?R>u)N_Ob#AETof*Dgy(HQ`>Fxu$Ou4w=J^3Kt z@6MyvsNUa9tA3QWe&3z<<-p1<46A}Pk@Ysr^eeCLiNOtt=M294ntm?#-Mr1_w%^(0 z)P-uV|HDxDA3OmROqf1B2`bc{##Kb=rNgB7DT^os$_}~gE5*%RX(L_HZ)|C%xudT= z!KB_$kkn`K6@qNEB1%b%*7FFDwe_nCtt=_g4wf*5#=M5uwSGyWC`BY-+;rJ+)-chH zPE-&1lxQF4N9oeF~lqt=Yv)tZ(f6TmRtD zvqxt(MLSdosp^jT*NjOF5Uqzht%vNfNS9gnztq22B&@W`bdq z;`lvRYQdOpbV8VO#;?d&^O>D*aq>g^aJwr#L{G&B@z*0I+9F7LVq#y^#MAJpLlM(Y zqq@WqU}Bg9ac3@0H1ay^#i!VKxv0xNtb%HYmnW@LZ-&d4o5=UBl%H7XoY$C^w{yBe zS(HJcSLZntH5HxpkQ`NBX**xk)vKG2(3OVK3!FVmTy-Gz8_Xy4Bv zT**@2f6;P3f#tJ{1;a-NH6%>L(7iHnZLFxuG9s)3)$FO^?zZ~FJ!L}N;=Ram(?+MwbjA#8qNeal zsOSo;CiBOm?sxHyW2WwteeGIj+6{lSwx%y!vioy#xxtA}*XdXGtFMfWUA7t8wZX65 z-Zn!1Hu;P1pAF5!X|KMjgXb>(3kv_?e&IiSBUs%P2iA*kfiHMd%p9{S=5O7*!KV-= znBuN)2Or!CGGNtUU-n0an@!1vZ_BC}_$)8*G4DQoFQkGQgs=t0P!_n1n+|4Y1%L># z2!sLk;2JIwJk_NS+AoFjV1b>W24|-eBXuMU2RH$D))c4S#bq|WI8wgRuGQm~S5opb z?C%DQ{?|%`fC2A`AdTr>-l}0?b}ZXc2>V(p(}3i3w@AHadks(MT-onGP z4m*tJ$!u8SYXZg^O zM#Y8DvXd0uwVHrk)|HL#jhxXxJYqUf5l5j0GMEO$v_+&R^FRL)Pv6RK@Lft^)r}xi zG1@tKZ4Zc}r*288m!A1FBG!)zIOp|n9!%>v3M@_<$?(Y1HUJLCJpOh%j_qPFVV*|F zf)$&@ObIV??v$EftQ;{$t~R(o?0S5nB_9Bbw_lcgnGBMti->SU#+7(&Iqtd z+{M~yO#bO}A^1yT93}IgsKdTPanDNEe)>Qutq-Hqt@3($~yOIH#PZDa8 zRs$VD&H|x^HJW8+-M*(?=;0a>*hK_mZmzvN=_CC!p~R=G6i1Yi>a3SR*0S4P7y7!Id*Y@-z)e9ea6x&zT#a!L&r8yugeiwj^7eu8=kjd3!9 zgmKKYVA@C*J(LkR?Q9F9->pe&VrGqQvtd~J<)H1l6&|GACM~y0S-Rnpt)*2n^$F6x znMQ&X(g6LnbVzu(>iw9pZe(m_LWhz6X!JJLQ}<6#-T#rc+c7tNrafTp1yZB>!v><} z?r}>E+XWUYODPo%XS2lBknY_M$y|lzKQ$U6G(Jzgcid%<{r%a)rk#dqPaph2gv49P zwi`V^Ies!~bLgj>eVRE#$Cm*#pygJuYF{%gd2}^~U7g^0$c>|yTP=l{=LO0N2Ap{r z8QncZh|&#Av2a2=F@RYO(=)cXDg4^TbUUlqR3>?rdUYfomD*vPzTLZwDr~@)wf34V z7j_Iss{qxMm-eT1JN6rQ9=yIbY5l7J^1i6J>7;|*#=G7be|a|lHUDSC{z}p2`)k+K z{ojECAYiV1A!2HJpgtl57(L=ISF`wodq(viugXM-@gGfMoJ!1AoQNPhtx3X+oH6DAj3K`oHK$fxAmTHbwPWHSB00#@~E!pzIe)?`uErBX6UPn*5sBe zF{u4%OEU0G$>75mVaXy;!EflMUdQ6tK;ehy6(6jx%U@=ghuob~P8A-ojR(=5ZdK{A z4hyPdZyoO)*!egou?=th*#5}v6|6VRdyR?90DH&XbdYXEeZDpFb7$YOLW?Ls7=qLl2h_v(zVqRoiEe9&_v%ei4N{lM zG^tC4O&|43s-*$Q=rf9TM_RCT+$s(0f0iRx7Fc3%^&VP%M5uUVDb9FIIb{U)02CA* zo!Pf7Lu!^aE&MG(2SnLg8uIl~@^zY0@n2Le2w7p2(Pbot-~%>8;n0_o7M zvHdKvtOB`GRu9WQ^F6nZ3df7!m92gHdxP$xh`f(77JxObh}~X|{^T50+p!~xaP)u{fgfK`mcS)ALI`h(BHzJ532)?y18 z8r4sjMuyRshKZZav*G{>PfyPjcAz46&poSmb!A7Exkl?o5Q@;m4?wIC_#O_e+m1Mqaf6-3lWX|@n8;LIbm?QV%P3_mp#we{4?|1q%`W!H?aSV z^?_w}s|$=o7=U50PxMlqyN=a#BPKk%5{%&W&@s+dOkD03MpF)qKBnq7&|)dq>9TJf zTcfcA^VkM8YU`Fw4&=Uu&9iI(3JX6%sojL0Chn!!i9CuOTfFOrSzm4 zF9VQU^xVus9~_A=M}R7vN1A-ed2%nZ2U1=sKZT`c@G;HmiIu}Xol{-dctQVhJV-&) znsKHp0t5nRt8yuxaeyb0sH^s-XDGRfk+5%(krJ&BMWT=HpOxk->I@*X`X&lu^w5}$ zjgQvVG(|mlG`5cK65_ny0aM*N1%mn^(j47*U01RJbZcSPbW%M;v)0xiKL5xFoh_s% za87GBIHH9#^P3G{89!?}#jmK5C~phsQLEdejjjzew(1DCaDLYPSdjow?nAhb;_a-l z1Tmc+X{nFedvBVi+io5!Ue1C!AN8n?uu7-!=)6tLQ_@am4_KeW-~ zRo)|0>t2upFa7l`@N3e*Uj7q{dqdkqoaR_!jo^b?e!bZXzOAiupUC=t=fYZk>#vZe z*vF&w0gRdU9B%qG@Z&^_e0c6CW6k7+o}j;>Fa!HR#r@KG=c^&-t>pfE`6B;lu5#yX zGv|DzeNdYsZUGBx4~{$P_RW6eb<+h*aq^m_NQxl7wSq5SUQ~*akDTP6Z#gVqu@rV{ zIrY2TXe3+S=%aJXJl1r)^6l!3&JQpirmY983p}_=x}?zIF7lIFc0Z=2^d^S}f*F2(F6-w-DRf zSCDXSm6M49#uNfkaKC6ZJrmjJ*VBT#7)5Z{g(2K~ldZf^e~4Y`;fd0)fLj0+P?M>w z7m6EOURtNQNxITv!&2$Mf{oTJ*I^I>zJ2UXrj%jdi+RALY_Vz|Tum)Rq5`tabRp;XjCUZWkySH?BcM*yX z^gy7~YP!HTZqI@Ea7jTP$Wg~|NnJdSNc0(+$?dAKMwP*;`*M~gh;R}tDPt@A{)n-BwH}M- z)d{DBg|XYScUe?B+wgc3#AQtqUdYF!m`&>cGtOy%mk8{9P-rQ1)p#2QLO%$fBU)ml z8jA2$OV;(nK(cTuH#F=b_Mz~ecuaT@RwYc7rklWXGcZ_*_&hpzbnw(mls{+ag#01M zexq5RLJ5ObtCyw-qSHOy(d%l^x(pX4ptrFdw&W-+fe;AOX=-y&nZrjdMH2gSlpL); z027}iJrqgXn74+rmlOdZn(J+xnpNlOE1yt^OGB`pRg>A8nH1DMTU>~fXB)^0d*J;` zi!g+8s5W=cqe@SR$%9Rm{YS6d4G@cftu-U!q!kfPpDnd5U?mD1v-4Vj&%_p#CSrN< z&meaztlmT0FmdLLPkHA>uYQ-}pQh)~UNwqFF)Crme`)P*gkOPB!5AheTwwqwE5EHU zIi^UUhhmFNqnhTTA)5;=Gqu<^TwvNc@T(fA0e=o*Gb0zReZ~ioCoj{R!eQsiR* z(KPu-?Kdj}NaiW?(x^@CWZGz{IC|ZG3XHg^dR;7Jv0{k-ZodS!P19C0f9>vT=JWyUQXNFqatgrq}-M4_cs)t zWH>hAzTA0dlT*n((e*3FxB7zTl%uDs|NWrNVOvWA;spAlx0z&}vHDxsZ6qBG2(FWx zv>K{8?|Jg|37`11ggm*n&|i%!B&&UO_h67jC?CnGm6Jh?cHLp7r^nCt&Pf$&pnVOx zaKxp=63I6-rFZtPW_KOle38m1;7+=O-VDhEXf6H34(c| z(QaLe3cFE(62~%u$T1=qpC42Mlo^U8^24*Hl-{HC@Ju7(2S_pT))5$yJBxI+8};#( z_0a$}A&Y0JxojgGYAl}PmWC!!!!c!vW!a$(^3D4!J#~O1hEF1)nWT&ms0`J8dghhj zv;R$;KH23ahT{xp-O(}t3lkFP9=-zx5tdF%DYlTE)?;cqD+7+eXeQa4r%I@P6!6|I z;|@_qy+WaE@&_^igmZ%y&RCz-jLFs~)%_O6F>ElQLWscecq8~}Zs}W1aHj^8Frpb0 z+71w)MjtH!H2P_VdtDcRij)PRD4#4R>>VBGmb{ z%-Oa0V9Le6m|Hf_nYYcCn%G&M=$xb&2z|(8E2O{SXEhepi2z1jh$UC1Ih%_Ed-CEJ zo_lYB+OxHNlUKHdWlAPS8GcuCWXbY;&W=m^!E;$6k`gAlbU`bjq<#Eq?7!NF|DBZs zqi4OJ3iJT;*E5lRlMcU;M@iLt!5~b8HgtfV%1=Tr66`7anMH3M*O}87iA;qZlM=Xs zp#dwYAA3r7(DXTDX7aed5p0(9@$*LYLvL8-PfasTlE(=i?dfb|&?7pR4R6fGJ1PJ| zPa6Uaa@8KSQQ(44WMSgSuA08b5Ak4&`zeUHvehk;d_yA$euSFO0(#_UqpN4(LH#9` z_91yB9zvUkmK0U^aYp8|rW!izn9fSlNDFnJoexA0o{y6oo$mVcoSk#FHSWCze?XU> z?V;h=M`#G5;+9hJ(E4N)dYagpjoayt&)LyuGQz%c@L7>B<>{C!9`MMNL$z`pxCCWu zxDaYy$GL?KaJC6hi!q!(<9HdunRX!wo0I$DSTE!FMXr!94MfyW+gSYT89tj|{8CaL>c_ z51x?BsEVa7Iap(}vY8TgIrs1?hs0%$4|dCdjbAV_f;iyp8VqFOmEiXUO@&BnF&W*xV49?-gePRVPiGARXE(I{M} zcr&c%eR+YlxBokk<03vsZWaCVln`HE)Mzj^iY>+FTb_)1`3nD?pC)NGxU@;KT*s(cWoVA;hW{JuZfKW3qX$Z06^{ zaLtM-Dj`gX#v)euw6i}sD0!+~g#zD#p{iHlN;si9UTmjaN0hT|V^ zpuDyZ3_c)41Jppnm|NSL`n55J#^&f1zZc&R;Z(~#T)waNATDKa;W;K=%Qu&xT+}31 z_MoI2-OLP&X;Vfot%~?V@zwK$V4300(gpZD*%r;0jNscS^fQo%$2tjCMGb8eBo%&TKv_kO` z7d~Kw%l-vAdvy`$Jl^4|YN9XFNmAw%dAJgYPKIcwOlD}Y0=%}*Y1lLVhQdS1wa>T_ zo&C=4!7A;^C!&_I3SvFcV-bJfJWYZMnR!J9gV1vF-o%Ql>#UNpumDp3k5NQ(gVy$o zMG~v=(A>Hod#l!1wI6xky<4+msOZdVtLkA$S?I8YD<@ei-~6#-W4-+WBf z7r@T1l)Gpb#AH>LNP#bnT`rc0z4+Xz_6y^t zV;WD}KqYWFnI!^l%{1xkvYP>;yvplC%8dF3RjGt zx;oPumcFMmdOM=uc5x`o1*j<(TzvjzlbzF#Tc`GuE^lvWu5Kz?zu7ABrdl7PE0-aG zj_R0RGH^Z;hx^J@)X5<(WL%ak2357t>PgO9XSvW`H?yTqTcKtpObk&S;T(N^ zLT(kebOITc=ekPE32V}U`0fqAy|E1HG55^FCC(JSB-r#df}2$jo9bgjsh^5$1wpza zhXJAuR+kop1C>x-1I7}RbxHPmcNWErT%4%k?exiRW_?y%EH Date: Mon, 22 Jan 2024 18:36:23 +0800 Subject: [PATCH 447/637] update truncate. --- metagpt/actions/execute_code.py | 21 ++++++++++----------- 1 file changed, 10 insertions(+), 11 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 851794b91..a5a766ab2 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -237,8 +237,10 @@ class ExecutePyCode(ExecuteCode, Action): def truncate(result: str, keep_len: int = 2000, is_success: bool = True): - """执行失败的代码, 展示result后keep_len个字符; 执行成功的代码, 展示result前keep_len个字符。""" + """对于超出keep_len个字符的result: 执行失败的代码, 展示result后keep_len个字符; 执行成功的代码, 展示result前keep_len个字符。""" desc = f"Executed code {'successfully. ' if is_success else 'failed, please reflect the cause of bug and then debug. '}" + is_same_desc = False + if is_success: desc += f"Truncated to show only first {keep_len} characters\n" else: @@ -246,20 +248,17 @@ def truncate(result: str, keep_len: int = 2000, is_success: bool = True): if result.startswith(desc): result = result[len(desc) :] + is_same_desc = True + + if result.strip().startswith(" keep_len: result = result[-keep_len:] if not is_success else result[:keep_len] - if not result: - result = "No output about your code. Only when importing packages it is normal case. Recap and go ahead." - return result, False + return desc + result, is_success - if result.strip().startswith(" Date: Mon, 22 Jan 2024 18:52:55 +0800 Subject: [PATCH 448/637] fix bugs --- .gitignore | 1 + examples/example.faiss | Bin 12333 -> 0 bytes examples/example.pkl | Bin 624 -> 0 bytes metagpt/actions/write_code_review.py | 2 +- tests/data/rsp_cache.json | 3 ++- tests/metagpt/provider/test_zhipuai_api.py | 4 ++-- tests/metagpt/roles/test_engineer.py | 2 +- tests/metagpt/test_context_mixin.py | 1 + tests/metagpt/test_role.py | 2 +- 9 files changed, 9 insertions(+), 6 deletions(-) delete mode 100644 examples/example.faiss delete mode 100644 examples/example.pkl diff --git a/.gitignore b/.gitignore index ed45cb260..3c762de4c 100644 --- a/.gitignore +++ b/.gitignore @@ -174,5 +174,6 @@ htmlcov htmlcov.* *.dot *.pkl +*.faiss *-structure.csv *-structure.json diff --git a/examples/example.faiss b/examples/example.faiss deleted file mode 100644 index 58094619004ac7b01cf52e596e1bb4bf254f4322..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 12333 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 bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Args:\n param1: The first parameter.\n\n Returns:\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Args:\n msg: Human readable string describing the exception.\n\n Attributes:\n msg: Human readable string describing the exception.\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\nimport pytest\nfrom metagpt.actions.write_docstring import WriteDocstring\ncode = '\\ndef add_numbers(a: int, b: int):\\n return a + b\\n\\n\\nclass Person:\\n def __init__(self, name: str, age: int):\\n self.name = name\\n self.age = age\\n\\n def greet(self):\\n return f\"Hello, my name is {self.name} and I am {self.age} years old.\"\\n'\n\n@pytest.mark.asyncio\n@pytest.mark.parametrize(('style', 'part'), [('google', 'Args:'), ('numpy', 'Parameters'), ('sphinx', ':param name:')], ids=['google', 'numpy', 'sphinx'])\nasync def test_write_docstring(style: str, part: str, context):\n ret = await WriteDocstring(context=context).run(code, style=style)\n assert part in ret\n\n@pytest.mark.asyncio\nasync def test_write():\n code = await WriteDocstring.write_docstring(__file__)\n assert code\n```": "```python\ndef add_numbers(a: int, b: int):\n \"\"\"Adds two numbers together.\n\n Args:\n a: The first number.\n b: The second number.\n\n Returns:\n The sum of a and b.\n \"\"\"\n ...\n\nclass Person:\n \"\"\"Represents a person.\n\n Args:\n name: The name of the person.\n age: The age of the person.\n\n Attributes:\n name: The name of the person.\n age: The age of the person.\n \"\"\"\n\n def __init__(self, name: str, age: int):\n ...\n \n def greet(self):\n \"\"\"Generates a greeting message.\n\n Returns:\n A greeting string.\n \"\"\"\n ...\n```" } \ No newline at end of file diff --git a/tests/metagpt/provider/test_zhipuai_api.py b/tests/metagpt/provider/test_zhipuai_api.py index ad2ececa2..798209710 100644 --- a/tests/metagpt/provider/test_zhipuai_api.py +++ b/tests/metagpt/provider/test_zhipuai_api.py @@ -17,7 +17,7 @@ default_resp = { } -async def mock_zhipuai_acreate_stream(self, **kwargs): +async def mock_zhipuai_acreate_stream(**kwargs): class MockResponse(object): async def _aread(self): class Iterator(object): @@ -37,7 +37,7 @@ async def mock_zhipuai_acreate_stream(self, **kwargs): return MockResponse() -async def mock_zhipuai_acreate(self, **kwargs) -> dict: +async def mock_zhipuai_acreate(**kwargs) -> dict: return default_resp diff --git a/tests/metagpt/roles/test_engineer.py b/tests/metagpt/roles/test_engineer.py index 383d28096..d263a8a2f 100644 --- a/tests/metagpt/roles/test_engineer.py +++ b/tests/metagpt/roles/test_engineer.py @@ -99,7 +99,7 @@ def test_parse_code(): def test_todo(): role = Engineer() - assert role.todo == any_to_name(WriteCode) + assert role.action_description == any_to_name(WriteCode) @pytest.mark.asyncio diff --git a/tests/metagpt/test_context_mixin.py b/tests/metagpt/test_context_mixin.py index 1ef0e4832..4389dc251 100644 --- a/tests/metagpt/test_context_mixin.py +++ b/tests/metagpt/test_context_mixin.py @@ -109,6 +109,7 @@ async def test_config_priority(): if not home_dir.exists(): assert gpt4t is None gpt35 = Config.default() + gpt35.llm.model = "gpt-3.5-turbo-1106" gpt4 = Config.default() gpt4.llm.model = "gpt-4-0613" diff --git a/tests/metagpt/test_role.py b/tests/metagpt/test_role.py index 1b843795c..7e707803b 100644 --- a/tests/metagpt/test_role.py +++ b/tests/metagpt/test_role.py @@ -131,7 +131,7 @@ async def test_recover(): role.recovered = True role.latest_observed_msg = Message(content="recover_test") role.rc.state = 0 - assert role.first_action == any_to_name(MockAction) + assert role.action_description == any_to_name(MockAction) rsp = await role.run() assert rsp.cause_by == any_to_str(MockAction) From 1bf9e023e4f682aa825f1f0a2d321aebe2f1637b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Mon, 22 Jan 2024 18:53:40 +0800 Subject: [PATCH 449/637] add new test. --- tests/metagpt/actions/test_execute_code.py | 40 +++++++++++++++++++--- 1 file changed, 36 insertions(+), 4 deletions(-) diff --git a/tests/metagpt/actions/test_execute_code.py b/tests/metagpt/actions/test_execute_code.py index ecddccf6f..21627e6f9 100644 --- a/tests/metagpt/actions/test_execute_code.py +++ b/tests/metagpt/actions/test_execute_code.py @@ -1,7 +1,6 @@ import pytest from metagpt.actions.execute_code import ExecutePyCode, truncate -from metagpt.schema import Message @pytest.mark.asyncio @@ -11,9 +10,6 @@ async def test_code_running(): assert output[1] is True output = await pi.run({"code": "print('hello world!')", "language": "python"}) assert output[1] is True - code_msg = Message("print('hello world!')") - output = await pi.run(code_msg) - assert output[1] is True @pytest.mark.asyncio @@ -67,6 +63,15 @@ def test_truncate(): output, is_success = truncate("hello world", 5, False) assert "Truncated to show only last 5 characters\nworld" in output assert not is_success + # 异步 + output, is_success = truncate(" Date: Mon, 22 Jan 2024 18:54:21 +0800 Subject: [PATCH 450/637] update _process_code. --- metagpt/actions/execute_code.py | 16 +++------------- 1 file changed, 3 insertions(+), 13 deletions(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index a5a766ab2..6591f479f 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -165,7 +165,7 @@ class ExecutePyCode(ExecuteCode, Action): # 如果在Python脚本中运行,__file__ 变量存在 return False - def _process_code(self, code: Union[str, Dict, Message], language: str = None) -> Tuple: + def _process_code(self, code: Union[str, Dict], language: str = None) -> Tuple: language = language or "python" if isinstance(code, str) and Path(code).suffix in (".py", ".txt"): code = Path(code).read_text(encoding="utf-8") @@ -173,20 +173,10 @@ class ExecutePyCode(ExecuteCode, Action): if isinstance(code, str): return code, language + if isinstance(code, dict): assert "code" in code - if "language" not in code: - code["language"] = "python" - code, language = code["code"], code["language"] - elif isinstance(code, Message): - if isinstance(code.content, dict) and "language" not in code.content: - code.content["language"] = "python" - code, language = code.content["code"], code.content["language"] - elif isinstance(code.content, str): - code, language = code.content, language - else: - raise ValueError(f"Not support code type {type(code).__name__}.") - + code = code["code"] return code, language async def run_cell(self, cell: NotebookNode, cell_index: int) -> Tuple[bool, str]: From 246d88748ebedc64af7b135bad3c9839623a4d3d Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 22 Jan 2024 19:11:33 +0800 Subject: [PATCH 451/637] update rsp cache --- tests/data/rsp_cache.json | 16 +++++++++++++++- 1 file changed, 15 insertions(+), 1 deletion(-) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index af1c93b23..462d4b084 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -262,5 +262,19 @@ "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n150\n": "It seems like you have evaluated the given numbers correctly. Here are the results:\n\n10 14: Sure\n11 12: Impossible\n4 4 10: Sure\n4 9 11: Sure\n5 7 8: Likely\n5 6 6: Likely\n10 10 11: Impossible\n1 3 3: Impossible\n\nAs for the number 150, it is not clear what operation or combination of numbers it is associated with. If you have a specific question or context for the number 150, please provide more details so that I can assist you further.", "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n5\n": "The numbers 10 and 14 can reach 24, so that is sure. The numbers 11 and 12 cannot reach 24, so that is impossible. The numbers 4, 4, and 10 can reach 24, so that is sure. The numbers 4, 9, and 11 can reach 24, so that is sure. The numbers 5, 7, and 8 cannot reach 24, but they are within a reasonable range, so that is likely. The numbers 5, 6, and 6 cannot reach 24, but they are within a reasonable range, so that is likely. The numbers 10, 10, and 11 cannot reach 24, so that is impossible. The numbers 1, 3, and 3 cannot reach 24, so that is impossible. The number 5 by itself cannot reach 24.", "Evaluate if given numbers can reach 24 (sure/likely/impossible)\n10 14\n10 + 14 = 24\nsure\n11 12\n11 + 12 = 23\n12 - 11 = 1\n11 * 12 = 132\n11 / 12 = 0.91\nimpossible\n4 4 10\n4 + 4 + 10 = 8 + 10 = 18\n4 * 10 - 4 = 40 - 4 = 36\n(10 - 4) * 4 = 6 * 4 = 24\nsure\n4 9 11\n9 + 11 + 4 = 20 + 4 = 24\nsure\n5 7 8\n5 + 7 + 8 = 12 + 8 = 20\n(8 - 5) * 7 = 3 * 7 = 21\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n5 6 6\n5 + 6 + 6 = 17\n(6 - 5) * 6 = 1 * 6 = 6\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely\n10 10 11\n10 + 10 + 11 = 31\n(11 - 10) * 10 = 10\n10 10 10 are all too big\nimpossible\n1 3 3\n1 * 3 * 3 = 9\n(1 + 3) * 3 = 12\n1 3 3 are all too small\nimpossible\n7 9 10\n": "7 * 9 - 10 = 63 - 10 = 53\n(9 - 7) * 10 = 2 * 10 = 20\nI cannot obtain 24 now, but numbers are within a reasonable range\nlikely", - "### Requirements\n1. Add docstrings to the given code following the google style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Args:\n param1: The first parameter.\n\n Returns:\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Args:\n msg: Human readable string describing the exception.\n\n Attributes:\n msg: Human readable string describing the exception.\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\nimport pytest\nfrom metagpt.actions.write_docstring import WriteDocstring\ncode = '\\ndef add_numbers(a: int, b: int):\\n return a + b\\n\\n\\nclass Person:\\n def __init__(self, name: str, age: int):\\n self.name = name\\n self.age = age\\n\\n def greet(self):\\n return f\"Hello, my name is {self.name} and I am {self.age} years old.\"\\n'\n\n@pytest.mark.asyncio\n@pytest.mark.parametrize(('style', 'part'), [('google', 'Args:'), ('numpy', 'Parameters'), ('sphinx', ':param name:')], ids=['google', 'numpy', 'sphinx'])\nasync def test_write_docstring(style: str, part: str, context):\n ret = await WriteDocstring(context=context).run(code, style=style)\n assert part in ret\n\n@pytest.mark.asyncio\nasync def test_write():\n code = await WriteDocstring.write_docstring(__file__)\n assert code\n```": "```python\ndef add_numbers(a: int, b: int):\n \"\"\"Adds two numbers together.\n\n Args:\n a: The first number.\n b: The second number.\n\n Returns:\n The sum of a and b.\n \"\"\"\n ...\n\nclass Person:\n \"\"\"Represents a person.\n\n Args:\n name: The name of the person.\n age: The age of the person.\n\n Attributes:\n name: The name of the person.\n age: The age of the person.\n \"\"\"\n\n def __init__(self, name: str, age: int):\n ...\n \n def greet(self):\n \"\"\"Generates a greeting message.\n\n Returns:\n A greeting string.\n \"\"\"\n ...\n```" + "### Requirements\n1. Add docstrings to the given code following the google style.\n2. Replace the function body with an Ellipsis object(...) to reduce output.\n3. If the types are already annotated, there is no need to include them in the docstring.\n4. Extract only class, function or the docstrings for the module parts from the given Python code, avoiding any other text.\n\n### Input Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n return isinstance(param1, int)\n\nclass ExampleError(Exception):\n def __init__(self, msg: str):\n self.msg = msg\n```\n\n### Output Example\n```python\ndef function_with_pep484_type_annotations(param1: int) -> bool:\n \"\"\"Example function with PEP 484 type annotations.\n\n Extended description of function.\n\n Args:\n param1: The first parameter.\n\n Returns:\n The return value. True for success, False otherwise.\n \"\"\"\n ...\n\nclass ExampleError(Exception):\n \"\"\"Exceptions are documented in the same way as classes.\n\n The __init__ method was documented in the class level docstring.\n\n Args:\n msg: Human readable string describing the exception.\n\n Attributes:\n msg: Human readable string describing the exception.\n \"\"\"\n ...\n```\n#SYSTEM_MSG_END#```python\nimport pytest\nfrom metagpt.actions.write_docstring import WriteDocstring\ncode = '\\ndef add_numbers(a: int, b: int):\\n return a + b\\n\\n\\nclass Person:\\n def __init__(self, name: str, age: int):\\n self.name = name\\n self.age = age\\n\\n def greet(self):\\n return f\"Hello, my name is {self.name} and I am {self.age} years old.\"\\n'\n\n@pytest.mark.asyncio\n@pytest.mark.parametrize(('style', 'part'), [('google', 'Args:'), ('numpy', 'Parameters'), ('sphinx', ':param name:')], ids=['google', 'numpy', 'sphinx'])\nasync def test_write_docstring(style: str, part: str, context):\n ret = await WriteDocstring(context=context).run(code, style=style)\n assert part in ret\n\n@pytest.mark.asyncio\nasync def test_write():\n code = await WriteDocstring.write_docstring(__file__)\n assert code\n```": "```python\ndef add_numbers(a: int, b: int):\n \"\"\"Adds two numbers together.\n\n Args:\n a: The first number.\n b: The second number.\n\n Returns:\n The sum of a and b.\n \"\"\"\n ...\n\nclass Person:\n \"\"\"Represents a person.\n\n Args:\n name: The name of the person.\n age: The age of the person.\n\n Attributes:\n name: The name of the person.\n age: The age of the person.\n \"\"\"\n\n def __init__(self, name: str, age: int):\n ...\n \n def greet(self):\n \"\"\"Generates a greeting message.\n\n Returns:\n A greeting string.\n \"\"\"\n ...\n```", + "\nWrite a coherent passage of 4 short paragraphs. The end sentence of each paragraph must be: It isn't difficult to do a handstand if you just stand on your hands. It caught him off guard that space smelled of seared steak. When she didn’t like a guy who was trying to pick her up, she started using sign language. Each person who knows you has a different perception of who you are.\n\nMake a plan then write. Your output should be like:\n\nPlan:\n\n\nPassage:\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Plan:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"Introduce the concept of doing a handstand.\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"Describe the surprising smell of space.\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"Introduce a woman using sign language to deter unwanted attention.\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"Discuss the varied perceptions people have of an individual.\"\n }\n]\n```\n\nPassage:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"It isn't difficult to do a handstand if you just stand on your hands.\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"It caught him off guard that space smelled of seared steak.\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"When she didn’t like a guy who was trying to pick her up, she started using sign language.\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"Each person who knows you has a different perception of who you are.\"\n }\n]\n```", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 4:\nDiscuss the varied perceptions people have of an individual.\n": "Choice 4: Discuss the varied perceptions people have of an individual.\n\nAnalyzing this choice, discussing the varied perceptions people have of an individual can be promising as it allows for a deeper understanding of how different people perceive and interpret the actions, behaviors, and characteristics of an individual. This can lead to insights into the complexities of human interactions and relationships, and can provide valuable information for personal growth and self-awareness.\n\nBy exploring the reasons behind differing perceptions, it can also lead to increased empathy and understanding of others, which is beneficial for building stronger relationships and effective communication.\n\nFurthermore, discussing varied perceptions can also shed light on societal influences, biases, and stereotypes that may impact how individuals are perceived, which can be valuable for promoting inclusivity and diversity.\n\nThe best choice is 4.", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 3:\nIntroduce a woman using sign language to deter unwanted attention.\n": "Choice 3: Introduce a woman using sign language to deter unwanted attention.\n\nAnalysis:\nIntroducing a woman using sign language can be a promising choice to deter unwanted attention. Sign language is a visual and expressive form of communication, and it can be an effective way to convey a message without drawing unnecessary attention. By using sign language, the woman can communicate with others discreetly and avoid attracting unwanted attention from individuals who may be seeking to engage her in conversation or interaction.\n\nAdditionally, using sign language can create a sense of privacy and exclusivity, as it may not be easily understood by everyone in the vicinity. This can help the woman maintain a level of personal space and discourage unwelcome advances or interactions.\n\nFurthermore, introducing sign language into the interaction can also serve as a way to promote inclusivity and awareness of different forms of communication, which can contribute to a more understanding and respectful environment.\n\nConclusion:\nThe best choice is 3.", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 1:\nIntroduce the concept of doing a handstand.\n": "Choice 1: Introduce the concept of doing a handstand.\n\nAnalyzing the choice:\nIntroducing the concept of doing a handstand can be a fun and engaging way to promote physical activity and coordination. It can also help build strength and balance. However, it's important to ensure that proper safety measures and instructions are provided to prevent any potential injuries.\n\nConclusion:\nThe best choice is 1.", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 2:\nDescribe the surprising smell of space.\n": "Choice 2: \n\nThe surprising smell of space has been described by astronauts as a mix of gunpowder, seared steak, hot metal, and raspberries. This unexpected combination of scents is due to the volatile organic compounds that are released from the materials in the spacecraft when they are exposed to the vacuum of space. These compounds react with the radiation from the sun, creating the unique smell that astronauts have reported. The smell of space is intriguing and unexpected, providing a sensory experience that is truly out of this world.\n\nThe best choice is 2.", + "\nWrite a coherent passage of 4 short paragraphs. The end sentence of each paragraph must be: Introduce the concept of doing a handstand.\n\nMake a plan then write. Your output should be like:\n\nPlan:\n\n\nPassage:\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Plan:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"Decide on a suitable location to practice handstands.\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"Warm up your body with some stretching exercises.\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"Practice against a wall to build strength and balance.\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"Gradually work on holding the handstand position without support.\"\n }\n]\n```\n\nPassage:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"Decide on a suitable location to practice handstands.\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"Before attempting a handstand, it's important to decide on a suitable location to practice. A clear, open space with a soft surface, such as a yoga mat or grass, is ideal for safety and comfort.\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"Warm up your body with some stretching exercises.\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"To prepare your body for the physical demands of a handstand, start with a series of stretching exercises. Focus on the wrists, shoulders, and core muscles to ensure they are adequately warmed up and flexible.\"\n },\n {\n \"node_id\": \"5\",\n \"node_state_instruction\": \"Practice against a wall to build strength and balance.\"\n },\n {\n \"node_id\": \"6\",\n \"node_state_instruction\": \"When learning to do a handstand, it can be helpful to practice against a wall. This provides support and allows you to focus on building strength and balance in the proper alignment. Gradually work on kicking up into the handstand position and holding it for short periods of time.\"\n },\n {\n \"node_id\": \"7\",\n \"node_state_instruction\": \"Gradually work on holding the handstand position without support.\"\n },\n {\n \"node_id\": \"8\",\n \"node_state_instruction\": \"As you gain confidence and strength, begin to practice holding the handstand position without the support of a wall. Focus on engaging your core and maintaining a straight line from your wrists to your hips. With consistent practice, you'll be able to hold a handstand with control and stability.\"\n }\n]\n```", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 1:\nDecide on a suitable location to practice handstands.\n": "Choice 1: Find a spacious and flat area with soft ground, such as a grassy field or a padded gym floor. This will minimize the risk of injury and provide a stable surface for practicing handstands.\n\nThe best choice is 1", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 2:\nWarm up your body with some stretching exercises.\n": "Stretching exercises are a great way to warm up the body before engaging in physical activity. They help to increase blood flow to the muscles, improve flexibility, and reduce the risk of injury. By performing stretching exercises, you can prepare your body for more strenuous activities and improve your overall performance.\n\nThe best choice is 2.", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 3:\nPractice against a wall to build strength and balance.\n": "Choice 3: Practice against a wall to build strength and balance.\n\nAnalyzing choice 3, practicing against a wall can be a promising option for building strength and balance. The wall provides a stable surface to push against, which can help in developing strength in the muscles. It also offers support, allowing for balance practice without the risk of falling. Additionally, using a wall as a tool for practice can help in improving posture and body alignment.\n\nThe best choice is 3.", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 4:\nGradually work on holding the handstand position without support.\n": "Choice 4: Gradually work on holding the handstand position without support.\n\nAnalyzing this choice, working on holding the handstand position without support is a promising option for improving strength, balance, and body control. This choice allows for gradual progression, which is important for building the necessary strength and stability to maintain the handstand position. It also encourages the development of proprioception and spatial awareness, which are essential for mastering this challenging skill.\n\nThe best choice is 4.", + "\nWrite a coherent passage of 4 short paragraphs. The end sentence of each paragraph must be: Decide on a suitable location to practice handstands.\n\nMake a plan then write. Your output should be like:\n\nPlan:\n\n\nPassage:\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Plan:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"Decide on a suitable location to practice handstands.\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"Consider a flat and stable surface, such as a grassy field or a gym mat.\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"Ensure there is enough space around you to avoid any obstacles.\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"Practice near a wall or with a spotter for safety.\"\n }\n]\n```\n\nPassage:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"Decide on a suitable location to practice handstands.\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"Consider a flat and stable surface, such as a grassy field or a gym mat.\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"Ensure there is enough space around you to avoid any obstacles.\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"Practice near a wall or with a spotter for safety.\"\n }\n]\n```", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 3:\nEnsure there is enough space around you to avoid any obstacles.\n": "Choice 3: \n\nAnalyzing this choice, ensuring there is enough space around you to avoid any obstacles is a crucial step in maintaining safety and preventing accidents. By creating a buffer zone, you can react to unexpected obstacles or changes in the environment without risking collisions or other hazards.\n\nThe best choice is 3.", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 2:\nConsider a flat and stable surface, such as a grassy field or a gym mat.\n": "Choice 2: Considering a flat and stable surface, such as a grassy field or a gym mat, is the most promising option. This choice provides a safe and secure environment for the activity, reducing the risk of injury. A grassy field offers a natural and soft surface, while a gym mat provides cushioning and stability. Both options allow for comfortable movement and can accommodate various physical activities.\n\nThe best choice is 2.", + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 4:\nPractice near a wall or with a spotter for safety.\n": "Choice 4: \n\nPractice near a wall or with a spotter for safety.\n\nAnalyzing this choice, practicing near a wall or with a spotter provides a safety measure to prevent accidents or injuries. When practicing a new skill or exercise, having a wall nearby can provide support and stability, reducing the risk of falling or losing balance. Similarly, having a spotter can offer assistance and guidance, ensuring that the practice is done safely and effectively.\n\nThe best choice is 4." } \ No newline at end of file From 8b1378308500fbec611d7d5e661ab59d6c30c9b7 Mon Sep 17 00:00:00 2001 From: geekan Date: Mon, 22 Jan 2024 19:33:58 +0800 Subject: [PATCH 452/637] add rsp cache --- tests/data/rsp_cache.json | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index 462d4b084..92a600b67 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -276,5 +276,6 @@ "\nWrite a coherent passage of 4 short paragraphs. The end sentence of each paragraph must be: Decide on a suitable location to practice handstands.\n\nMake a plan then write. Your output should be like:\n\nPlan:\n\n\nPassage:\n\n\n\nEach output should be strictly a list of nodes, in json format, like this:\n```json\n [\n {\n \"node_id\": str = \"unique identifier for a solution, can be an ordinal\",\n \"node_state_instruction\": \"specified sample of solution\",\n },\n ...\n ]\n```\n": "Plan:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"Decide on a suitable location to practice handstands.\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"Consider a flat and stable surface, such as a grassy field or a gym mat.\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"Ensure there is enough space around you to avoid any obstacles.\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"Practice near a wall or with a spotter for safety.\"\n }\n]\n```\n\nPassage:\n```json\n[\n {\n \"node_id\": \"1\",\n \"node_state_instruction\": \"Decide on a suitable location to practice handstands.\"\n },\n {\n \"node_id\": \"2\",\n \"node_state_instruction\": \"Consider a flat and stable surface, such as a grassy field or a gym mat.\"\n },\n {\n \"node_id\": \"3\",\n \"node_state_instruction\": \"Ensure there is enough space around you to avoid any obstacles.\"\n },\n {\n \"node_id\": \"4\",\n \"node_state_instruction\": \"Practice near a wall or with a spotter for safety.\"\n }\n]\n```", "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 3:\nEnsure there is enough space around you to avoid any obstacles.\n": "Choice 3: \n\nAnalyzing this choice, ensuring there is enough space around you to avoid any obstacles is a crucial step in maintaining safety and preventing accidents. By creating a buffer zone, you can react to unexpected obstacles or changes in the environment without risking collisions or other hazards.\n\nThe best choice is 3.", "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 2:\nConsider a flat and stable surface, such as a grassy field or a gym mat.\n": "Choice 2: Considering a flat and stable surface, such as a grassy field or a gym mat, is the most promising option. This choice provides a safe and secure environment for the activity, reducing the risk of injury. A grassy field offers a natural and soft surface, while a gym mat provides cushioning and stability. Both options allow for comfortable movement and can accommodate various physical activities.\n\nThe best choice is 2.", - "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 4:\nPractice near a wall or with a spotter for safety.\n": "Choice 4: \n\nPractice near a wall or with a spotter for safety.\n\nAnalyzing this choice, practicing near a wall or with a spotter provides a safety measure to prevent accidents or injuries. When practicing a new skill or exercise, having a wall nearby can provide support and stability, reducing the risk of falling or losing balance. Similarly, having a spotter can offer assistance and guidance, ensuring that the practice is done safely and effectively.\n\nThe best choice is 4." + "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 4:\nPractice near a wall or with a spotter for safety.\n": "Choice 4: \n\nPractice near a wall or with a spotter for safety.\n\nAnalyzing this choice, practicing near a wall or with a spotter provides a safety measure to prevent accidents or injuries. When practicing a new skill or exercise, having a wall nearby can provide support and stability, reducing the risk of falling or losing balance. Similarly, having a spotter can offer assistance and guidance, ensuring that the practice is done safely and effectively.\n\nThe best choice is 4.", + "### Requirements\n1. Please summarize the latest dialogue based on the reference information (secondary) and dialogue history (primary). Do not include text that is irrelevant to the conversation.\n- The context is for reference only. If it is irrelevant to the user's search request history, please reduce its reference and usage.\n2. If there are citable links in the context, annotate them in the main text in the format [main text](citation link). If there are none in the context, do not write links.\n3. The reply should be graceful, clear, non-repetitive, smoothly written, and of moderate length, in {LANG}.\n\n### Dialogue History (For example)\nA: MLOps competitors\n\n### Current Question (For example)\nA: MLOps competitors\n\n### Current Reply (For example)\n1. Alteryx Designer: etc. if any\n2. Matlab: ditto\n3. IBM SPSS Statistics\n4. RapidMiner Studio\n5. DataRobot AI Platform\n6. Databricks Lakehouse Platform\n7. Amazon SageMaker\n8. Dataiku\n#SYSTEM_MSG_END#\n### Reference Information\nABC cleanser is preferred by many with oily skin.\nL'Oreal is a popular brand with many positive reviews.\n\n### Dialogue History\n\nuser: Which facial cleanser is good for oily skin?\n\n### Current Question\nuser: Which facial cleanser is good for oily skin?\n\n### Current Reply: Based on the information, please write the reply to the Question\n\n\n": "Based on the information provided, ABC cleanser is preferred by many with oily skin. It is a popular choice for individuals with oily skin due to its effectiveness. Additionally, L'Oreal is a well-known brand with many positive reviews, and they offer a range of products suitable for oily skin. Both of these options could be good choices for individuals with oily skin." } \ No newline at end of file From 91c5c7208936bacfbeac5938b0a01e1f7ee47c12 Mon Sep 17 00:00:00 2001 From: Arnaud Gelas Date: Mon, 22 Jan 2024 20:10:11 +0100 Subject: [PATCH 453/637] Fix prompt logic when defining to who the message should be sent. With the previous logic, it was possible to reach an undefined state where it was not meant to be sent to Engineer, QaEngineer, nor NoOne. --- metagpt/actions/run_code.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/run_code.py b/metagpt/actions/run_code.py index 30b06f1a6..885f4e12c 100644 --- a/metagpt/actions/run_code.py +++ b/metagpt/actions/run_code.py @@ -42,8 +42,8 @@ Determine the ONE file to rewrite in order to fix the error, for example, xyz.py Determine if all of the code works fine, if so write PASS, else FAIL, WRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION ## Send To: -Please write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors, -WRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION. +Please write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer, +WRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION. --- You should fill in necessary instruction, status, send to, and finally return all content between the --- segment line. """ From 987eb6d38be5301f3c071c7f13fdd77b02cd095a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 23 Jan 2024 17:06:22 +0800 Subject: [PATCH 454/637] fix: now support parsing code in message.content when using tools_call. --- metagpt/provider/openai_api.py | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 3358b3aad..dad44087c 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -31,6 +31,7 @@ from metagpt.provider.base_llm import BaseLLM from metagpt.provider.constant import GENERAL_FUNCTION_SCHEMA from metagpt.provider.llm_provider_registry import register_provider from metagpt.schema import Message +from metagpt.utils.common import CodeParser from metagpt.utils.cost_manager import Costs from metagpt.utils.exceptions import handle_exception from metagpt.utils.token_counter import ( @@ -247,6 +248,11 @@ class OpenAILLM(BaseLLM): ) return self._parse_arguments(message.tool_calls[0].function.arguments) elif message.tool_calls is None and message.content is not None: + # reponse is code, fix openai tools_call respond bug. + code_formats = ("```", '"""', "'''") + if message.content.startswith(code_formats) and message.content.endswith(code_formats): + code = CodeParser.parse_code(None, message.content) + return {"language": "python", "code": code} # reponse is message return {"language": "markdown", "code": self.get_choice_text(rsp)} else: From 31813f2512b176b1838c13a703cc640846362da6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 23 Jan 2024 17:16:15 +0800 Subject: [PATCH 455/637] add new test for tool_calls_rsp. --- tests/metagpt/provider/test_openai.py | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index 2e5799475..7a771bcac 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -87,6 +87,16 @@ class TestOpenAI: messages.append( ChatCompletionMessage(content="Completed a python code for hello world!", role="assistant", tool_calls=None) ) + # 添加 openai tool calls respond bug, code 出现在ChatCompletionMessage.content中 + messages.extend( + [ + ChatCompletionMessage(content="```python\nprint('hello world')```", role="assistant", tool_calls=None), + ChatCompletionMessage(content="'''python\nprint('hello world')'''", role="assistant", tool_calls=None), + ChatCompletionMessage( + content='"""python\nprint(\'hello world\')"""', role="assistant", tool_calls=None + ), + ] + ) choices = [ Choice(finish_reason="tool_calls", logprobs=None, index=i, message=msg) for i, msg in enumerate(messages) ] From a06d8023d640ac3e87853d4b51aed595c919fe05 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 23 Jan 2024 17:36:19 +0800 Subject: [PATCH 456/637] update CodeParser.parse_code. --- metagpt/provider/openai_api.py | 2 +- metagpt/utils/common.py | 4 ++-- tests/metagpt/provider/test_openai.py | 1 + 3 files changed, 4 insertions(+), 3 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index dad44087c..fc741f038 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -251,7 +251,7 @@ class OpenAILLM(BaseLLM): # reponse is code, fix openai tools_call respond bug. code_formats = ("```", '"""', "'''") if message.content.startswith(code_formats) and message.content.endswith(code_formats): - code = CodeParser.parse_code(None, message.content) + code = CodeParser.parse_code(None, message.content, start_ends=r'["\'`]{3}') return {"language": "python", "code": code} # reponse is message return {"language": "markdown", "code": self.get_choice_text(rsp)} diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index b20b4acd2..36392debc 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -264,10 +264,10 @@ class CodeParser: return block_dict @classmethod - def parse_code(cls, block: str, text: str, lang: str = "") -> str: + def parse_code(cls, block: str, text: str, lang: str = "", start_ends: str = "```") -> str: if block: text = cls.parse_block(block, text) - pattern = rf"```{lang}.*?\s+(.*?)```" + pattern = rf"{start_ends}{lang}.*?\s+(.*?){start_ends}" match = re.search(pattern, text, re.DOTALL) if match: code = match.group(1) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index 7a771bcac..1743fed92 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -139,6 +139,7 @@ class TestOpenAI: assert "code" in code assert "language" in code assert "hello world" in code["code"] + logger.info(f'code is : {code["code"]}') if "Completed a python code for hello world!" == code["code"]: code["language"] == "markdown" From cff4eff78d3d7b015f0cd49cd22e5dfe3276186d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 23 Jan 2024 17:46:41 +0800 Subject: [PATCH 457/637] update CodeParser.parse_code. --- metagpt/provider/openai_api.py | 2 +- metagpt/utils/common.py | 3 ++- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index fc741f038..dad44087c 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -251,7 +251,7 @@ class OpenAILLM(BaseLLM): # reponse is code, fix openai tools_call respond bug. code_formats = ("```", '"""', "'''") if message.content.startswith(code_formats) and message.content.endswith(code_formats): - code = CodeParser.parse_code(None, message.content, start_ends=r'["\'`]{3}') + code = CodeParser.parse_code(None, message.content) return {"language": "python", "code": code} # reponse is message return {"language": "markdown", "code": self.get_choice_text(rsp)} diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 36392debc..ed73cb061 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -8,6 +8,7 @@ Add generic class-to-string and object-to-string conversion functionality. @Modified By: mashenquan, 2023/11/27. Bug fix: `parse_recipient` failed to parse the recipient in certain GPT-3.5 responses. +@Modified By: liubangbang, 2024/01/23. Update: support [```, ''', \"\"\" ] codes in CodeParser.parse_code. """ from __future__ import annotations @@ -264,7 +265,7 @@ class CodeParser: return block_dict @classmethod - def parse_code(cls, block: str, text: str, lang: str = "", start_ends: str = "```") -> str: + def parse_code(cls, block: str, text: str, lang: str = "", start_ends: str = r'["\'`]{3}') -> str: if block: text = cls.parse_block(block, text) pattern = rf"{start_ends}{lang}.*?\s+(.*?){start_ends}" From 0cc0a16e521d23d9d6fa5adee1120722b32a3e02 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 23 Jan 2024 17:50:04 +0800 Subject: [PATCH 458/637] add new test for tool_calls_rsp. --- tests/metagpt/provider/test_openai.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index 1743fed92..77820a5f8 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -95,6 +95,10 @@ class TestOpenAI: ChatCompletionMessage( content='"""python\nprint(\'hello world\')"""', role="assistant", tool_calls=None ), + ChatCompletionMessage( + content="'''python\nprint(\"hello world\")'''", role="assistant", tool_calls=None + ), + ChatCompletionMessage(content="```python\nprint('hello world')```", role="assistant", tool_calls=None), ] ) choices = [ From bcda7ac951df9ede69b60ae3dccb30ad10203541 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 23 Jan 2024 17:53:50 +0800 Subject: [PATCH 459/637] add comments for openai tools_call respond bug. --- metagpt/provider/openai_api.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index dad44087c..386c36c22 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -248,7 +248,8 @@ class OpenAILLM(BaseLLM): ) return self._parse_arguments(message.tool_calls[0].function.arguments) elif message.tool_calls is None and message.content is not None: - # reponse is code, fix openai tools_call respond bug. + # reponse is code, fix openai tools_call respond bug, + # The response content is `code``, but it appears in the content instead of the arguments. code_formats = ("```", '"""', "'''") if message.content.startswith(code_formats) and message.content.endswith(code_formats): code = CodeParser.parse_code(None, message.content) From 519f22f7bbc1229e5b8a38b4430ec283e1d02f3f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Tue, 23 Jan 2024 20:53:08 +0800 Subject: [PATCH 460/637] update CodeInterpreter._write_and_exec_code --- metagpt/roles/code_interpreter.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 11ede6068..3991862d1 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -97,7 +97,7 @@ class CodeInterpreter(Role): if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions - return code["code"], result, success + return code["code"] if code["language"] != "markdown" else "", result, success async def _write_code(self): todo = WriteCodeByGenerate() if not self.use_tools else WriteCodeWithTools(selected_tools=self.tools) From 0e4197395d6f75f2cc51b98339e8109ede008185 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 23 Jan 2024 19:11:58 +0800 Subject: [PATCH 461/637] 1. update CodePlanAndChangeContext in schema.py 2. add 'await' word in _update_prd function of write_prd.py --- metagpt/actions/action.py | 5 +++- .../actions/write_code_plan_and_change_an.py | 21 +++++++------ metagpt/actions/write_prd.py | 2 +- metagpt/actions/write_prd_an.py | 2 +- metagpt/roles/engineer.py | 30 +++++++------------ metagpt/schema.py | 30 +++++++++++++++---- 6 files changed, 52 insertions(+), 38 deletions(-) diff --git a/metagpt/actions/action.py b/metagpt/actions/action.py index addc672bc..1b93213f7 100644 --- a/metagpt/actions/action.py +++ b/metagpt/actions/action.py @@ -15,6 +15,7 @@ from pydantic import BaseModel, ConfigDict, Field, model_validator from metagpt.actions.action_node import ActionNode from metagpt.context_mixin import ContextMixin from metagpt.schema import ( + CodePlanAndChangeContext, CodeSummarizeContext, CodingContext, RunCodeContext, @@ -28,7 +29,9 @@ class Action(SerializationMixin, ContextMixin, BaseModel): model_config = ConfigDict(arbitrary_types_allowed=True) name: str = "" - i_context: Union[dict, CodingContext, CodeSummarizeContext, TestingContext, RunCodeContext, str, None] = "" + i_context: Union[ + dict, CodingContext, CodeSummarizeContext, TestingContext, RunCodeContext, CodePlanAndChangeContext, str, None + ] = "" prefix: str = "" # aask*时会加上prefix,作为system_message desc: str = "" # for skill manager node: ActionNode = Field(default=None, exclude=True) diff --git a/metagpt/actions/write_code_plan_and_change_an.py b/metagpt/actions/write_code_plan_and_change_an.py index 188520ba8..3fac22242 100644 --- a/metagpt/actions/write_code_plan_and_change_an.py +++ b/metagpt/actions/write_code_plan_and_change_an.py @@ -118,8 +118,8 @@ CODE_PLAN_AND_CHANGE_CONTEXT = """ ## Design {design} -## Tasks -{tasks} +## Task +{task} ## Legacy Code {code} @@ -139,8 +139,8 @@ Role: You are a professional engineer; The main goal is to complete incremental ## Design {design} -## Tasks -{tasks} +## Task +{task} ## Legacy Code ```Code @@ -189,13 +189,16 @@ class WriteCodePlanAndChange(Action): async def run(self, *args, **kwargs): self.llm.system_prompt = "You are a professional software engineer, your primary responsibility is to " "meticulously craft comprehensive incremental development plan and deliver detailed incremental change" - requirement = self.i_context.requirement_doc.content - prd = "\n".join([doc.content for doc in self.i_context.prd_docs]) - design = "\n".join([doc.content for doc in self.i_context.design_docs]) - tasks = "\n".join([doc.content for doc in self.i_context.tasks_docs]) + prd_doc = await self.repo.docs.prd.get(filename=self.i_context.prd_filename) + design_doc = await self.repo.docs.system_design.get(filename=self.i_context.design_filename) + task_doc = await self.repo.docs.task.get(filename=self.i_context.task_filename) code_text = await self.get_old_codes() context = CODE_PLAN_AND_CHANGE_CONTEXT.format( - requirement=requirement, prd=prd, design=design, tasks=tasks, code=code_text + requirement=self.i_context.requirement, + prd=prd_doc.content, + design=design_doc.content, + task=task_doc.content, + code=code_text, ) return await WRITE_CODE_PLAN_AND_CHANGE_NODE.fill(context=context, llm=self.llm, schema="json") diff --git a/metagpt/actions/write_prd.py b/metagpt/actions/write_prd.py index 000c1731e..823786893 100644 --- a/metagpt/actions/write_prd.py +++ b/metagpt/actions/write_prd.py @@ -147,7 +147,7 @@ class WritePRD(Action): async def _update_prd(self, req: Document, prd_doc: Document) -> Document: new_prd_doc: Document = await self._merge(req, prd_doc) - self.repo.docs.prd.save_doc(doc=new_prd_doc) + await self.repo.docs.prd.save_doc(doc=new_prd_doc) await self._save_competitive_analysis(new_prd_doc) await self.repo.resources.prd.save_pdf(doc=new_prd_doc) return new_prd_doc diff --git a/metagpt/actions/write_prd_an.py b/metagpt/actions/write_prd_an.py index 4baa46b12..9898be55b 100644 --- a/metagpt/actions/write_prd_an.py +++ b/metagpt/actions/write_prd_an.py @@ -33,7 +33,7 @@ ORIGINAL_REQUIREMENTS = ActionNode( REFINED_REQUIREMENTS = ActionNode( key="Refined Requirements", expected_type=str, - instruction="Place the New user's requirements here.", + instruction="Place the New user's original requirements here.", example="Create a 2048 game with a new feature that ...", ) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 3cf52fc35..ae4f40ac7 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -210,18 +210,16 @@ class Engineer(Role): node = await self.rc.todo.run() code_plan_and_change = node.instruct_content.model_dump_json() dependencies = { - self.rc.todo.i_context.requirement_doc.filename, - self.rc.todo.i_context.prd_docs[0].filename, - self.rc.todo.i_context.design_docs[0].filename, - self.rc.todo.i_context.tasks_docs[0].filename, + REQUIREMENT_FILENAME, + self.rc.todo.i_context.prd_filename, + self.rc.todo.i_context.design_filename, + self.rc.todo.i_context.task_filename, } - - code_plan_and_change_filename = os.path.join(CODE_PLAN_AND_CHANGE_FILE_REPO, CODE_PLAN_AND_CHANGE_FILENAME) await self.project_repo.resources.code_plan_and_change.save( - filename=code_plan_and_change_filename, content=code_plan_and_change, dependencies=dependencies + filename=self.rc.todo.i_context.filename, content=code_plan_and_change, dependencies=dependencies ) await self.project_repo.docs.code_plan_and_change.save( - filename=Path(code_plan_and_change_filename).with_suffix(".md").name, + filename=Path(self.rc.todo.i_context.filename).with_suffix(".md").name, content=node.content, dependencies=dependencies, ) @@ -350,18 +348,10 @@ class Engineer(Role): async def _new_code_plan_and_change_action(self): """Create a WriteCodePlanAndChange action for subsequent to-do actions.""" - requirement_doc = await self.project_repo.docs.requirement.get(REQUIREMENT_FILENAME) - prd_docs = await self.project_repo.docs.prd.get_all() - design_docs = await self.project_repo.docs.system_design.get_all() - task_docs = await self.project_repo.docs.task.get_all() - - code_plan_and_change_context = CodePlanAndChangeContext( - requirement_doc=requirement_doc, - prd_docs=prd_docs, - design_docs=design_docs, - task_docs=task_docs, - ) - self.rc.todo = WriteCodePlanAndChange(i_context=code_plan_and_change_context, llm=self.llm) + files = self.project_repo.all_files + requirement = str(self.rc.memory.get_by_role("Human")[0]) + code_plan_and_change_ctx = CodePlanAndChangeContext.loads(files, requirement=requirement) + self.rc.todo = WriteCodePlanAndChange(i_context=code_plan_and_change_ctx, context=self.context, llm=self.llm) @property def action_description(self) -> str: diff --git a/metagpt/schema.py b/metagpt/schema.py index 88e1712fc..ee194afbd 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -37,10 +37,12 @@ from pydantic import ( ) from metagpt.const import ( + CODE_PLAN_AND_CHANGE_FILENAME, MESSAGE_ROUTE_CAUSE_BY, MESSAGE_ROUTE_FROM, MESSAGE_ROUTE_TO, MESSAGE_ROUTE_TO_ALL, + PRDS_FILE_REPO, SYSTEM_DESIGN_FILE_REPO, TASK_FILE_REPO, ) @@ -470,12 +472,28 @@ class BugFixContext(BaseContext): filename: str = "" -class CodePlanAndChangeContext(BaseContext): - filename: str = "" - requirement_doc: Document - prd_docs: List[Document] - design_docs: List[Document] - task_docs: List[Document] +class CodePlanAndChangeContext(BaseModel): + filename: str = CODE_PLAN_AND_CHANGE_FILENAME + requirement: str = "" + prd_filename: str = "" + design_filename: str = "" + task_filename: str = "" + + @staticmethod + def loads(filenames: List, **kwargs) -> CodePlanAndChangeContext: + ctx = CodePlanAndChangeContext(requirement=kwargs.get("requirement", "")) + for filename in filenames: + filename = Path(filename) + if filename.is_relative_to(PRDS_FILE_REPO): + ctx.prd_filename = filename.name + continue + if filename.is_relative_to(SYSTEM_DESIGN_FILE_REPO): + ctx.design_filename = filename.name + continue + if filename.is_relative_to(TASK_FILE_REPO): + ctx.task_filename = filename.name + continue + return ctx # mermaid class view From 62c128602fa8ffdae5c4dc4daf488d003053d39d Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 24 Jan 2024 09:51:18 +0800 Subject: [PATCH 462/637] 1. update bug of getting requirement_doc 2. modify prompt --- metagpt/actions/write_code.py | 2 +- metagpt/actions/write_code_review.py | 13 +++++------ metagpt/roles/engineer.py | 7 +++--- tests/data/incremental_dev_project/mock.py | 2 +- .../test_write_code_plan_and_change_an.py | 23 +++++++++---------- tests/metagpt/actions/test_write_prd_an.py | 1 - tests/metagpt/test_incremental_dev.py | 2 +- 7 files changed, 24 insertions(+), 26 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index df1e383e7..ef6d1708d 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -101,7 +101,7 @@ class WriteCode(Action): test_doc = await self.repo.test_outputs.get(filename="test_" + coding_context.filename + ".json") code_plan_and_change_doc = await self.repo.docs.code_plan_and_change.get(filename=CODE_PLAN_AND_CHANGE_FILENAME) code_plan_and_change = code_plan_and_change_doc.content if code_plan_and_change_doc else "" - requirement_doc = await self.repo.docs.requirement.get(filename=REQUIREMENT_FILENAME) + requirement_doc = await self.repo.docs.get(filename=REQUIREMENT_FILENAME) summary_doc = None if coding_context.design_doc and coding_context.design_doc.filename: summary_doc = await self.repo.docs.code_summary.get(filename=coding_context.design_doc.filename) diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index f130542f9..b320be6ee 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -137,7 +137,8 @@ class WriteCodeReview(Action): async def run(self, *args, **kwargs) -> CodingContext: iterative_code = self.i_context.code_doc.content - k = self.context.config.code_review_k_times or 1 + # k = self.context.config.code_review_k_times or 1 + k = 1 code_plan_and_change_doc = await self.repo.get(filename=CODE_PLAN_AND_CHANGE_FILENAME) code_plan_and_change = code_plan_and_change_doc.content if code_plan_and_change_doc else "" mode = "incremental" if code_plan_and_change else "normal" @@ -154,20 +155,18 @@ class WriteCodeReview(Action): if not code_plan_and_change: context = "\n".join( [ - "## System Design\n" + str(self.context.design_doc) + "\n", + "## System Design\n" + str(self.i_context.design_doc) + "\n", "## Tasks\n" + task_content + "\n", "## Code Files\n" + code_context + "\n", ] ) else: - requirement_doc = await self.repo.get(filename=REQUIREMENT_FILENAME) - user_requirement = requirement_doc.content if requirement_doc else "" - + requirement_doc = await self.repo.docs.get(filename=REQUIREMENT_FILENAME) context = "\n".join( [ - "## User New Requirements\n" + user_requirement + "\n", + "## User New Requirements\n" + str(requirement_doc) + "\n", "## Code Plan And Change\n" + code_plan_and_change + "\n", - "## System Design\n" + str(self.context.design_doc) + "\n", + "## System Design\n" + str(self.i_context.design_doc) + "\n", "## Tasks\n" + task_content + "\n", "## Code Files\n" + code_context + "\n", ] diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index ae4f40ac7..8635e5fcf 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -215,10 +215,10 @@ class Engineer(Role): self.rc.todo.i_context.design_filename, self.rc.todo.i_context.task_filename, } - await self.project_repo.resources.code_plan_and_change.save( + await self.project_repo.docs.code_plan_and_change.save( filename=self.rc.todo.i_context.filename, content=code_plan_and_change, dependencies=dependencies ) - await self.project_repo.docs.code_plan_and_change.save( + await self.project_repo.resources.code_plan_and_change.save( filename=Path(self.rc.todo.i_context.filename).with_suffix(".md").name, content=node.content, dependencies=dependencies, @@ -349,7 +349,8 @@ class Engineer(Role): async def _new_code_plan_and_change_action(self): """Create a WriteCodePlanAndChange action for subsequent to-do actions.""" files = self.project_repo.all_files - requirement = str(self.rc.memory.get_by_role("Human")[0]) + requirement_doc = await self.project_repo.docs.get(REQUIREMENT_FILENAME) + requirement = requirement_doc.content if requirement_doc else "" code_plan_and_change_ctx = CodePlanAndChangeContext.loads(files, requirement=requirement) self.rc.todo = WriteCodePlanAndChange(i_context=code_plan_and_change_ctx, context=self.context, llm=self.llm) diff --git a/tests/data/incremental_dev_project/mock.py b/tests/data/incremental_dev_project/mock.py index 5c5191cf2..f2eb71359 100644 --- a/tests/data/incremental_dev_project/mock.py +++ b/tests/data/incremental_dev_project/mock.py @@ -373,7 +373,7 @@ REFINED_TASKS_JSON = { } CODE_PLAN_AND_CHANGE_SAMPLE = { - "Plan": '\n1. Plan for gui.py: Develop the GUI using Tkinter to replace the command-line interface. Start by setting up the main window and event handling. Then, add widgets for displaying the game status, results, and feedback. Implement interactive elements for difficulty selection and visualize the guess history. Finally, create animations for guess feedback and ensure responsiveness across different screen sizes.\n```python\nclass GUI:\n- pass\n+ def __init__(self):\n+ self.setup_window()\n+\n+ def setup_window(self):\n+ # Initialize the main window using Tkinter\n+ pass\n+\n+ def bind_events(self):\n+ # Bind button clicks and other events\n+ pass\n+\n+ def update_feedback(self, message: str):\n+ # Update the feedback label with the given message\n+ pass\n+\n+ def update_attempts(self, attempts: int):\n+ # Update the attempts label with the number of attempts\n+ pass\n+\n+ def update_history(self, history: list):\n+ # Update the history view with the list of past guesses\n+ pass\n+\n+ def show_difficulty_selector(self):\n+ # Show buttons or a dropdown for difficulty selection\n+ pass\n+\n+ def animate_guess_result(self, correct: bool):\n+ # Trigger an animation for correct or incorrect guesses\n+ pass\n```\n\n2. Plan for main.py: Modify the main.py to initialize the GUI and start the event-driven game loop. Ensure that the GUI is the primary interface for user interaction.\n```python\nclass Main:\n def main(self):\n- user_interface = UI()\n- user_interface.start()\n+ graphical_user_interface = GUI()\n+ graphical_user_interface.setup_window()\n+ graphical_user_interface.bind_events()\n+ # Start the Tkinter main loop\n+ pass\n\n if __name__ == "__main__":\n main_instance = Main()\n main_instance.main()\n```\n\n3. Plan for ui.py: Refactor ui.py to work with the new GUI class. Remove command-line interactions and delegate display and input tasks to the GUI.\n```python\nclass UI:\n- def display_message(self, message: str):\n- print(message)\n+\n+ def display_message(self, message: str):\n+ # This method will now pass the message to the GUI to display\n+ pass\n\n- def get_user_input(self, prompt: str) -> str:\n- return input(prompt)\n+\n+ def get_user_input(self, prompt: str) -> str:\n+ # This method will now trigger the GUI to get user input\n+ pass\n\n- def show_attempts(self, attempts: int):\n- print(f"Number of attempts: {attempts}")\n+\n+ def show_attempts(self, attempts: int):\n+ # This method will now update the GUI with the number of attempts\n+ pass\n\n- def show_history(self, history: list):\n- print("Guess history:")\n- for guess in history:\n- print(guess)\n+\n+ def show_history(self, history: list):\n+ # This method will now update the GUI with the guess history\n+ pass\n```\n\n4. Plan for game.py: Ensure game.py remains mostly unchanged as it contains the core game logic. However, make minor adjustments if necessary to integrate with the new GUI.\n```python\nclass Game:\n # No changes required for now\n```\n' + "Code Plan And Change": '\n1. Plan for gui.py: Develop the GUI using Tkinter to replace the command-line interface. Start by setting up the main window and event handling. Then, add widgets for displaying the game status, results, and feedback. Implement interactive elements for difficulty selection and visualize the guess history. Finally, create animations for guess feedback and ensure responsiveness across different screen sizes.\n```python\nclass GUI:\n- pass\n+ def __init__(self):\n+ self.setup_window()\n+\n+ def setup_window(self):\n+ # Initialize the main window using Tkinter\n+ pass\n+\n+ def bind_events(self):\n+ # Bind button clicks and other events\n+ pass\n+\n+ def update_feedback(self, message: str):\n+ # Update the feedback label with the given message\n+ pass\n+\n+ def update_attempts(self, attempts: int):\n+ # Update the attempts label with the number of attempts\n+ pass\n+\n+ def update_history(self, history: list):\n+ # Update the history view with the list of past guesses\n+ pass\n+\n+ def show_difficulty_selector(self):\n+ # Show buttons or a dropdown for difficulty selection\n+ pass\n+\n+ def animate_guess_result(self, correct: bool):\n+ # Trigger an animation for correct or incorrect guesses\n+ pass\n```\n\n2. Plan for main.py: Modify the main.py to initialize the GUI and start the event-driven game loop. Ensure that the GUI is the primary interface for user interaction.\n```python\nclass Main:\n def main(self):\n- user_interface = UI()\n- user_interface.start()\n+ graphical_user_interface = GUI()\n+ graphical_user_interface.setup_window()\n+ graphical_user_interface.bind_events()\n+ # Start the Tkinter main loop\n+ pass\n\n if __name__ == "__main__":\n main_instance = Main()\n main_instance.main()\n```\n\n3. Plan for ui.py: Refactor ui.py to work with the new GUI class. Remove command-line interactions and delegate display and input tasks to the GUI.\n```python\nclass UI:\n- def display_message(self, message: str):\n- print(message)\n+\n+ def display_message(self, message: str):\n+ # This method will now pass the message to the GUI to display\n+ pass\n\n- def get_user_input(self, prompt: str) -> str:\n- return input(prompt)\n+\n+ def get_user_input(self, prompt: str) -> str:\n+ # This method will now trigger the GUI to get user input\n+ pass\n\n- def show_attempts(self, attempts: int):\n- print(f"Number of attempts: {attempts}")\n+\n+ def show_attempts(self, attempts: int):\n+ # This method will now update the GUI with the number of attempts\n+ pass\n\n- def show_history(self, history: list):\n- print("Guess history:")\n- for guess in history:\n- print(guess)\n+\n+ def show_history(self, history: list):\n+ # This method will now update the GUI with the guess history\n+ pass\n```\n\n4. Plan for game.py: Ensure game.py remains mostly unchanged as it contains the core game logic. However, make minor adjustments if necessary to integrate with the new GUI.\n```python\nclass Game:\n # No changes required for now\n```\n' } REFINED_CODE_INPUT_SAMPLE = """ diff --git a/tests/metagpt/actions/test_write_code_plan_and_change_an.py b/tests/metagpt/actions/test_write_code_plan_and_change_an.py index 33114dfcf..383e791b8 100644 --- a/tests/metagpt/actions/test_write_code_plan_and_change_an.py +++ b/tests/metagpt/actions/test_write_code_plan_and_change_an.py @@ -14,7 +14,7 @@ from metagpt.actions.write_code_plan_and_change_an import ( REFINED_TEMPLATE, WriteCodePlanAndChange, ) -from metagpt.schema import CodePlanAndChangeContext, Document +from metagpt.schema import CodePlanAndChangeContext from tests.data.incremental_dev_project.mock import ( CODE_PLAN_AND_CHANGE_SAMPLE, DESIGN_SAMPLE, @@ -38,19 +38,19 @@ async def test_write_code_plan_and_change_an(mocker): root.instruct_content.model_dump = mock_code_plan_and_change mocker.patch("metagpt.actions.write_code_plan_and_change_an.WriteCodePlanAndChange.run", return_value=root) - requirement_doc = Document() - prd_docs = [Document()] - design_docs = [Document()] - tasks_docs = [Document()] + requirement = "New requirement" + prd_filename = "prd.md" + design_filename = "design.md" + task_filename = "task.md" code_plan_and_change_context = CodePlanAndChangeContext( - requirement_doc=requirement_doc, - prd_docs=prd_docs, - design_docs=design_docs, - tasks_docs=tasks_docs, + requirement=requirement, + prd_filename=prd_filename, + design_filename=design_filename, + task_filename=task_filename, ) - node = await WriteCodePlanAndChange(context=code_plan_and_change_context).run() + node = await WriteCodePlanAndChange(i_context=code_plan_and_change_context).run() - assert "Plan" in node.instruct_content.model_dump() + assert "Code Plan And Change" in node.instruct_content.model_dump() @pytest.mark.asyncio @@ -68,5 +68,4 @@ async def test_refine_code(mocker): summary_log="", ) code = await WriteCode().write_code(prompt=prompt) - assert code assert "def" in code diff --git a/tests/metagpt/actions/test_write_prd_an.py b/tests/metagpt/actions/test_write_prd_an.py index e8e347e5c..378ce42c3 100644 --- a/tests/metagpt/actions/test_write_prd_an.py +++ b/tests/metagpt/actions/test_write_prd_an.py @@ -38,7 +38,6 @@ async def test_write_prd_an(mocker): prompt = NEW_REQ_TEMPLATE.format( requirements=NEW_REQUIREMENT_SAMPLE, old_prd=PRD_SAMPLE, - project_name="", ) node = await REFINED_PRD_NODE.fill(prompt, llm) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 41ba785c4..7bc319ed2 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -108,7 +108,7 @@ def get_incremental_dev_result(idea, project_name, use_review=True): if project_path.exists(): raise Exception(f"Project {project_name} not exists") check_or_create_base_tag(project_path) - args = [idea, "--inc", "--project-path", project_path] + args = [idea, "--inc", "--project-path", project_path, "--n-round", "20"] if not use_review: args.append("--no-code-review") result = runner.invoke(app, args) From ff5e7deb215d0a5f4014808c12e527e74d7889b0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 24 Jan 2024 10:52:30 +0800 Subject: [PATCH 463/637] add strip for result. --- metagpt/tools/libs/web_scraping.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/libs/web_scraping.py b/metagpt/tools/libs/web_scraping.py index e8e73f123..921fca809 100644 --- a/metagpt/tools/libs/web_scraping.py +++ b/metagpt/tools/libs/web_scraping.py @@ -19,4 +19,4 @@ async def scrape_web_playwright(url, *urls): web = await PlaywrightWrapper("chromium").run(url, *urls) # Return the inner text content of the web page - return {"inner_text": web.inner_text, "html": web.html} + return {"inner_text": web.inner_text.strip(), "html": web.html.strip()} From dfe49a3312ae457e6e2de51de25fdfaf42a99418 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 24 Jan 2024 10:53:03 +0800 Subject: [PATCH 464/637] update return value. --- metagpt/roles/code_interpreter.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 3991862d1..b1526cd95 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -97,7 +97,7 @@ class CodeInterpreter(Role): if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions - return code["code"] if code["language"] != "markdown" else "", result, success + return code["code"] if code.get("language", None) != "markdown" else "", result, success async def _write_code(self): todo = WriteCodeByGenerate() if not self.use_tools else WriteCodeWithTools(selected_tools=self.tools) From 3f2b512d297e166b762c2af291096b9e3c21486f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 24 Jan 2024 10:53:37 +0800 Subject: [PATCH 465/637] new file: tests/metagpt/tools/libs/test_web_scraping.py --- tests/metagpt/tools/libs/test_web_scraping.py | 23 +++++++++++++++++++ 1 file changed, 23 insertions(+) create mode 100644 tests/metagpt/tools/libs/test_web_scraping.py diff --git a/tests/metagpt/tools/libs/test_web_scraping.py b/tests/metagpt/tools/libs/test_web_scraping.py new file mode 100644 index 000000000..c11960e68 --- /dev/null +++ b/tests/metagpt/tools/libs/test_web_scraping.py @@ -0,0 +1,23 @@ +import pytest + +from metagpt.tools.libs.web_scraping import scrape_web_playwright + + +@pytest.mark.asyncio +async def test_scrape_web_playwright(): + test_url = "https://www.deepwisdom.ai" + + result = await scrape_web_playwright(test_url) + + # Assert that the result is a dictionary + assert isinstance(result, dict) + + # Assert that the result contains 'inner_text' and 'html' keys + assert "inner_text" in result + assert "html" in result + + # Assert startswith and endswith + assert not result["inner_text"].startswith(" ") + assert not result["inner_text"].endswith(" ") + assert not result["html"].startswith(" ") + assert not result["html"].endswith(" ") From e42dc522c2b2dd88c1fb7247fd893d54e23cfb85 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 24 Jan 2024 10:59:32 +0800 Subject: [PATCH 466/637] Fix bug in WriteCode --- metagpt/actions/write_code.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index ef6d1708d..a7f53badf 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -114,7 +114,7 @@ class WriteCode(Action): code_context = coding_context.code_doc.content elif code_plan_and_change: code_context = await self.get_codes( - coding_context.task_doc, exclude=self.context.filename, project_repo=self.repo, mode="incremental" + coding_context.task_doc, exclude=self.i_context.filename, project_repo=self.repo, mode="incremental" ) else: code_context = await self.get_codes( @@ -128,17 +128,17 @@ class WriteCode(Action): user_requirement=requirement_doc.content if requirement_doc else "", code_plan_and_change=code_plan_and_change, design=coding_context.design_doc.content if coding_context.design_doc else "", - tasks=coding_context.task_doc.content if coding_context.task_doc else "", + task=coding_context.task_doc.content if coding_context.task_doc else "", code=code_context, logs=logs, feedback=bug_feedback.content if bug_feedback else "", - filename=self.context.filename, + filename=self.i_context.filename, summary_log=summary_doc.content if summary_doc else "", ) else: prompt = PROMPT_TEMPLATE.format( design=coding_context.design_doc.content if coding_context.design_doc else "", - tasks=coding_context.task_doc.content if coding_context.task_doc else "", + task=coding_context.task_doc.content if coding_context.task_doc else "", code=code_context, logs=logs, feedback=bug_feedback.content if bug_feedback else "", From 0c8a844f5a2e4a7f2e93584bb64ad3bedae64c97 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 24 Jan 2024 12:06:07 +0800 Subject: [PATCH 467/637] add strip for result. --- metagpt/actions/execute_code.py | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_code.py index 6591f479f..6a4a9abb8 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_code.py @@ -123,7 +123,10 @@ class ExecutePyCode(ExecuteCode, Action): return parsed_output for i, output in enumerate(outputs): - if output["output_type"] == "stream": + if output["output_type"] == "stream" and not any( + tag in output["text"] + for tag in ["| INFO | metagpt", "| ERROR | metagpt", "| WARNING | metagpt"] + ): parsed_output += output["text"] elif output["output_type"] == "display_data": if "image/png" in output["data"]: From 0353f36f0d9c04452a421e702ee984fbabc973c3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 24 Jan 2024 15:21:32 +0800 Subject: [PATCH 468/637] new file: examples/crawle_webpage.py --- examples/crawle_webpage.py | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) create mode 100644 examples/crawle_webpage.py diff --git a/examples/crawle_webpage.py b/examples/crawle_webpage.py new file mode 100644 index 000000000..2c616035f --- /dev/null +++ b/examples/crawle_webpage.py @@ -0,0 +1,22 @@ +# -*- encoding: utf-8 -*- +""" +@Date : 2024/01/24 15:11:27 +@Author : orange-crow +@File : crawle_webpage.py +""" + +from metagpt.roles.code_interpreter import CodeInterpreter + + +async def main(): + prompt = """Get data from `paperlist` table in https://papercopilot.com/statistics/iclr-statistics/iclr-2024-statistics/, + and save it to a csv file. paper title must include `multiagent` or `large language model`. *notice: print key data*""" + ci = CodeInterpreter(goal=prompt, use_tools=True) + + await ci.run(prompt) + + +if __name__ == "__main__": + import asyncio + + asyncio.run(main()) From 094d7c26df35907f99fdaaf79a9f3260f2e92f15 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Wed, 24 Jan 2024 16:30:55 +0800 Subject: [PATCH 469/637] fixbug: The startup parameters of the program have been lost. --- metagpt/roles/engineer.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 7c91ec6f9..0a50bb26d 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -292,7 +292,7 @@ class Engineer(Role): summarizations[ctx].append(filename) for ctx, filenames in summarizations.items(): ctx.codes_filenames = filenames - self.summarize_todos.append(SummarizeCode(i_context=ctx, llm=self.llm)) + self.summarize_todos.append(SummarizeCode(i_context=ctx, context=self.context, llm=self.llm)) if self.summarize_todos: self.set_todo(self.summarize_todos[0]) From 3450c240c96c7162b232698e3f46c3fc1aae644b Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 24 Jan 2024 16:44:10 +0800 Subject: [PATCH 470/637] remove mode of get_codes function --- metagpt/actions/write_code.py | 26 +++++++++----------------- metagpt/actions/write_code_review.py | 14 ++++++-------- metagpt/roles/engineer.py | 6 +++--- 3 files changed, 18 insertions(+), 28 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index a7f53badf..f0eb699bc 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -16,7 +16,6 @@ """ import json -from typing import Literal from pydantic import Field from tenacity import retry, stop_after_attempt, wait_random_exponential @@ -114,7 +113,7 @@ class WriteCode(Action): code_context = coding_context.code_doc.content elif code_plan_and_change: code_context = await self.get_codes( - coding_context.task_doc, exclude=self.i_context.filename, project_repo=self.repo, mode="incremental" + coding_context.task_doc, exclude=self.i_context.filename, project_repo=self.repo, use_inc=True ) else: code_context = await self.get_codes( @@ -155,39 +154,31 @@ class WriteCode(Action): return coding_context @staticmethod - async def get_codes( - task_doc: Document, exclude: str, project_repo: ProjectRepo, mode: Literal["normal", "incremental"] = "normal" - ) -> str: + async def get_codes(task_doc: Document, exclude: str, project_repo: ProjectRepo, use_inc: bool = False) -> str: """ - Get code snippets based on different modes. + Get code snippets that meet the requirements in various scenarios. Attributes: task_doc (Document): Document object of the task file. exclude (str): Specifies the filename to be excluded from the code snippets. project_repo (ProjectRepo): ProjectRepo object of the project. - mode (str): Specifies the mode, either "normal" or "incremental" (default is "normal"). + use_inc (bool): Specifies whether is incremental development. Returns: str: Code snippets. - - Description: - If mode is set to "normal", it returns code snippets for the regular coding phase, - i.e., all the code generated before writing the current file. - - If mode is set to "incremental", it returns code snippets for generating the code plan and change, - building upon the existing code in the "normal" mode and adding code for the current file's older versions. """ if not task_doc: return "" if not task_doc.content: task_doc = project_repo.docs.task.get(filename=task_doc.filename) m = json.loads(task_doc.content) - code_filenames = m.get(TASK_LIST.key, []) if mode == "normal" else m.get(REFINED_TASK_LIST.key, []) + code_filenames = m.get(TASK_LIST.key, []) if use_inc else m.get(REFINED_TASK_LIST.key, []) codes = [] src_file_repo = project_repo.srcs - if mode == "incremental": + if use_inc: src_files = src_file_repo.all_files + # Get the old workspace that are created by the previous WriteCodePlanAndChange action old_file_repo = project_repo.git_repo.new_file_repository(relative_path=project_repo.old_workspace) old_files = old_file_repo.all_files # Get the union of the files in the src and old workspaces @@ -213,7 +204,7 @@ class WriteCode(Action): continue codes.append(f"----- {filename}\n```{doc.content}```") - elif mode == "normal": + else: for filename in code_filenames: # Exclude the current file to get the context code snippets for generating the current file if filename == exclude: @@ -222,4 +213,5 @@ class WriteCode(Action): if not doc: continue codes.append(f"----- {filename}\n```{doc.content}```") + return "\n".join(codes) diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index b320be6ee..b8c350f11 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -137,11 +137,8 @@ class WriteCodeReview(Action): async def run(self, *args, **kwargs) -> CodingContext: iterative_code = self.i_context.code_doc.content - # k = self.context.config.code_review_k_times or 1 - k = 1 - code_plan_and_change_doc = await self.repo.get(filename=CODE_PLAN_AND_CHANGE_FILENAME) - code_plan_and_change = code_plan_and_change_doc.content if code_plan_and_change_doc else "" - mode = "incremental" if code_plan_and_change else "normal" + k = self.context.config.code_review_k_times or 1 + for i in range(k): format_example = FORMAT_EXAMPLE.format(filename=self.i_context.code_doc.filename) task_content = self.i_context.task_doc.content if self.i_context.task_doc else "" @@ -149,10 +146,10 @@ class WriteCodeReview(Action): self.i_context.task_doc, exclude=self.i_context.filename, project_repo=self.repo.with_src_path(self.context.src_workspace), - mode=mode, + use_inc=self.config.inc, ) - if not code_plan_and_change: + if not self.config.inc: context = "\n".join( [ "## System Design\n" + str(self.i_context.design_doc) + "\n", @@ -162,10 +159,11 @@ class WriteCodeReview(Action): ) else: requirement_doc = await self.repo.docs.get(filename=REQUIREMENT_FILENAME) + code_plan_and_change_doc = await self.repo.get(filename=CODE_PLAN_AND_CHANGE_FILENAME) context = "\n".join( [ "## User New Requirements\n" + str(requirement_doc) + "\n", - "## Code Plan And Change\n" + code_plan_and_change + "\n", + "## Code Plan And Change\n" + str(code_plan_and_change_doc) + "\n", "## System Design\n" + str(self.i_context.design_doc) + "\n", "## Tasks\n" + task_content + "\n", "## Code Files\n" + code_context + "\n", diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 8635e5fcf..3475a95fd 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -23,7 +23,7 @@ import json import os from collections import defaultdict from pathlib import Path -from typing import Literal, Set +from typing import Set from metagpt.actions import Action, WriteCode, WriteCodeReview, WriteTasks from metagpt.actions.fix_bug import FixBug @@ -100,7 +100,7 @@ class Engineer(Role): m = json.loads(task_msg.content) return m.get(TASK_LIST.key) or m.get(REFINED_TASK_LIST.key) - async def _act_sp_with_cr(self, review=False, mode: Literal["normal", "incremental"] = "normal") -> Set[str]: + async def _act_sp_with_cr(self, review=False) -> Set[str]: changed_files = set() for todo in self.code_todos: """ @@ -118,7 +118,7 @@ class Engineer(Role): coding_context = await action.run() dependencies = {coding_context.design_doc.root_relative_path, coding_context.task_doc.root_relative_path} - if mode == "incremental": + if self.config.inc: dependencies.add(os.path.join(CODE_PLAN_AND_CHANGE_FILE_REPO, CODE_PLAN_AND_CHANGE_FILENAME)) await self.project_repo.srcs.save( filename=coding_context.filename, From db8ae71afacc93c77ebe35408eee750dca620d4e Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 24 Jan 2024 17:17:40 +0800 Subject: [PATCH 471/637] update comment in get_codes function --- metagpt/actions/write_code.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index f0eb699bc..05fffd4fc 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -156,16 +156,16 @@ class WriteCode(Action): @staticmethod async def get_codes(task_doc: Document, exclude: str, project_repo: ProjectRepo, use_inc: bool = False) -> str: """ - Get code snippets that meet the requirements in various scenarios. + Get codes for generating the current file in various scenarios. Attributes: task_doc (Document): Document object of the task file. - exclude (str): Specifies the filename to be excluded from the code snippets. + exclude (str): The file to be generated. Specifies the filename to be excluded from the code snippets. project_repo (ProjectRepo): ProjectRepo object of the project. - use_inc (bool): Specifies whether is incremental development. + use_inc (bool): Whether is the incremental development scenario. Defaults to False. Returns: - str: Code snippets. + str: Code snippets for generating the current file. """ if not task_doc: return "" From 92384d77dd189cbe498f9d905dc49470d7fbf128 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 24 Jan 2024 17:17:40 +0800 Subject: [PATCH 472/637] update comment in get_codes function --- metagpt/actions/write_code.py | 22 ++++++++++++---------- 1 file changed, 12 insertions(+), 10 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index f0eb699bc..750e0cfa5 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -156,16 +156,16 @@ class WriteCode(Action): @staticmethod async def get_codes(task_doc: Document, exclude: str, project_repo: ProjectRepo, use_inc: bool = False) -> str: """ - Get code snippets that meet the requirements in various scenarios. + Get codes for generating the exclude file in various scenarios. Attributes: task_doc (Document): Document object of the task file. - exclude (str): Specifies the filename to be excluded from the code snippets. + exclude (str): The file to be generated. Specifies the filename to be excluded from the code snippets. project_repo (ProjectRepo): ProjectRepo object of the project. - use_inc (bool): Specifies whether is incremental development. + use_inc (bool): Whether is the incremental development scenario. Defaults to False. Returns: - str: Code snippets. + str: Codes for generating the exclude file. """ if not task_doc: return "" @@ -176,27 +176,28 @@ class WriteCode(Action): codes = [] src_file_repo = project_repo.srcs + # Incremental development scenario if use_inc: src_files = src_file_repo.all_files - # Get the old workspace that are created by the previous WriteCodePlanAndChange action + # Get the old workspace contained the old codes and old workspace are created in previous CodePlanAndChange old_file_repo = project_repo.git_repo.new_file_repository(relative_path=project_repo.old_workspace) old_files = old_file_repo.all_files # Get the union of the files in the src and old workspaces union_files_list = list(set(src_files) | set(old_files)) for filename in union_files_list: - # Exclude the current file from the all code snippets to get the context code snippets for generating + # Exclude the current file from the all code snippets if filename == exclude: - # If the file is in the old workspace, use the legacy code + # If the file is in the old workspace, use the old code # Exclude unnecessary code to maintain a clean and focused main.py file, ensuring only relevant and # essential functionality is included for the project’s requirements if filename in old_files and filename != "main.py": - # Use legacy code + # Use old code doc = await old_file_repo.get(filename=filename) # If the file is in the src workspace, skip it else: continue codes.insert(0, f"-----Now, {filename} to be rewritten\n```{doc.content}```\n=====") - # The context code snippets are generated from the src workspace + # The code snippets are generated from the src workspace else: doc = await src_file_repo.get(filename=filename) # If the file does not exist in the src workspace, skip it @@ -204,9 +205,10 @@ class WriteCode(Action): continue codes.append(f"----- {filename}\n```{doc.content}```") + # Normal scenario else: for filename in code_filenames: - # Exclude the current file to get the context code snippets for generating the current file + # Exclude the current file to get the code snippets for generating the current file if filename == exclude: continue doc = await src_file_repo.get(filename=filename) From 48cb6c6a78a44ff3bd308d838cd09623e43535c6 Mon Sep 17 00:00:00 2001 From: voidking Date: Wed, 24 Jan 2024 17:42:11 +0800 Subject: [PATCH 473/637] feat: support config2.yaml --- .github/workflows/unittest.yaml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/workflows/unittest.yaml b/.github/workflows/unittest.yaml index fd56c42fb..dbad0441f 100644 --- a/.github/workflows/unittest.yaml +++ b/.github/workflows/unittest.yaml @@ -50,6 +50,7 @@ jobs: run: | export ALLOW_OPENAI_API_CALL=0 echo "${{ secrets.METAGPT_KEY_YAML }}" | base64 -d > config/key.yaml + echo "${{ secrets.METAGPT_CONFIG2_YAML }}" | base64 -d > ~/.metagpt/config2.yaml pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt - name: Show coverage report run: | From 5af94ae1331d5c4e0cbaeb4292f85702e86dee23 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 24 Jan 2024 17:51:17 +0800 Subject: [PATCH 474/637] update comment in get_codes function --- metagpt/actions/write_code.py | 2 +- metagpt/roles/engineer.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 750e0cfa5..fcadd5f72 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -162,7 +162,7 @@ class WriteCode(Action): task_doc (Document): Document object of the task file. exclude (str): The file to be generated. Specifies the filename to be excluded from the code snippets. project_repo (ProjectRepo): ProjectRepo object of the project. - use_inc (bool): Whether is the incremental development scenario. Defaults to False. + use_inc (bool): Indicates whether the scenario involves incremental development. Defaults to False. Returns: str: Codes for generating the exclude file. diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index 3475a95fd..ee1a019bd 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -342,7 +342,7 @@ class Engineer(Role): summarizations[ctx].append(filename) for ctx, filenames in summarizations.items(): ctx.codes_filenames = filenames - self.summarize_todos.append(SummarizeCode(i_context=ctx, llm=self.llm)) + self.summarize_todos.append(SummarizeCode(i_context=ctx, context=self.context, llm=self.llm)) if self.summarize_todos: self.set_todo(self.summarize_todos[0]) From 448215613d994143c7cee65cb7acf67fd5663ad4 Mon Sep 17 00:00:00 2001 From: voidking Date: Wed, 24 Jan 2024 18:00:54 +0800 Subject: [PATCH 475/637] feat: support config2.yaml --- .github/workflows/unittest.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unittest.yaml b/.github/workflows/unittest.yaml index dbad0441f..87ccbf144 100644 --- a/.github/workflows/unittest.yaml +++ b/.github/workflows/unittest.yaml @@ -50,7 +50,7 @@ jobs: run: | export ALLOW_OPENAI_API_CALL=0 echo "${{ secrets.METAGPT_KEY_YAML }}" | base64 -d > config/key.yaml - echo "${{ secrets.METAGPT_CONFIG2_YAML }}" | base64 -d > ~/.metagpt/config2.yaml + mkdir -p ~/.metagpt && echo "${{ secrets.METAGPT_CONFIG2_YAML }}" | base64 -d > ~/.metagpt/config2.yaml pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt - name: Show coverage report run: | From 02b4608f8408f255f7a114c4f326ec7dae7bb912 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 24 Jan 2024 18:04:26 +0800 Subject: [PATCH 476/637] changed {tasks} to {task} in PROMPT_TEMPLATE --- metagpt/actions/summarize_code.py | 2 +- metagpt/actions/write_code.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/summarize_code.py b/metagpt/actions/summarize_code.py index 2b5546546..644d5d6a9 100644 --- a/metagpt/actions/summarize_code.py +++ b/metagpt/actions/summarize_code.py @@ -28,7 +28,7 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc ----- # Tasks ```text -{tasks} +{task} ``` ----- {code_blocks} diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index fcadd5f72..7f5dae414 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -44,7 +44,7 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc {design} ## Tasks -{tasks} +{task} ## Legacy Code ```Code From 7e328e543197b6ae07ea643429bd4bf70fffa96d Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 24 Jan 2024 18:05:26 +0800 Subject: [PATCH 477/637] changed {tasks} to {task} in PROMPT_TEMPLATE --- metagpt/actions/summarize_code.py | 2 +- metagpt/actions/write_code.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/summarize_code.py b/metagpt/actions/summarize_code.py index 644d5d6a9..0f6e0e69d 100644 --- a/metagpt/actions/summarize_code.py +++ b/metagpt/actions/summarize_code.py @@ -26,7 +26,7 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc {system_design} ``` ----- -# Tasks +# Task ```text {task} ``` diff --git a/metagpt/actions/write_code.py b/metagpt/actions/write_code.py index 7f5dae414..0b86ac1bb 100644 --- a/metagpt/actions/write_code.py +++ b/metagpt/actions/write_code.py @@ -43,7 +43,7 @@ ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenc ## Design {design} -## Tasks +## Task {task} ## Legacy Code From 431e53617e0d5803a336ac7f641d521b27357326 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Wed, 24 Jan 2024 18:16:47 +0800 Subject: [PATCH 478/637] changed tasks to task --- metagpt/actions/project_management.py | 4 ++-- metagpt/actions/summarize_code.py | 2 +- metagpt/actions/write_code_review.py | 4 ++-- tests/metagpt/actions/test_write_code_plan_and_change_an.py | 2 +- 4 files changed, 6 insertions(+), 6 deletions(-) diff --git a/metagpt/actions/project_management.py b/metagpt/actions/project_management.py index d417bf538..67a614d6f 100644 --- a/metagpt/actions/project_management.py +++ b/metagpt/actions/project_management.py @@ -22,7 +22,7 @@ from metagpt.schema import Document, Documents NEW_REQ_TEMPLATE = """ ### Legacy Content -{old_tasks} +{old_task} ### New Requirements {context} @@ -77,7 +77,7 @@ class WriteTasks(Action): return node async def _merge(self, system_design_doc, task_doc) -> Document: - context = NEW_REQ_TEMPLATE.format(context=system_design_doc.content, old_tasks=task_doc.content) + context = NEW_REQ_TEMPLATE.format(context=system_design_doc.content, old_task=task_doc.content) node = await REFINED_PM_NODE.fill(context, self.llm, schema=self.prompt_schema) task_doc.content = node.instruct_content.model_dump_json() return task_doc diff --git a/metagpt/actions/summarize_code.py b/metagpt/actions/summarize_code.py index 0f6e0e69d..d21b62f83 100644 --- a/metagpt/actions/summarize_code.py +++ b/metagpt/actions/summarize_code.py @@ -110,7 +110,7 @@ class SummarizeCode(Action): format_example = FORMAT_EXAMPLE prompt = PROMPT_TEMPLATE.format( system_design=design_doc.content, - tasks=task_doc.content, + task=task_doc.content, code_blocks="\n".join(code_blocks), format_example=format_example, ) diff --git a/metagpt/actions/write_code_review.py b/metagpt/actions/write_code_review.py index b8c350f11..da636eb36 100644 --- a/metagpt/actions/write_code_review.py +++ b/metagpt/actions/write_code_review.py @@ -153,7 +153,7 @@ class WriteCodeReview(Action): context = "\n".join( [ "## System Design\n" + str(self.i_context.design_doc) + "\n", - "## Tasks\n" + task_content + "\n", + "## Task\n" + task_content + "\n", "## Code Files\n" + code_context + "\n", ] ) @@ -165,7 +165,7 @@ class WriteCodeReview(Action): "## User New Requirements\n" + str(requirement_doc) + "\n", "## Code Plan And Change\n" + str(code_plan_and_change_doc) + "\n", "## System Design\n" + str(self.i_context.design_doc) + "\n", - "## Tasks\n" + task_content + "\n", + "## Task\n" + task_content + "\n", "## Code Files\n" + code_context + "\n", ] ) diff --git a/tests/metagpt/actions/test_write_code_plan_and_change_an.py b/tests/metagpt/actions/test_write_code_plan_and_change_an.py index 383e791b8..741773e2b 100644 --- a/tests/metagpt/actions/test_write_code_plan_and_change_an.py +++ b/tests/metagpt/actions/test_write_code_plan_and_change_an.py @@ -60,7 +60,7 @@ async def test_refine_code(mocker): user_requirement=NEW_REQUIREMENT_SAMPLE, code_plan_and_change=CODE_PLAN_AND_CHANGE_SAMPLE, design=DESIGN_SAMPLE, - tasks=TASKS_SAMPLE, + task=TASKS_SAMPLE, code=REFINED_CODE_INPUT_SAMPLE, logs="", feedback="", From d51d262238f9327819c06a402e0579e33c986093 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Wed, 24 Jan 2024 20:02:07 +0800 Subject: [PATCH 479/637] fixbug: module 'mock' has no attribute 'patch' fixbug: unit test --- setup.py | 1 - .../actions/test_project_management_an.py | 2 +- .../actions/test_rebuild_sequence_view.py | 1 + .../test_write_code_plan_and_change_an.py | 2 +- tests/metagpt/utils/test_redis.py | 24 ++++++++++--------- tests/metagpt/utils/test_s3.py | 17 ++++++------- 6 files changed, 23 insertions(+), 24 deletions(-) diff --git a/setup.py b/setup.py index 1ba08c636..d1445e3f8 100644 --- a/setup.py +++ b/setup.py @@ -46,7 +46,6 @@ extras_require["test"] = [ "chromadb==0.4.14", "gradio==3.0.0", "grpcio-status==1.48.2", - "mock==5.1.0", "pylint==3.0.3", "pybrowsers", ] diff --git a/tests/metagpt/actions/test_project_management_an.py b/tests/metagpt/actions/test_project_management_an.py index aa759aec8..ddbb56569 100644 --- a/tests/metagpt/actions/test_project_management_an.py +++ b/tests/metagpt/actions/test_project_management_an.py @@ -37,7 +37,7 @@ async def test_project_management_an(mocker): root.instruct_content.model_dump = mock_refined_tasks_json mocker.patch("metagpt.actions.project_management_an.REFINED_PM_NODE.fill", return_value=root) - prompt = NEW_REQ_TEMPLATE.format(old_tasks=TASKS_SAMPLE, context=dict_to_markdown(REFINED_DESIGN_JSON)) + prompt = NEW_REQ_TEMPLATE.format(old_task=TASKS_SAMPLE, context=dict_to_markdown(REFINED_DESIGN_JSON)) node = await REFINED_PM_NODE.fill(prompt, llm) assert "Refined Logic Analysis" in node.instruct_content.model_dump() diff --git a/tests/metagpt/actions/test_rebuild_sequence_view.py b/tests/metagpt/actions/test_rebuild_sequence_view.py index 49d444f2f..e0e65c3cc 100644 --- a/tests/metagpt/actions/test_rebuild_sequence_view.py +++ b/tests/metagpt/actions/test_rebuild_sequence_view.py @@ -17,6 +17,7 @@ from metagpt.utils.git_repository import ChangeType @pytest.mark.asyncio +@pytest.mark.skip async def test_rebuild(context): # Mock data = await aread(filename=Path(__file__).parent / "../../data/graph_db/networkx.json") diff --git a/tests/metagpt/actions/test_write_code_plan_and_change_an.py b/tests/metagpt/actions/test_write_code_plan_and_change_an.py index 741773e2b..9cd51398f 100644 --- a/tests/metagpt/actions/test_write_code_plan_and_change_an.py +++ b/tests/metagpt/actions/test_write_code_plan_and_change_an.py @@ -55,7 +55,7 @@ async def test_write_code_plan_and_change_an(mocker): @pytest.mark.asyncio async def test_refine_code(mocker): - mocker.patch("metagpt.actions.write_code.WriteCodePlanAndChange.write_code", return_value=REFINED_CODE_SAMPLE) + mocker.patch.object(WriteCode, "_aask", return_value=REFINED_CODE_SAMPLE) prompt = REFINED_TEMPLATE.format( user_requirement=NEW_REQUIREMENT_SAMPLE, code_plan_and_change=CODE_PLAN_AND_CHANGE_SAMPLE, diff --git a/tests/metagpt/utils/test_redis.py b/tests/metagpt/utils/test_redis.py index 748c44f54..6fd4250a6 100644 --- a/tests/metagpt/utils/test_redis.py +++ b/tests/metagpt/utils/test_redis.py @@ -9,23 +9,25 @@ from unittest.mock import AsyncMock import pytest -from metagpt.config2 import Config from metagpt.utils.redis import Redis -async def async_mock_from_url(*args, **kwargs): - mock_client = AsyncMock() - mock_client.set.return_value = None - mock_client.get.side_effect = [b"test", b""] - return mock_client - - @pytest.mark.asyncio async def test_redis(mocker): - redis = Config.default().redis - mocker.patch("aioredis.from_url", return_value=async_mock_from_url()) + async def async_mock_from_url(*args, **kwargs): + mock_client = AsyncMock() + mock_client.set.return_value = None + mock_client.get.return_value = b"test" + return mock_client - conn = Redis(redis) + mocker.patch("aioredis.from_url", return_value=async_mock_from_url()) + mock_config = mocker.Mock() + mock_config.to_url.return_value = "http://mock.com" + mock_config.username = "mockusername" + mock_config.password = "mockpwd" + mock_config.db = "0" + + conn = Redis(mock_config) await conn.set("test", "test", timeout_sec=0) assert await conn.get("test") == b"test" await conn.close() diff --git a/tests/metagpt/utils/test_s3.py b/tests/metagpt/utils/test_s3.py index 4dc3b1e42..b26ebe94d 100644 --- a/tests/metagpt/utils/test_s3.py +++ b/tests/metagpt/utils/test_s3.py @@ -8,8 +8,8 @@ import uuid from pathlib import Path +import aioboto3 import aiofiles -import mock import pytest from metagpt.config2 import Config @@ -18,21 +18,18 @@ from metagpt.utils.s3 import S3 @pytest.mark.asyncio -@mock.patch("aioboto3.Session") -async def test_s3(mock_session_class): +async def test_s3(mocker): # Set up the mock response data = await aread(__file__, "utf-8") - mock_session_object = mock.Mock() - reader_mock = mock.AsyncMock() + reader_mock = mocker.AsyncMock() reader_mock.read.side_effect = [data.encode("utf-8"), b"", data.encode("utf-8")] - type(reader_mock).url = mock.PropertyMock(return_value="https://mock") - mock_client = mock.AsyncMock() + type(reader_mock).url = mocker.PropertyMock(return_value="https://mock") + mock_client = mocker.AsyncMock() mock_client.put_object.return_value = None mock_client.get_object.return_value = {"Body": reader_mock} mock_client.__aenter__.return_value = mock_client mock_client.__aexit__.return_value = None - mock_session_object.client.return_value = mock_client - mock_session_class.return_value = mock_session_object + mocker.patch.object(aioboto3.Session, "client", return_value=mock_client) # Prerequisites s3 = Config.default().s3 @@ -55,7 +52,7 @@ async def test_s3(mock_session_class): # Mock session env s3.access_key = "ABC" - type(reader_mock).url = mock.PropertyMock(return_value="") + type(reader_mock).url = mocker.PropertyMock(return_value="") try: conn = S3(s3) res = await conn.cache("ABC", ".bak", "script") From f9f6dbefa8bf94be64c1c5b917402f5de939453f Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 25 Jan 2024 10:33:46 +0800 Subject: [PATCH 480/637] Add the tar command to extract the .zip file --- tests/metagpt/test_incremental_dev.py | 26 +++++++++++++++++++------- 1 file changed, 19 insertions(+), 7 deletions(-) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 7bc319ed2..1baa82a72 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -105,8 +105,16 @@ def log_and_check_result(result, tag_name="refine"): def get_incremental_dev_result(idea, project_name, use_review=True): project_path = TEST_DATA_PATH / "incremental_dev_project" / project_name - if project_path.exists(): - raise Exception(f"Project {project_name} not exists") + # Check if the project path exists + if not project_path.exists(): + # If the project does not exist, extract the project file + try: + # Use the tar command to extract the .zip file + subprocess.run(["tar", "-xf", f"{project_path}.zip", "-C", str(project_path.parent)], check=True) + except subprocess.CalledProcessError as e: + # If the extraction fails, throw an exception + raise Exception(f"Failed to unzip project {project_name}. Error: {e}") + check_or_create_base_tag(project_path) args = [idea, "--inc", "--project-path", project_path, "--n-round", "20"] if not use_review: @@ -146,14 +154,18 @@ def check_or_create_base_tag(project_path): logger.info("Base tag doesn't exist.") # Add and commit the current code if 'base' tag doesn't exist add_cmd = ["git", "add", "."] - commit_cmd = ["git", "commit", "-m", "Initial commit"] try: subprocess.run(add_cmd, check=True) + logger.info("Files added successfully.") + except subprocess.CalledProcessError as e: + logger.error(f"Failed to add files: {e}") + + commit_cmd = ["git", "commit", "-m", "Initial commit"] + try: subprocess.run(commit_cmd, check=True) - logger.info("Added and committed all files with the message 'Initial commit'.") - except Exception as e: - logger.error("Failed to add and commit all files.") - raise e + logger.info("Committed all files with the message 'Initial commit'.") + except subprocess.CalledProcessError as e: + logger.error(f"Failed to commit: {e.stderr}") # Add 'base' tag add_base_tag_cmd = ["git", "tag", "base"] From 0faae9368f18a27c690a92c652013578cb2dda3f Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 25 Jan 2024 11:52:14 +0800 Subject: [PATCH 481/637] Add the tar command to extract the .zip file --- tests/metagpt/test_incremental_dev.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 1baa82a72..6a26f9b83 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -113,7 +113,7 @@ def get_incremental_dev_result(idea, project_name, use_review=True): subprocess.run(["tar", "-xf", f"{project_path}.zip", "-C", str(project_path.parent)], check=True) except subprocess.CalledProcessError as e: # If the extraction fails, throw an exception - raise Exception(f"Failed to unzip project {project_name}. Error: {e}") + raise Exception(f"Failed to extract project {project_name}. Error: {e}") check_or_create_base_tag(project_path) args = [idea, "--inc", "--project-path", project_path, "--n-round", "20"] From f1a4197a8221ec17624b871d9eda695ff6a058ba Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 25 Jan 2024 14:04:14 +0800 Subject: [PATCH 482/637] rm make tools in ci for now --- metagpt/prompts/tool_types.py | 2 +- metagpt/roles/code_interpreter.py | 27 --------------------- tests/metagpt/roles/run_code_interpreter.py | 21 ++++------------ 3 files changed, 6 insertions(+), 44 deletions(-) diff --git a/metagpt/prompts/tool_types.py b/metagpt/prompts/tool_types.py index 42d9c1ece..381fb25ad 100644 --- a/metagpt/prompts/tool_types.py +++ b/metagpt/prompts/tool_types.py @@ -25,7 +25,7 @@ The current task is about feature engineering. when performing it, please adhere # Prompt for using tools of "model_train" type MODEL_TRAIN_PROMPT = """ The current task is about training a model, please ensure high performance: -- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as lightGBM, XGBoost, CatBoost, etc. +- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as XGBoost, CatBoost, etc. - If non-numeric columns exist, perform label encode together with all steps. - Use the data from previous task result directly, do not mock or reload data yourself. - Set suitable hyperparameters for the model, make metrics as high as possible. diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 11ede6068..8c7a4bc68 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -1,5 +1,3 @@ -from datetime import datetime - from pydantic import Field from metagpt.actions.ask_review import ReviewConst @@ -8,15 +6,12 @@ from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWi from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.logs import logger from metagpt.roles import Role -from metagpt.roles.tool_maker import ToolMaker from metagpt.schema import Message, Task, TaskResult -from metagpt.utils.save_code import save_code_file class CodeInterpreter(Role): auto_run: bool = True use_tools: bool = False - make_udfs: bool = False # whether to save user-defined functions use_code_steps: bool = False execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) tools: list[str] = [] @@ -47,19 +42,6 @@ class CodeInterpreter(Role): def working_memory(self): return self.rc.working_memory - async def _plan_and_act(self): - rsp = await super()._plan_and_act() - - # save code using datetime.now or keywords related to the goal of your project (plan.goal). - project_record = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") - save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") - - # make tools out of workable codes for future use - if self.make_udfs: - await self.make_tools() - - return rsp - async def _act_on_task(self, current_task: Task) -> TaskResult: code, result, is_success = await self._write_and_exec_code() task_result = TaskResult(code=code, result=result, is_success=is_success) @@ -108,12 +90,3 @@ class CodeInterpreter(Role): code = await todo.run(context=context, plan=self.planner.plan, temperature=0.0) return code, todo - - async def make_tools(self): - """Make user-defined functions(udfs, aka tools) for pure generation code.""" - logger.info("Plan completed. Now start to make tools ...") - tool_maker = ToolMaker() - for task in self.planner.plan.get_finished_tasks(): - await tool_maker.make_tool( - code=task.code, instruction=task.instruction, task_id=task.task_id, auto_run=self.auto_run - ) diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py index 766a25998..e41507256 100644 --- a/tests/metagpt/roles/run_code_interpreter.py +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -9,9 +9,7 @@ from metagpt.schema import Plan from metagpt.utils.recovery_util import load_history, save_history -async def run_code_interpreter( - role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir, tools -): +async def run_code_interpreter(role_class, requirement, auto_run, use_tools, use_code_steps, save_dir, tools): """ The main function to run the MLEngineer with optional history loading. @@ -25,16 +23,13 @@ async def run_code_interpreter( """ if role_class == "ci": - role = CodeInterpreter( - goal=requirement, auto_run=auto_run, use_tools=use_tools, make_udfs=make_udfs, tools=tools - ) + role = CodeInterpreter(goal=requirement, auto_run=auto_run, use_tools=use_tools, tools=tools) else: role = MLEngineer( goal=requirement, auto_run=auto_run, use_tools=use_tools, use_code_steps=use_code_steps, - make_udfs=make_udfs, tools=tools, ) @@ -50,10 +45,10 @@ async def run_code_interpreter( try: await role.run(requirement) except Exception as e: - save_path = save_history(role, save_dir) - logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") + save_history(role, save_dir) + if __name__ == "__main__": # requirement = "Run data analysis on sklearn Iris dataset, include a plot" @@ -73,8 +68,6 @@ if __name__ == "__main__": role_class = "mle" auto_run = True use_tools = True - make_udfs = False - use_udfs = False tools = [] # tools = ["FillMissingValue", "CatCross", "non_existing_test"] @@ -84,13 +77,9 @@ if __name__ == "__main__": auto_run: bool = auto_run, use_tools: bool = use_tools, use_code_steps: bool = False, - make_udfs: bool = make_udfs, - use_udfs: bool = use_udfs, save_dir: str = save_dir, tools=tools, ): - await run_code_interpreter( - role_class, requirement, auto_run, use_tools, use_code_steps, make_udfs, use_udfs, save_dir, tools - ) + await run_code_interpreter(role_class, requirement, auto_run, use_tools, use_code_steps, save_dir, tools) fire.Fire(main) From 526025bbe3e88d51b6bcc7cc97aa87f3549871fc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 25 Jan 2024 15:01:05 +0800 Subject: [PATCH 483/637] change file name: crawle_webpage.py -> crawl_webpage.py --- examples/{crawle_webpage.py => crawl_webpage.py} | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) rename examples/{crawle_webpage.py => crawl_webpage.py} (94%) diff --git a/examples/crawle_webpage.py b/examples/crawl_webpage.py similarity index 94% rename from examples/crawle_webpage.py rename to examples/crawl_webpage.py index 2c616035f..35413d2ff 100644 --- a/examples/crawle_webpage.py +++ b/examples/crawl_webpage.py @@ -2,7 +2,7 @@ """ @Date : 2024/01/24 15:11:27 @Author : orange-crow -@File : crawle_webpage.py +@File : crawl_webpage.py """ from metagpt.roles.code_interpreter import CodeInterpreter From 54a08747db5b16fb7680556b9051466cf121d3f9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Thu, 25 Jan 2024 15:02:52 +0800 Subject: [PATCH 484/637] chore --- metagpt/roles/code_interpreter.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index b1526cd95..7a4ced59b 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -97,7 +97,7 @@ class CodeInterpreter(Role): if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions - return code["code"] if code.get("language", None) != "markdown" else "", result, success + return code["code"] if code.get("language") != "markdown" else "", result, success async def _write_code(self): todo = WriteCodeByGenerate() if not self.use_tools else WriteCodeWithTools(selected_tools=self.tools) From 7aa89a3204645d4491916a86726c14abd01b3c4c Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 25 Jan 2024 15:19:48 +0800 Subject: [PATCH 485/637] minor update --- metagpt/roles/code_interpreter.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 026fec562..d1136a1d4 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -79,7 +79,11 @@ class CodeInterpreter(Role): if ReviewConst.CHANGE_WORD[0] in review: counter = 0 # redo the task again with help of human suggestions - return code["code"] if code.get("language") != "markdown" else "", result, success + py_code = ( + code["code"] if code.get("language") != "markdown" else "" + ) # use python code as final code; for markdown, return the rendered result instead of the code itself + + return py_code, result, success async def _write_code(self): todo = WriteCodeByGenerate() if not self.use_tools else WriteCodeWithTools(selected_tools=self.tools) From 51169d7a69f8ce48a34ac8674f6f310d04cd2acb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Thu, 25 Jan 2024 15:40:16 +0800 Subject: [PATCH 486/637] feat: + compatible with windows path --- metagpt/actions/design_api.py | 2 +- metagpt/actions/prepare_documents.py | 2 +- metagpt/roles/engineer.py | 2 +- metagpt/utils/dependency_file.py | 20 +++++++++++++------- metagpt/utils/file_repository.py | 19 +++++++++++++------ 5 files changed, 29 insertions(+), 16 deletions(-) diff --git a/metagpt/actions/design_api.py b/metagpt/actions/design_api.py index 4060d211c..cb6013538 100644 --- a/metagpt/actions/design_api.py +++ b/metagpt/actions/design_api.py @@ -46,7 +46,7 @@ class WriteDesign(Action): ) async def run(self, with_messages: Message, schema: str = None): - # Use `git status` to identify which PRD documents have been modified in the `docs/prds` directory. + # Use `git status` to identify which PRD documents have been modified in the `docs/prd` directory. changed_prds = self.repo.docs.prd.changed_files # Use `git status` to identify which design documents in the `docs/system_designs` directory have undergone # changes. diff --git a/metagpt/actions/prepare_documents.py b/metagpt/actions/prepare_documents.py index 84a4fc1d7..ab069dc11 100644 --- a/metagpt/actions/prepare_documents.py +++ b/metagpt/actions/prepare_documents.py @@ -48,5 +48,5 @@ class PrepareDocuments(Action): # Write the newly added requirements from the main parameter idea to `docs/requirement.txt`. doc = await self.repo.docs.save(filename=REQUIREMENT_FILENAME, content=with_messages[0].content) # Send a Message notification to the WritePRD action, instructing it to process requirements using - # `docs/requirement.txt` and `docs/prds/`. + # `docs/requirement.txt` and `docs/prd/`. return ActionOutput(content=doc.content, instruct_content=doc) diff --git a/metagpt/roles/engineer.py b/metagpt/roles/engineer.py index ee1a019bd..40ade2110 100644 --- a/metagpt/roles/engineer.py +++ b/metagpt/roles/engineer.py @@ -178,7 +178,7 @@ class Engineer(Role): is_pass, reason = await self._is_pass(summary) if not is_pass: todo.i_context.reason = reason - tasks.append(todo.i_context.dict()) + tasks.append(todo.i_context.model_dump()) await self.project_repo.docs.code_summary.save( filename=Path(todo.i_context.design_filename).name, diff --git a/metagpt/utils/dependency_file.py b/metagpt/utils/dependency_file.py index 7cf9a1d49..c8b3bc4a4 100644 --- a/metagpt/utils/dependency_file.py +++ b/metagpt/utils/dependency_file.py @@ -9,6 +9,7 @@ from __future__ import annotations import json +import re from pathlib import Path from typing import Set @@ -36,7 +37,9 @@ class DependencyFile: """Load dependencies from the file asynchronously.""" if not self._filename.exists(): return - self._dependencies = json.loads(await aread(self._filename)) + json_data = await aread(self._filename) + json_data = re.sub(r"\\+", "/", json_data) # Compatible with windows path + self._dependencies = json.loads(json_data) @handle_exception async def save(self): @@ -60,17 +63,20 @@ class DependencyFile: key = Path(filename).relative_to(root) except ValueError: key = filename - + skey = re.sub(r"\\+", "/", str(key)) # Compatible with windows path if dependencies: relative_paths = [] for i in dependencies: try: - relative_paths.append(str(Path(i).relative_to(root))) + s = str(Path(i).relative_to(root)) except ValueError: - relative_paths.append(str(i)) - self._dependencies[str(key)] = relative_paths - elif str(key) in self._dependencies: - del self._dependencies[str(key)] + s = str(i) + s = re.sub(r"\\+", "/", s) # Compatible with windows path + relative_paths.append(s) + + self._dependencies[skey] = relative_paths + elif skey in self._dependencies: + del self._dependencies[skey] if persist: await self.save() diff --git a/metagpt/utils/file_repository.py b/metagpt/utils/file_repository.py index 94d6fe76d..d2a06963a 100644 --- a/metagpt/utils/file_repository.py +++ b/metagpt/utils/file_repository.py @@ -101,21 +101,28 @@ class FileRepository: path_name = self.workdir / filename if not path_name.exists(): return None + if not path_name.is_file(): + return None doc.content = await aread(path_name) return doc - async def get_all(self) -> List[Document]: + async def get_all(self, filter_ignored=True) -> List[Document]: """Get the content of all files in the repository. :return: List of Document instances representing files. """ docs = [] - for root, dirs, files in os.walk(str(self.workdir)): - for file in files: - file_path = Path(root) / file - relative_path = file_path.relative_to(self.workdir) - doc = await self.get(relative_path) + if filter_ignored: + for f in self.all_files: + doc = await self.get(f) docs.append(doc) + else: + for root, dirs, files in os.walk(str(self.workdir)): + for file in files: + file_path = Path(root) / file + relative_path = file_path.relative_to(self.workdir) + doc = await self.get(relative_path) + docs.append(doc) return docs @property From 1f90bc58cc56485b0355243bba98398a8d9547b3 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 25 Jan 2024 16:25:23 +0800 Subject: [PATCH 487/637] 1. Update the code compression package 2. Modify the prompt of REFINED_TEMPLATE --- .../actions/write_code_plan_and_change_an.py | 4 ++-- tests/data/incremental_dev_project/Gomoku.zip | Bin 94089 -> 94000 bytes .../dice_simulator_new.zip | Bin 56384 -> 56393 bytes .../number_guessing_game.zip | Bin 53755 -> 53764 bytes .../incremental_dev_project/pygame_2048.zip | Bin 60632 -> 63452 bytes .../simple_add_calculator.zip | Bin 56538 -> 56546 bytes .../incremental_dev_project/snake_game.zip | Bin 119511 -> 108388 bytes .../incremental_dev_project/word_cloud.zip | Bin 12762 -> 12752 bytes 8 files changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/write_code_plan_and_change_an.py b/metagpt/actions/write_code_plan_and_change_an.py index 3fac22242..708808050 100644 --- a/metagpt/actions/write_code_plan_and_change_an.py +++ b/metagpt/actions/write_code_plan_and_change_an.py @@ -172,11 +172,11 @@ Role: You are a professional engineer; The main goal is to complete incremental 2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets. 3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import. 4. Follow design: YOU MUST FOLLOW "Data structures and interfaces". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design. -5. Follow Code Plan And Change: If there is any Incremental Change or Legacy Code files contain "{filename} to be rewritten", you must merge it into the code file according to the plan. +5. Follow Code Plan And Change: If there is any Incremental Change that is marked by the git diff format using '+' and '-' for add/modify/delete code, or Legacy Code files contain "{filename} to be rewritten", you must merge it into the code file according to the plan. 6. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE. 7. Before using a external variable/module, make sure you import it first. 8. Write out EVERY CODE DETAIL, DON'T LEAVE TODO. -9. Attention: Retain content that is not related to incremental development but important for consistency and clarity.". +9. Attention: Retain details that are not related to incremental development but are important for maintaining the consistency and clarity of the old code. """ WRITE_CODE_PLAN_AND_CHANGE_NODE = ActionNode.from_children("WriteCodePlanAndChange", [CODE_PLAN_AND_CHANGE]) diff --git a/tests/data/incremental_dev_project/Gomoku.zip b/tests/data/incremental_dev_project/Gomoku.zip index d6c6b8d16148047dadbc180ad3974b612c48dac5..23649565ab58c4b07ee08680cb20370d933c1b39 100644 GIT binary patch delta 3940 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 diff --git a/tests/data/incremental_dev_project/dice_simulator_new.zip b/tests/data/incremental_dev_project/dice_simulator_new.zip index 4b8d3f038addf84e74e0d1a15a5209d3a263e921..4752ab4c55876bdb8d17a6ebc34aa7e2cf006786 100644 GIT binary patch delta 3861 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 delta 3564 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 delta 2540 zcmY+F30M=?7RP5U5P>2D5FR^7NSKU-2neW!Kz*WRHG&lp6wrr0Ktv3+fJLl6>xb0m zasX>ZlvR+;5O$G8CF}^cw6!kPDyX$}se2!V-prs+^L=yj`~T0mXL4umz4=eE+W(5w zOr-v5>az%f&?Hv2@mA%@+e*{0`jdqABnt>E)Ia3oq9#;U!yg+RjZF7;2Q{6gYHR1g zkny-+!}y4@Z#5)EnM0<`6mB|e$e(R-)Kc{++?HzB1qtGu49+fM^mzKR=CS^7bL$el zVs73$C9i*6)Ht<~b0vG?&bq&L6=ttX=c(}w)&8lrp0|FzjOQ}T{x(rI9y*~j6dU*Q z(-VoomcoYI{+kyQj;-M6pLy+5zNh4Bc=ldZ zpx=`Rlk!K$#(Pg@aNHW3YMbm_-6i~b6BE1Q?Zjw)Nmyb2i~nZHCMRkZZadfTU@TLp z6`->3E@e}sv*YvgFe>o3Rp=ZaQSJXmC|}a# z>Eu2TIXf7K-l$dvc5nTCLt;>)hf$8vUk=L#Rv9phX!w9?5os%`oEKWF;Z>ygb$Q+l zP1SNLE*@J2-tmqwvqh*F3(GczED0Acx(rZ_sp-}vWgtv8C!OI7WWs7iOmdnQ>_n=x zhDLI6n59w>4|u`0BMTSYWF#iF>_3+byfZh*Qzyej2_qSy?Dav19{#}u?n zXw*9sOCE%ke29v4F$$m+9QXi z{Iy1m3oXE+T#O=}8H~l4$6`ad8`|%}xR2*)fZk~u_5EK)oOVQ~=1pgHV1Abw44u|R z_up9B;NhwXIu#Uha-FWyhKLF`WaS15uFQKD4nC!)o0!xViqHAch zMZ(-yUL!&Z7S+#$ptIhLn|?-wCRrTDuKFVE3L!{Np`|LchZAxsnG28Q!Q>SjDG)7Y zyY5=djK5VYLR~Bx`v_r;f<~KtV0JwpI(*cjRzV>-E7g`zAgrTN*b*k2R_B27SbT^x z&+BN?9c=1@$ut}(7+T8S#n2a`1NdOj;DA=LSmsN?{stQT=*MJk;EcIHg9-i=gf-IW ziAvf-CB}IH7z5`>{Yf;7Q2iGW%zwd1jppY94T4Zz{*0>%HuxFM-{_g_g52<9smsDUSKcFVBmRWjuqMb+OCQTr^E0_$&k%F479Qb#I4m|9#A_uW!3u2`h z>B(@hIc!%Nzz}mv>7unVNJ_PUUL}okS^Qb156=h8@cF2&qk3%w!Nu2gCaJz<>%U|W z-)+ftW-w)=XIi>(| z97kL59`rm&M*KcWM}gItjG=pwbew+#aGK7gcQ3i-ZkX+d5d w(~~QDuV>})x8pKKRL`Fai$=`Q%Xe*G)LOt)O%&+C;}LD``(lEaC{XqJ8*8o&q5uE@ diff --git a/tests/data/incremental_dev_project/pygame_2048.zip b/tests/data/incremental_dev_project/pygame_2048.zip index 2f0457be626f4f60d27563c5bdc7c86940260c35..93e9cf0fe6fa29b344701c93dc7b5b34d3e41e87 100644 GIT binary patch delta 6280 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 TU;9z0Sor@j0fpjgqFVn8F&a!% delta 3332 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 lq5h*V`hbbN zESYMBkc32$>ij>$>jIM$YX< zj!Cc|N5z1}VyUw>2MWSsUNp#yiPGh;ST?x9)i_(IpX#a6?qKjvwYVw2XYlyLHU9j? zT;qV=teG_L=8JrTrjYVCCLhU0=aLd+YpmN=Bxmie>UHpuC9Mff49TC2d4ETjKy+ey zW~J!h&aX783uE&RX>ud(dPwA&lRfHVX#-Wkr}NsoDh&I*ZCUag-jDqJKDT+mYI zH1MTC;)~^Gkw2M;vTG06vAB-AE?gRj)_a~A`L4ct#;g4=r&Lg0Vq7y@VyCUR{jnDB zacgqx7HYWIX$ea0<5RmlcqL8KKZ`qF1w9hWk{Wgox*T0=>h?j%oTKVc?w|K#=F*Pm z#-uFaEYatzUi|^bPv2>Ef`RoIt8BbGY9#M$_p<2A)jyo%-(2}aL(zo1?ue(#wtXRS z8@9xT#Kp%&W<>gWbM2<1SpyNjYe(LBIOcD>x6N9kUDO!0Wv$Ie4yk4PuM5uD+WB5T z6yz&VRptIUeC7FfV`O>8|u15~Cyo9fYMto{r z);Ev;e5b1MjN7BD$E@Ak^cSjRrJX7WzE$#d=}?PsHoG+N)HO%Dgv7GFL2b#&= zlVd&Y4c67x?&mA|yy_QsVu-RiuuXPqg}_f54v5UI_WzTrHQRJ(G| z=#{7d&9~KO#d>~vW1huV%;%n00jKWy;aBHAE*_MhEtus?eWDJWT+lMSHC1Mjc~+;u z_g|VGq0>iYhX+=C6W4I_?uk3iPkuU)ILyfFsQrJ-uljxvS+r=Z)tB4 z&LyV*(w}N4<4dZ4c{%fKx~G1p`QE}WK7XF^@?zs;#rXXPZ%{_uIl~>UCyz`w<$5-{ zIyG3R6|ltOQPfY!@k`M2`|ZF}-BkZeY41f`z05xCzO`A$c51k7_wkXMonEt_w_6!@ zk4-PwSTl@A3JQI0sfJGf!QC{H^_$3fY-8rAXuD@d1%~?PEbOm25*3jXmQz?6RTPmM z9t_mrN8xPMvS&~GKUuUlcE+W?-SdN#xw^@LYc6rW>z*t9)=@2N&r&0d3G$EhNNq3H z+J9qoTQ9e!i0{X-{&f4Mp7wYhhs6f0@Yc|5Ze<1a)bYJt3DM(jb5oh=6NS5;G`n9d zI*_-n$aYbSy9R~`O|UCz5&qk0UagOl%#=eWdhNpCUNagVZx7YN;Syf!7 z6GN&Mtgf11uUOppYtMX~UP4>!gj&bFJGfX5Om%CziWyVIYylGif-t|n$K$%dVri(b zSj$m&uQ`$`Oz79G)%#+FrPe=o2@EUq#rR&HjtSW_m_v_T?vtIv9PgGNr8re(lR5GhjE5~A1lgGndD-z z;e2v46nwMbbr+dcLO5+g%O6dQ@CC_^EKy2L3_9ZsyWdn6t6{|%4>(6Dr7H5O7h3#t zzao1Uc{=8>AeUTwx;VDl9%dYAB;Ow!t0{1CQc6wKK1%_AIgyqaL2wg;r_b7fmyo#D zSlefcW;NQ7u8{T^QDXx)6ez$@nI3A@@bK`6HZ(b-uINqFI7h)OgX_)-V3iB)X1Vb3 z7Rflf(vn{*fMf`uA|C_mg5km2q>93>Ive1N2-;wZ$O^scDd9Vr8f;;NLA%v9SlK{9@oGBrgh;Ck6#G{UxgY^2 zZ(_cQ-30sIrO~@l07gDE7B)(tl|lU`0b2zp>aU`n%u{RPTN>>`mo=A~ee2GUg>mju8IqH*DO61YI4y~~GW-Lb)drC3ue z`nS4lZpW7nWKAhff7(?93v&L+G!F>oBd>*md@X^iZP=A*_HGRHR1*>zV0u${r`D-uxskaqbU0(g(Xlnx1d0_7bN zShktY?cL1BkFQeD&B%_c0?^q)XQH?8(d!xoXBmmt1n`u>iE9!l*+v&__+9|_8LalSE?aO*q+N?{G1M!N z!s2clm{Fh&KIsl-A4gBAggs6oAHS0Z!_H6W@r2g}P{&}#bqN?G(JtqPJ*;IgoM1^3 zwk2C) zlO(?ajO^v1S)VzWe1+M$#<>0~9&YNhVMk$BUjQ6cn3m|8p^par6g>V*F2%T8raD4p zpDY%SP(0T3$FSQ-bVZAUdKfuC!CQyu_LT!~L+W41L*qd=SodE#lRRh#M;Vk3t^&(k z;x0qin-pwOpgZZs6ffP>hC6u}lJ7tsn9SjPf}Z3KC`cuWj_KNJi7MKnGh3V_&RrvAiy+5e2bKRGcm-s}{ z;*?#!AmruxUf-zcFIx96y5G3I*X!!QwtXXC{2gaLq3O^A3*A3;bq@b78l|^KPmfAE zoq4+0J}Z^JX*=09?0NEi52982E0F4`abIV4GZ(B>jyfW-bCjY`InakoZ2>c z;gX+PSWr~zxKy9nQTOM}PEB%8ZdGpFNPFAqqhxx8-Gqcmi{l+q?rQ2jO0c~eQ<$^v zm2L4{kn|i|9~8OfUhq1l>Vm7kz3VJ^aCO;0V{+Tu-;%FBc-we3dwquHwky>c`zQyM zki54qe>VQ>=0%BD-BD5UNQrCiJ*`-5R(o~9F4Z3$ z?2ArJ((am(zcaG`Q9{JepSQ+b+0bxK?c0IAmPdV(Fn3$03%8oZ`z)e-`UD#^>d$32 zW;L$Qu&D4#uxVV^*q46oz307@mirF2^Ba6@FZ5I|U)lS^;{zi%d|K~ny+`32#nq1{ z#Ln$pa6PE*pWpM-roH&N?D@sM&;iZY!2#2EFWtXTFK-vUbVsYHNz7QIgc5s`jHM&{ z)}Ff6c5LazUn`?pyUNaQXZ6rgQ5F-@Bn3f}*C? zSGRb~Nw?dg^Vlr3=t7ajqe4pCbIHQcpO(0rc`8rJdpaVrE~2#daM~Md*PMu)Q+;Vo zxnA*a1_x)J>{t;pY?_0vXRV*vFA0lh?6mD4>EqBo`C9LsWdkpU?!3NW-%9^QiOa&7 zDVKG6w0mi(TikiIpEjvpSe*Zzpq?re+O;MQ-Cu0to9(tju=V&myOQ47-*5V|^@iDS zWWKjLZPwtBdefvj-@5IscS6Th&XByPknX+j655x$7aG(*bZe;lY2Kjy^NOn69}?=G zMkaY)UYOd}ZAm)@sVE$GJH0D&>OWs}J`U-{osW#|6|%2Sa1zE(@#i2ejeUhy9Ib@$0@_wF7=fuM&*FL3R#W97bd zX)@QdDH~L{W>2St%Yc>S^G%awpiA1m6`i?7*1nZw<<>-5l2#I(wAnImOG(y_P7Y*b zXv*9yC9Sl-f~*+H^xf+?x8L^+mv^GX{6L93UCCh4Wx0j!E#q=;FE{55qpLW>o2n&> zvR3G%bF!2zC39=Mxm?=n73Iy-0}WhwSK|)OE3JuF*wB_!T*$Ll`4ph*+PHAIL~%~% zFLQ={?dEcW~E@ZlXlfD&J7YzAVpg5PL8Z<$-O&a z@{WeTB-k4J;IPa}`#qGW4nZgJgf)M(bBN3;dHj5Vy!WBw6gIDZoy#@s@|GJmD67Ke z`@P9r#+m&Jasb9ZVe^G|X7X{W)Kx%*a(m?yMYL7UT$wuri86Le>H6lJ|rlC(NhtEWK8Zv%8{>T~T2B7QzYTFc&0+Bfc znc`+WFXLkGet~SsIwCtVuvvgcknx$Z03A0`|6o%HgFhfwA>O5dTQm}(W6bspBBGHM zx=zHBXfy#0gL~1)107@@+At{wIU*5+#~@RN6&7B=f!{-U40bunJVt?fESgHGDnLaH z?vv11fUKxr#&%$b-mqlam_c z)!_-dHyLSArW0x8p|2U>gD4%cUIYSWJTfbgh+4qdL@;y%vZk_#Oplx-hS&{=dKt^zW10L>dMh+kYN(k&-2tiB=CI}-k{oF+iPgD2=t08D3CUD${%oUk#Im%!j z$?j>G#28*|L}Ka+BOB85Jj`M7CLRfOzyq#qLWb7-I+)()HW31WJ0@p;CiroKM9kun z4V*!eU9=e|yN$@ShL;%HHsfBl@M3KAF^pZd@Z0s6NrIBM;GkF{XL*Z1-kYu7*i|2# zw<1d=CIJN>2Gg5;EaBKzKIb7{;@Gnlji+uAnGW)^{5a_Xf8txb4f|#hnI19A5=5zd zLYrB{DL557tz_k+smPHloeBW^EW8}`w<9sh&4pFaH34Es-HtoJE|7T8KL(1SZ#$nn za~>P(o`$=ig$1<#d@-CzL*|5R1Z}p_7w}fGBEI2MF8EtN({F%_z8!9!@8MeVuB(m2+u_R zWcE#ix=h~eL#bewg*=I=kr9n(?+A0K&f+m5eXytT3|6A?5}3ot-0Vf0AfJ}T*D#_H zC}$%FuIh|6ptTcs{hDl?O|ukKXY(pXE8svjUark|Q6CY0hn`{idH1^}u6^(<$UFb6u!aR)VKa$Ns9r#5fNCI5VT);(7W@KK`PVg7t z*a+rf1oL+D3+~tm!FzDQjk|F;03D3)#C<#hDq}m_@}Ygw3DzuCJln_XO{58K`_X6AXFKSM9UgFL zKc27;7}<~xOE-s4bNHR04&^zxosBspCoFS0QJ>Lt1#JvdbMbpCiIELyH$XQ;TQDlT-;1Q0lZhziaoep5gg0Ib0e3PU*+*~r4HspW3E+jt7BN;) z))56TK7g}0$zt01h#0mX;H81S-&KIi`1yeJ6+1X%B_Ue2XR8v4@wvRHKO1pwWf-YLH|KqREZRL9^zN^z8dZw!XzgTN%NTqcMtIr zLR}0tfUgf@3bh)>Swv^lh@ty1FBdcu0t#_eW`%fgk{Ov-Qgj1}f}4eyLc#)iVxt&* zi}(})P+Np4b{0t!*~(xZjcOC!(<}m?Bbdd5$n?b~F`PZZBcZl5V<%%!F2+4q*9`W> zh}Ruqt#nem2+kB^hJ!?=E!xCjeU!(5Tp|A`PGjv+T**2{=B1!>i4crCh6(&wdCf6i zZ=z{1>^Kg7cuZPD41;+DdhM9N81jyzvE+3f$Y4YITDv)@l}Jaio#^0;68?G$>mV)H zmq?e>K^6}!<+V=bUm;G>r8vFoM5e#&6vLmTy!xnjo#a!P(4`1YWw@6dyT}JCx{Uvk zD!fXRPs*g<&pRx3DCaws-XKl~%JHY}F)RO3j%KkR+-_o9P{9v)#44LAaDeeG#(^%l zWeJl{@bk;KO(O9zjs0{t6HNQw`H!#ik~_qIk_3|ik!k(AmT*+UCw+CFIN{UXcxwCu z(sKEO{~&B3b~y5*Gy&m5;=k#kCFq~zljafwb1EE9U1OaVKKc*Rkmn=;y;D-s9D-?+ z7nZQ?6rc3TE8>XHlH;i{uZcXWlHZ9{Z|Ko)jUlZPk6jNV8`A6EnnPcuboy+6%jm$P zia&i8^N%@dF(pF^A~0d5na)*4_;@GweXS@WSl|%%FMp(tpaNuJi|*w7^=u> z!_Oiyam$k`D$nxWL_-jr=`{+>s*xpe`&_CBujX}_FhYfOyH$UXIuRSTsJ=Tvq#8JVR{2HHva$fyfGdg z_BRYa!l0u8je)7nNOjD|&;Fcmkv7;jq7ldzavPDsFbgRS5}07=I%{N(=B`xfC@?vX zM};}&2b@PDp|?H)cT=2v`gwddetRAn2{*ASV|Z_Y&$3f4;^P^UDmlOj8w?Qyj6)-A zc)Kz7qTz$vfDbbGM;gPVU<6(lkqPBw0;?_}FDk+W0^9IuuNk*u@~;~MDT_eBbVz~J zm0TbgH{p@N7YIhw@0^<4A?TBp@Km?i}WK%KOPic9zoATa<9f#x9iMfxU{OzH(9XEXYvUE*8yUg?r(T)Y1c3fn=6vVe9 zH_8qlzuVC=O6s_ti6EWXmdhZ<*IjR>qA(_8uW4BU@QfB2rRhS7an2 ztp~ODhG~J zGf@J7evAMKXeS5|*X6*k^B~Z`6w^z1ao8}}mop$S89)Jc8UnZ|QvOIeg8N7kVJ>O7 zN6GJ$Qo@zRjUVZYF;k#KV{i1vkA%`a3US`4g*`Q!MD-V%eB=BaDxBluLIOg90@!d* z7S9{EBT%R@eo4{%&`Z8`^_ptI>%WVfSGTTcW?7i$IZ+(OXJk;1buLmGP=bYvy(NW| zSmD^*~ zYA=%QBGy#a-tf!O(!V{K+F8btt=sJQ?79q~Ul)y;zu^5fda1$JQpEMt^Q7B=@Y!(~ z>rmj`*D8|Sv^?(_=$~k_erfyt_P}tu;xJE!upDnQ_6EV+sK<{%cRy^agzf@ST>^) zKk8b9^{Wo5z2h#8_ySu(wZZvzUQskX*~soG*{mGZ?;xGn>|gtHC|gF^97iapycmYp zRM!1)u=I!Z^sVcQMoW@Nm5RflJRvE^&6oBBS}$KA)nC4p7ZPXfe@jui1`E$i4-*7? zSArK8l~b-)l`k1%uTcqKJ_t^w(iIa(gI#>*JoB-keCf^TgbeQ5scxKE*muB+$j||cTdT@?J|QUOr-V`PHaw?<<_y)e zStG?LDe^gw=F_N5!@a}MuxEcBbX;~fzHUA&#AFDVqBhbfB-0{E4ZmvV&^j9=RG$wp zv2Bx@`jKg}za38Xet@)kcFglIM(=qggUxU0&xjxvI$dG!@5Bl}1*x%8Q0^||RN7xx z-qaIpU2c7XFv6WG3lzvxu&tDVfz>1cA1YLM8*C;4SnD!GyZ|&s0x+VI;AE&hQT#e- zF}97`nPkTDQ;q z<@yY;|8n33TL1^OZ}#P{p2WhA@Gn>BwDE5Yy1Fix+b1|Jw<}oxTB2G#r~Wd%US zbs65Ixc_pf_W?&O{t}>mLTVXyqIlxrXHSH2if}o>;og<{%b}d3{<@*$@BL-!&PRv- zO|Gst?#5rQI>q~MPH^&x#3w9+Q5x9)CR?`y`kt`NNI!R6y8BJ^bsusojvIjH>PlYy z)uAdMAGaT^3a`F+LiM)sNL6>g$sgz6NGOr^6BRq#p>iT6hnT0Y6CMpZ!F?K=JZ=_RxA1}B_)bQ1^q(M0beDp91s$q)D*q@Yv?^uV{!b-}CN$F)bx8MpFJbeGj%C$Cx3Zq!k7U=yrTtSm@`!FBx!&3=hT_8EIfx;>Sq=hrw8JU@##tj|LzE zr!xRzB=po478DW?6%-Yf5ESHd@N{(vdZ6z4L4f4$VNCDR18hmUhmKOCHzu+AF5M>g zi`QE*P0+t9l)nF~isyXX^=CMF2mU_4C>|S~b@DaTZ2-o2trEDnU#v^#yq7mo*gW`V zx#j)T-;n>^vg_c&EI*lzFW2V7UyXS)sj~QjtenW#f;?9>&(nIW%Y8_RT_w--xuv2c z-)~m1^fNP3>qyWiTrJs}ezzE2_X*0*RoH!x^hxY_;z`&$RsZ(SBNuPO!- zr~Or%`jK@lq^$MlNwQ~6TV+QdXk6$Uql^m75DrucdNl)6b*oFu(P8%^$OunGj0AMQ z#XsB8Da$RTB+DDn6>m5FLP8*TN;skDoQjaC`YkdsUt!{G1Lar5I6QSdrUShLV;0+y zdBrt6d5Z5%S+8~-<(QBg6(ZT*8=%PY_P>REUiSi@6+c>DS}5(B z0h*`oJpEa}twM{H1wILGZ#FNex|yIz;yXg(t!+KCr6p>ZP~RFXnucBM>4uk2mK$+_ z`7OcNZwZSRLCgupxZ_u9qHd7h2rTu#n)9gLW9(Ye3jw{i1=k-7^Iq<2q+2*-3D_e( zJ-HmF2bdV$JghwV|B>UtY><%#PsbU4eunO{Yy=F(fw|L=7_0wrr$rh}dQ6Iu98koW z{bnv3>bYJ_{QXkH?EtpxqlM`*h1rA(Vdt1hEZmY_`utv#Zeiq9@oYQE4ZrhxG zVPX$n=X*O(C8)jah>t2u+h5(%F|qGf+LE?^>|lMHF`A%WyH`{$#saxu99>jFA#K;X zR~)nR_Kj=*#`kxBw!?j`x$>Jl-jKWm>F-l~=q)Um;@IbiG{|!o8-7jxpu}W>+lWJ3 zf`3*ep8WOMy*GE~DW-?hhp&=z+sD;fRFhZ6$W6Fbr4XP53*Ud!IZHgH(xf7CtzqB{ zZc7>MbE{~JnafNOVhVb($S)x`n&xM?wVRr?N%f^Bjnp|j^5XA*5A+}?eQkA6f=8Ib znS&EzV|~M&rQw`0%=i**n=p%m3a;~qGCV|-kX%DZ2A4r0WA0;H6)h(bEGi+^?pVI< z&s1*Qf_)wF5lR8*k!%+}mqXT3h(8gGu_28h+Wy2U9wZ##pH;2%c+Wq6!r18(_lH$5t5m z^sUK*Q-`{ooaA*@yS%A17cBu*NQx!+HyMLen6R{o9G%rj znsJA&Ej*F9@N}Dy;yrE_a}@TpML%|zR0Y% zPV~!B_G%sxJH{2PU2k6~WckfqjeePEeceS)qDhXWOV{AS2+?qq#nL=T#p$^DCwD2|Y?!@fc*O*;=h*{$&?&Xa87Pg!I%YZSu)ymr-n-|j z(#Ks}X=>XU)8Z%U6J1wIE-LvU1ud5Bp9HRH?0k0RM*LYvBt3l`SE?-I-hQ)-e`e?F zC$*`oEA@q~0pYm~}CiUs!XXdW%*2TA<$+U;XlnGYi?VzNHA=Ywu92BeJXV zUsTyQsY-6R3nBgd_i|SpQ-i$aYWX!7Ql)JtmUsew!Ozb46-s>eGQn<>%HFV}zPVkP`|VM9y1&TK@y^f>#`YO52pmgi!o23*32 z(b6*%J8rw>$-MX43sc;A{DM|mq;uP{usmOkw&RWZY*vvW2h_F~DTJ5GL{t5I1-955 zTRW%j#a@0qLR;PhCLv9D_<|5CM1D5P)KlA|?W)7Q^4N8|Q_}j%)oBG99vnh{|q@Bi8VzA(`z!xsEtN5R)X-Si^S!a!!QRuttN z_Nod)WA--x<0P%XsM%N3hK!;XMv5z*YMZzbA{LblCi#STG(+|ay`}+L&))k!Fdfla za8pagog1x@8pfF)uwyZAo`;2rwZ#k#!6?W zuG^2_keWsIPIMw=Lw`Bx1#=fZ95rv_8n`?Di1iwK zxcAx<{yzOuX^WXMbzj~Ty>%nhe#>6WUeGOZoChw3bir0>T?I7F}1Ts4_j zqsF$}d0AuAsL>Frbi3HLC%x+e%ij=C3?o88-q>#uQe|fa3_95N|xNIZFe6k%&bbX%@HpC zk?9hkmXhW5&@lBQ!1KYY@A_pL?@SBE;snkcuEB+Eqc;NbJ)Tc(y(S8&T_9Q>Ml9E8 z{^+dwU^W+`u*W^wX#XQQY;?ul^i6L{pEuUUXDOE%cvtjj5EGaEebXvD_ZFjtF1aS9 zB)@k2O8BLd>{%6VM?G#sh z*_XeAug(dyyU%ieJ!9}CAhv~jTKmub!JKPH>miHjLwDbG4$V)deLLgq?!ZQ1eCL_A zoo(61yEK3j7G^jx5d`SK3k<+n`jeAZ7HE8N>5MEJ!h*pj2w*Tykdy(SLVEK5j~Q#d zeivrOnnG{prLD)_X3usezVEv5i$aD;X$grd^D*7j*jhXsL>VtSe5tT6s%U36)9tM%@iL{iHQwSJEaefvFtG2tRVWe)GW?9vF=V{o0;wn?(WvBiDxc?V2e zDO|Ow@!Q!M`tanCQ6dw}_vHD7UY1>pR zHL|NJ9?iUrezs@7w!mO@?+gs;gqL9yI>_@+63+TEr&zdqmvO7IesEj(&GOOAJA%I} zi2DrReW@;gT(PCEAE;*9RoVFvfv{~tk0)+Pgs)_De3Syz>xSR9D^k?QF>ZoE z4*V=c8OTsxd>#|i$QQ^TP?Z(0Vgm0$JT!hF-l?CCI2&gpWp(o&rSf6C=al8mT)%U< z)O`B;#JwH>aXedYuXk~nXqZb|SE{?wVcz&HGcCqX{kCdzLV85 z?k`$5Onm=?(?&ve$J?Hqbj8alSFzo_9LZ$3+;A2aPUJ$BA;FQ&z-IBpP{QEyEzT{o z&OJQBb~*kJw!inijUSQUrxI&k4NGP@@73GrUL|>EjrXywml3(Yvw$MYV;}uVnt7gg zI3{6UdiS^fOy?#Cj%!5L4X^$(7rqibz8ixz;m_wLh>(rHq4%5VElWMc@R^Mp&nC)| z4D$>-20J>rq2t+g{r1;53%kXcr(Vj-WZfSA@wltn!;aLC-`T+pTc z%B)r~?{J>e_j@)+I2q&Z2X&vQttFy(LT(n=&j_CdGOk}1xm6>$F5hc^_KAc=eEh74 zzEUM&!eoRZK2NS@S{6>KYVXs9pr5&4qLJ+BxBNY`{5!oG*d+gGNTbB!tXOw~3g(|~ zcQt2qmG{p5dgt`eS*%%J;)(m3BxTRL_ofL8LS7D5QdEbI-3qlA^d3f;;HKAdyAz3( z`Pg$(sO_JvO}@^9U&3rXXo0<=_tiab;IrX1XWRU3p`s7DcQT6j>~-$Dla8bwSxGAd#k23EPr<$9?XhEwj;^mQ8f5K)O}$}x*0o?F6V zTo3#^A_zg}sKxgQKwkIG&hXW&qsan33c$BOs4J4;l95r!u!JC$ z2@pIrFLAc2PsNNDch>@A==XAC^CiE&5|*&Ltntvkca6wv z->X{rdQ$FhjM?$ADW^MHcDgn_n|fsZ$!hLqFx5|8bx^M`$DtU7UHuyHmQn95H=cZf zWZ>TRt8cfqn#+f5xsmF;$ZORDqSCK}h2+eBRy9(Ni(d$Ve=9G~-4DD{681FWg2bO5 z%lA!{9hdj9l-Z10BQ!NWv9#JUMvDe?JXh{*QN0wv>pp~d9Hzqiisj%+^w}jA7*RGM zOl#Vk3(;DnX6uZX9ECC><-DV$Trvnyy7KW+9x~vTCV7c4S%2iYNQQKmDSJl~-H=&s zsX?c!xf3``oBpS^JN=8j#Z=*1d+3hXYoyV9(pP&MudU6{$EF9hPt-PX^xwz?%ll8y zz3*&JAT_ZuQ+HY!^*B#{ow%Ds%`}h^{qiJCRsF zu1xq`mvrG`C1r=L+qo9$$W&MXvE!qA7Opx3^gk}@B#cOQ3}UO%9w24vRkZ?d8itC_ zj%02`82@MsBR-qf-phC}^n3UFWg`hXVkh-crHiWgTi8uo?ZI(t^a)h0Z^&$J3q}>> zZVxr#TJhTAN`(a;sGmPv(|sB$X4dn}XIsEGsxeaQ*;d}7a+MK2fAPkMzuwZB7Z=L1 zv@QhX2-%h3lC&TxEf6BeZ<6;??;C3&I5#SPu6=VNXI^7ab?wtE&`hDUi+9-_8gIxk z<`FO7l9`uY`kv(4!lUYvblr2B{((?dqM9~CW#3r3DhIDtvdHUwT|^!)*Q#;NIQcGC z4{h`l>1W0I8!|(z1w1)V)8pujN;M0mRrEdBGGZ@$D&%7x=&M439(nE_XB6UZZja(JG_p+JjzGdD4b#0x0)_$*52t6F|9SvNTCPk)Vr8) za^Z5K3r-TK1f_d%uP!j5MXB`t#5 z4-6Xo%5#QMHDjo=X6@0p3+UAh_$M2nX~bm$6pqFObrwJx+HJ*#c3a=jpt@(E<%$g< z3?>Q=^8nPy+pbpD=APcp&Xykb*8erPaJDMaH_@C_CJ+Agn7WHV**O?SoMKGkCztXt zD4}g5!tj!hgOq|@{W;zRwr52YjXKR?4R+j=c%K}V@eTx+-7`!6L`nmqj8fa^CCoL|6>*DJ#71y|t^EFkfZ`RiZYCMY-ks=hT>M4tbQ3?$yMh|)IMk6ao zUQDf~{OUtU?XQ^gJ!zB;2tc;HQ_9*Ece(wd@^#DBjfrA^naBaB&7wxn-S5rc8d6&i zXbfMZbA){S^Tum=X=<%+YKE)jlG|hbws^9Jr%T8ng5cqXpmhgW1aiqd49PrMu}i@F z>vzc8s)GgDgU2R++$pYjH%$Mor}<)GJgSm}dsbWi)sLJ|+AAUHSKxwQ48s)Wld1xB z+oc`>xu1SZE`AZqaB)22InMzYJ}u)~}Abzt!Qg%-~gtKQsSFI^j&}(}LAP-1&1~ zXQ&aSt#*ti0H5frL~x6YvayBH4(#4$(37O&e=7Zk3X2Gc0L*kM2>d&5s|%U zC@1aC&p!`56JbAVg&p=;irfY7 zB)_CGKGGAzRL;w9hXX-te)c(6^%q&`c#SgozU zblvae>$pPTZj#6QWit=I5ziL+b+>PO14pyO?3ccnWm!n$rS`B}rIN3PhM)3e72OqP z{&T;VntXgMHHp0F+G>oB<`2Gnt_ZJ;3zJIxAEF%pm>=F*(;R+3IlaE^dS`N*Px8Vx z7QYpp-PIq>MoFwjsj*{Wa*vYLbHK8s09mK&k%?Q12@;rg9^^t;~)aM&HcnC@3?CyT&fEo6WGQL*u zAgwjG6w>(|>gv8;xXgq<{fP2&2-zU1ypRyr&~E8HTKPaSB|$?)PchxNEClP4w4OWJ zlq!X|lU5*-MKO zyB*ULO~&}B_Q)C4gd#rk=zjRN@yuO4K4~4lqX*}ZSb^kY|ASb?o`)z_R`SHytTilQ ziA~zIM5VwRx>RlT!t=aU3F{l{aN@S-HwyVyjBe@%p4WRU8N2*_Q!g+|0f`_ZPPPb& zczv$nBW!7Eph~vmUcMc{T<=-GgXg2sB@Lgw?1wnh?*2q6yOu5miHTgho2@OA*v+Y> zahW4Vo;SJiW&S0VRyn+T1f8jL(=-p6bf4Y2D`6bH&yp znJgl=Wxfw{YUpx!s5Kd6-n@DbnfUsdu(rLKaQ9@7Mp9pS%pZ}-yY(77L^Pqb7GC)Z zIQtw*nQyKL!)d4L>70uT9q)-=Egf#Nfhm;S!tMAjwWDUSMqKO`qqhFQPLnxwv`arc z2){~60tfWW|9&TOYW%>3IJ{@c4xisAMlG^e`WFl3IFHl}$!F<7MeSQ{S=7o?VcJxM zIlC)=&iu0#baY51fdx<`Lccf=P~MXO02r(toM8b(PEVzrt%8O-G@c8P;I3ztG@a?x zvqL?(bwx$~kgyYmh4sx%MClgkr2R_7{P(j|SMHT=TN!n%ZBCTn*3s0f5>$M`**DQ> zPG%G>&$r|@2~)T|VPfCa+J)-%4Ws9(MOm>|(~i+pUVfH3EGlY5GK-sBp%{cLT>0gd z{Jn1_Yd0i*NRVCegS!v^=IrYUR=?lD@l9d2aFev+mm$sg-ab=9Gog3gGegV81};fz zQmre88S~KotRY#gi86qf-5p3R1`h+v%9x(s1uZC*K$4fKBV9`G)C{_);<5PE)9cqS;Ac>q0O}r1h~NlFrTiZRgi(IC-l;#ec50oj<{X9eA6xY zYaV@%yVGmdvJ8(IGRcdP-S0-!g^y-e(dlSuG9xmQJN?0(&S$@NM^0E+TV6ZdcW%Umbg6h5!Z8?OZYwp4nj;t}Se+`3poGtN4?nt|BplZ;x z6Co-n5T@b&UEZOfyg^+;Z0Dff6=yNS`#gd)x2Z z)gE2!E5Z{7Zujl+u~aTe+I%wP$5&#?Y=U`u=p@m{6Zd{lF2)|Z))@Dc2FE9~*)gAV z@#j+RDm;+)Gnd*&4gqeYsGLQ|TT6X|qy|S+Wf^w-?2w(H#@O@PU7630#@yGq9W&IH zMdv*elWrCu7s#S*jNCj<#S8u?dqLa(BrW?a%o6K+E8x%$(q_!zw+7 zrj@Gk((xDbO21kTuAf2PiOAe9_h}N5IjikLg1Ynb$CBDX#F%4`S3oQxLhziu*j_m0 z?zOH*ftRHHC!e@Z@A57ZQ#Fj?cC=ZaTOGP@wmtPSd${9ahLf}BeI)C{cXPFxU&HT8 z+!XEtWxn<)=2+f*Eh6btsSzz*@{0qHT>f3y{gi~sp|zh2xBkp}s8DT!pt)vrJLQN0 z70>V4*}^X=_hhI>21iR>6@C`olaY=0=gkLiEyqFv6J63EZWWJ?$y^GE1=%Y{aOzbOKiXh>pR_`4(MYz#4BtAO8~a7uC`Y$-|~tc_{{! zVPn?4>P#ekwE6YR6HoZQ?d5$!9q9Ta%sPCCV#Ds?ayr_i{BSb_z!hIPGo4!cDyW;&ii%d#-CQ}r6nn+Iib7}+87b$MM@JD z&B(SqJ`EuSw*#-WJ4j=kdluA(9+nqz#1~6b#h7^i=&ZHFiW%`U&3@>xua=Vx-@f2A zs&Vs+zw(c_A+65pIChB|_Xge-_2E}7bt9DBtREsA1VXy^GM6YOaZ1Z9%3Ng4#yS1O zFWdRL>lhXHlq3AtsdbV!Qg3k$^bfXL$Jjc4a*1m`xH3V_*nb8ICocZf{`woinx{~% zY2Rj~-?p9i#jJ7fUr((%ls_uTv+F5RDj%^-W;FKhbpGtkO--V*s`czU zNFGR2ooTZ)`JjQD*pSf}%SVnPNjP)UC%$DdH?d~Mu5r0+|OJ+?$M@l`QBOwCl1q#!Do&<4rwu(#?ba^g7 zsQ4>0?6%}jID-&&q^REyfbg2l4W@=t6_0>}rh)@p_?DoUmv#VFihnM6=Mol_B zd;QM5Q%Q4NXR@pq1}MJ>D4-;Z5Vk`0kDnymsd0bQWX<5b>Fymc&R&iNh+5m^qZyouA$0!>Nt}Pxph*)>3C$- z$AVg}1s%q>93cN=+Xp=40p4PY^o|#h!4&D>oPYs7w2C_qZ3>H^i*z28_ZQ}P`ogU~JYW<EQcZRm#cAMo z2pf}s{yuJpRmSPZ3cYqJ|Cm(qL3;xCEC+*}r$&yV*Y{0_%e~!O>Go<@(@j&ZY~{-B z;!#@gD3ZJJ;S}6e?g6d3H+8NOZ9D7NE;;^2^p{?{uBTZBzeK*H-7PK9y50egQX_qV zTVzhUPg;%_g{)GUW?IU$iac1kn8f$=%{_)n#R*A<Xe4=AbYst+xUee(Bn5FS|AwZLV%P zPZ_SJ=d+kg?^$<0>=%ept|a;^DBGnJ{&c=kjxDCRvCfEG`D2wej?j%|X0M%E@TCTO z2aKZ2IBaye=N4a)y53rw5^4`famc1r+>BTH+A^&(T6i<&lO_(Q&c0wV%l)yNeRqt6 ztL8hF^vazPhu1G&{50k;^NBv$Q>ibX123V$?T^(MU#TR2CY3!-w)tzvAN#Oh_ousS z$-SFeH>MVykncF0(yAAH=p@?auMG_l5~+UmO|p-W9rP~xNCMxL8?g4Offh~AH)UjAZ_ zdwzXcrQ+FbvE$DA@Sjw|G9L9VP@}1?%<`i7SqFQsg1Vh7jN*9F(UW3H{Rt;AKq`_mDvD-?3ztb zH-|)4&t7fc-uXe8oM;_@7N|lW z0|uLP050GohTPTxgyCcW3fevb-X&jfPnkK!5lrv z=$tOV1+P3}P*ZRU&=D;l!amYI)&->C*Ksin${+VUNUsNJYlzT-^Qbu@Qt+Jtgcz;| zKO7NGYff(7+4HFjCN7ACQKxlA{?~@VP#41Oe09t#Py9|nT~Fh=NK6_z74k_CZQbHOKA(Sq|JogsvT&xK~FQBqt) z+*+ytoE-M|dx*cEEx=$DXh6&#BlPbKSY-&Xg6Go#1|S#QFa+d)QS=WrsAdG6c*Mw? z!7d|6kZJ@K^gEiq2B#52F^SQE&(Z)oVl>Rr9|`oM18Nxq>hMIg0W&z84xGc3otXL` zBN*Ds4GcAbNJgs}XbsDs3U?cF@8UP{3h=V*f&u3}BxrAanwL0kkxOq+(`J zT^Z!T4Rm@zW`M{ElT~mS(xOj?g9l~+6)qZ#T>%_11L#3<^HXTq;D|YdR%{OC@(xYA z0AAn~i<1q4Ctg2_GH3ztz+W6Q8(QX@K1SoZ54$#2{D)`~C z;437UusInQjxaxRX0!!3FwRQH0(;w&jfAsX9NE3Mg(6xUF(?%a4)D4i6ky-(NSkN} z=)lwLkC;QdQwS|iM`9Oy2;t`ugPL;U0H^H%HcaRQXN(2P&zS?%a5$CbdzT~U6o=z9 zNnMWwQw}GC0Jfl@BXk$%qJJ1rac*p2yW=UKQO_ek94CMSbIA$8JSV7<>P~>@-vR@l z&{AbMr8g?nn>N{-1LSi)1yhSwTmb!?0S@>wnx+RkoljmOU=1X7Ilh}^T#oN%jzBaV z=;LxSXZW2kjC8>Ts!Pog&F6a30?+_MU7=P*p??^_FRmv8!NVgk?s{&Jd+QOM?RL@~ zPJb7&paBWp0baO1hDOQWPcpZCUO2K_*pC9;9iKBwSPz#1P2?ye66H-x57_EF+A zkjWQNf-7Tw>A@)9lLZE9L8d#9@3t@GyNsq!_yX%-;vMKl;miY*?m+$Y*PTLc({)JTauXr2un`M?av&r|)6hs=G5VnBU)%m&8$pGrF&1Og!caepZE8#H|~ z^fKrca16i-x(7l}MMM(^co1+3p$n~H0Ivl?2*1(v2?V$*w6hrq$YU-=1cs(ZxwW%_ z&OxUz7{Qq!2%{kg!Z1S9Coq6iP%9WJ{cH3O1FGd28~7>sly@JfjfA|}kWdX^-ROX{ zpfmE+<;wy)k;g3=N0SU7d&sFe8h~9PkTf*}${-Ol6)7z)s%e!pM? z+d@yJ&qi3AhhYvd@b0O!zJu#`A>f|7P+i&HfE#yDAzdFuX^+x_K~a!43{A^}D^aKXH$jVg zkiY1?(Ef;#Nqyu12cu77!-YR# zJkP~Io=!)!Nepz!|D75iN3pwD(WJ)vRJhI?MaVBi*+nzWn|vEU_(&|Aj3akRbtjQn()RhFhFqY{Dr# zoE;3_NCFg9=7^?A1T-;uP8{*kiKnnjcQGD2iN_htV0hHPE+^=abOM_g*uxlBCqd8; zFf=M?kMl$UWA`xzcFB-IH-<)4?sI}4lA&9t=MbY~M4gO{H3NN9RD%WGJP%L+2Nn%B zhNA-~QGnVR^sG%0`hE-2(}5}vPWA1b_kYw2vwZ{>R1 zlmGa$oWIMa|N1cJDBN+pxc?i(|1*Q5<5J+2On{J$;}l*k-TzJrbk782;P;rpnoNKf zQcrE;&$`wo6A+PZod_T+aaTnE%Il zmE%A#%qsox$b`x0e|hOiBGi( zY8K?r1s|9F7m*$0&O7e5Bp3h98ND2ryn1AGHV+yBdeLq{e}++^cJLD$KyzY$0}omQL)RN5WQQ7Wo&ym6z2WCI z&?;&Wkq?k#?)p>P@uvbaYW{1AckVxKeQ1jwq|Aq!r>6DS2;}93HYw1hVMK2>{aYA_ zVELCpSLvcQ+UX*??+{{=IJMn z2sPHR;GaP@ssP%vlE8>r!IlE30e;5}K_RsDvvd>fzzVvF0_5QLLdZ)<1|Y{&_UN9R zwniJ+!08>1kOhmNeXu@ZuBVmXd^z?8nhQh8N nOCA>k;_x{ij5@Uh5Q10xfg&XUGvW{I4{Q>u;i4Y^73BW_BT}e% diff --git a/tests/data/incremental_dev_project/word_cloud.zip b/tests/data/incremental_dev_project/word_cloud.zip index d15c1cf74cea5a7ac15ed29f75574351949df1cc..d8747d14d884677004c7f29ce2422319cb679eaa 100644 GIT binary patch delta 781 zcmY*VTSyd97@l)nYjww+!JVDeZPZoUcJ{c=>_%oQ-JqaQABK>yki^tNBndNcOL8mg zAu}((MQG7eS(J1!qYDb9_@Yu0V%SBYhl&Jyum_Wk=FDo~59jcG|Ns5h>GF5w4y_=@ zBrptPW~#!$;(_5mY_rY|w>xqgAMbnP%%8k-yQ)3??i0l+#u=^w|)tN2PoVOZ3+vL zkB2zF6)z=2ic^9dzl;KK$iE9!>VGoaawfvKpWvbtDD@}7m6WxR6d?F{nn6nfxwz16 Rz}`S+{8SvneD%;Z{sHx1=7#_P delta 758 zcmZ`#Ur1A76#u@vbatKFmFZQlo4Yz~_s`tjEl6Tz5C!7WM8TpW%T3eFj7>=;)I^F9 ziFIDGz`(GF2+AzMg8n!Jkp{jL&nc{tZR6aRWcffUQ>{DfIz%X;`jBi}c=!h+`Gn2vCm4fNq?}P0H1N*k$ z+k3eho$H-&)}*#nA1tvATyzXxi+0qNjyx*5(QN+O(_3)@MtA&Nd~)YkkLNJ42f2x9 zk$EThI0me^0mkVR{(LAHR{mPQNvtnKA7AI<5UNQ;NPuC?;xI6B%E%mENB}=)ffqO~ zuK85BAq=y0CKidry1^Nl9dRbUwk7>@0$6n&$CHuw9DEQt=o8HNht>lR1c`Ld3WZQC z(D4kpRxO4uoJ#iX5$%jX3R+;qlia`g#K=6{7PF{t@sMnEl92+n(+H);9Pmg271AIN zYpex$nSmvdak1JEJ+u|~X@o8`g@RRqvrR2QCD;@QE9;+$Np_dGp+x%5`TuGexe&YH zkX(dMg6d9(hq8o$AQ|G_5`IfiE%w5)J0B(pZzZUnqRsHdod)e@2DK On&`Q$2rarav%dj0DeW8p From 4680ff5e6232a9e17953c2db02140ab59f4baa80 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Thu, 25 Jan 2024 17:33:23 +0800 Subject: [PATCH 488/637] Only retain test_simple_add_calculator, skip other test cases --- tests/metagpt/test_incremental_dev.py | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 6a26f9b83..3e4a1b901 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -50,33 +50,39 @@ def test_simple_add_calculator(): log_and_check_result(result) +@pytest.mark.skip def test_number_guessing_game(): result = get_incremental_dev_result(IDEAS[1], PROJECT_NAMES[1]) log_and_check_result(result) +@pytest.mark.skip def test_word_cloud(): result = get_incremental_dev_result(IDEAS[2], PROJECT_NAMES[2]) log_and_check_result(result) +@pytest.mark.skip def test_gomoku(): result = get_incremental_dev_result(IDEAS[3], PROJECT_NAMES[3]) log_and_check_result(result) +@pytest.mark.skip def test_dice_simulator_new(): for i, (idea, project_name) in enumerate(zip(IDEAS[4:6], PROJECT_NAMES[4:6]), start=1): result = get_incremental_dev_result(idea, project_name) log_and_check_result(result, "refine_" + str(i)) +@pytest.mark.skip def test_refined_pygame_2048(): for i, (idea, project_name) in enumerate(zip(IDEAS[6:8], PROJECT_NAMES[6:8]), start=1): result = get_incremental_dev_result(idea, project_name) log_and_check_result(result, "refine_" + str(i)) +@pytest.mark.skip def test_refined_snake_game(): for i, (idea, project_name) in enumerate(zip(IDEAS[8:10], PROJECT_NAMES[8:10]), start=1): result = get_incremental_dev_result(idea, project_name) From cfadd54a3a5aee02416ea092087cdb65b6171611 Mon Sep 17 00:00:00 2001 From: geekan Date: Fri, 26 Jan 2024 15:02:34 +0800 Subject: [PATCH 489/637] Update token_counter.py --- metagpt/utils/token_counter.py | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/metagpt/utils/token_counter.py b/metagpt/utils/token_counter.py index 885eb37d7..feec20928 100644 --- a/metagpt/utils/token_counter.py +++ b/metagpt/utils/token_counter.py @@ -4,10 +4,11 @@ @Time : 2023/5/18 00:40 @Author : alexanderwu @File : token_counter.py -ref1: https://github.com/openai/openai-cookbook/blob/main/examples/How_to_count_tokens_with_tiktoken.ipynb -ref2: https://github.com/Significant-Gravitas/Auto-GPT/blob/master/autogpt/llm/token_counter.py -ref3: https://github.com/hwchase17/langchain/blob/master/langchain/chat_models/openai.py -ref4: https://ai.google.dev/models/gemini +ref1: https://openai.com/pricing +ref2: https://github.com/openai/openai-cookbook/blob/main/examples/How_to_count_tokens_with_tiktoken.ipynb +ref3: https://github.com/Significant-Gravitas/Auto-GPT/blob/master/autogpt/llm/token_counter.py +ref4: https://github.com/hwchase17/langchain/blob/master/langchain/chat_models/openai.py +ref5: https://ai.google.dev/models/gemini """ import tiktoken @@ -25,7 +26,10 @@ TOKEN_COSTS = { "gpt-4-32k": {"prompt": 0.06, "completion": 0.12}, "gpt-4-32k-0314": {"prompt": 0.06, "completion": 0.12}, "gpt-4-0613": {"prompt": 0.06, "completion": 0.12}, + "gpt-4-turbo-preview": {"prompt": 0.01, "completion": 0.03}, + "gpt-4-0125-preview": {"prompt": 0.01, "completion": 0.03}, "gpt-4-1106-preview": {"prompt": 0.01, "completion": 0.03}, + "gpt-4-1106-vision-preview": {"prompt": 0.01, "completion": 0.03}, "text-embedding-ada-002": {"prompt": 0.0004, "completion": 0.0}, "glm-3-turbo": {"prompt": 0.0, "completion": 0.0007}, # 128k version, prompt + completion tokens=0.005¥/k-tokens "glm-4": {"prompt": 0.0, "completion": 0.014}, # 128k version, prompt + completion tokens=0.1¥/k-tokens @@ -47,7 +51,10 @@ TOKEN_MAX = { "gpt-4-32k": 32768, "gpt-4-32k-0314": 32768, "gpt-4-0613": 8192, + "gpt-4-turbo-preview": 128000, + "gpt-4-0125-preview": 128000, "gpt-4-1106-preview": 128000, + "gpt-4-1106-vision-preview": 128000, "text-embedding-ada-002": 8192, "chatglm_turbo": 32768, "gemini-pro": 32768, From 59afc5301f55037e7b379497767f4af62fd65b31 Mon Sep 17 00:00:00 2001 From: geekan Date: Fri, 26 Jan 2024 15:08:08 +0800 Subject: [PATCH 490/637] update token counter --- metagpt/utils/token_counter.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/metagpt/utils/token_counter.py b/metagpt/utils/token_counter.py index feec20928..94506e373 100644 --- a/metagpt/utils/token_counter.py +++ b/metagpt/utils/token_counter.py @@ -79,7 +79,10 @@ def count_message_tokens(messages, model="gpt-3.5-turbo-0613"): "gpt-4-32k-0314", "gpt-4-0613", "gpt-4-32k-0613", + "gpt-4-turbo-preview", + "gpt-4-0125-preview", "gpt-4-1106-preview", + "gpt-4-1106-vision-preview", }: tokens_per_message = 3 # # every reply is primed with <|start|>assistant<|message|> tokens_per_name = 1 From a6bdd0201765e3f6b1dca8aa398f25496d3158b3 Mon Sep 17 00:00:00 2001 From: geekan Date: Fri, 26 Jan 2024 15:03:17 +0800 Subject: [PATCH 491/637] add ActionNode.from_pydantic --- metagpt/actions/action_node.py | 79 +++++++++++++++++++++++++++++++++- 1 file changed, 77 insertions(+), 2 deletions(-) diff --git a/metagpt/actions/action_node.py b/metagpt/actions/action_node.py index ca41c76a5..162ab90eb 100644 --- a/metagpt/actions/action_node.py +++ b/metagpt/actions/action_node.py @@ -12,7 +12,7 @@ import json from enum import Enum from typing import Any, Dict, List, Optional, Tuple, Type, Union -from pydantic import BaseModel, create_model, model_validator +from pydantic import BaseModel, Field, create_model, model_validator from tenacity import retry, stop_after_attempt, wait_random_exponential from metagpt.actions.action_outcls_registry import register_action_outcls @@ -186,11 +186,27 @@ class ActionNode: obj.add_children(nodes) return obj - def get_children_mapping(self, exclude=None) -> Dict[str, Tuple[Type, Any]]: + def get_children_mapping_old(self, exclude=None) -> Dict[str, Tuple[Type, Any]]: """获得子ActionNode的字典,以key索引""" exclude = exclude or [] return {k: (v.expected_type, ...) for k, v in self.children.items() if k not in exclude} + def get_children_mapping(self, exclude=None) -> Dict[str, Tuple[Type, Any]]: + """获得子ActionNode的字典,以key索引,支持多级结构""" + exclude = exclude or [] + mapping = {} + + def _get_mapping(node: "ActionNode", prefix: str = ""): + for key, child in node.children.items(): + if key in exclude: + continue + full_key = f"{prefix}{key}" + mapping[full_key] = (child.expected_type, ...) + _get_mapping(child, prefix=f"{full_key}.") + + _get_mapping(self) + return mapping + def get_self_mapping(self) -> Dict[str, Tuple[Type, Any]]: """get self key: type mapping""" return {self.key: (self.expected_type, ...)} @@ -616,3 +632,62 @@ class ActionNode: self.update_instruct_content(revise_contents) return revise_contents + + @classmethod + def from_pydantic(cls, model: Type[BaseModel], key: str = None): + """ + Creates an ActionNode tree from a Pydantic model. + + Args: + model (Type[BaseModel]): The Pydantic model to convert. + + Returns: + ActionNode: The root node of the created ActionNode tree. + """ + key = key or model.__name__ + root_node = cls(key=model.__name__, expected_type=Type[model], instruction="", example="") + + for field_name, field_model in model.model_fields.items(): + # Extracting field details + expected_type = field_model.annotation + instruction = field_model.description or "" + example = field_model.default + + # Check if the field is a Pydantic model itself. + # Use isinstance to avoid typing.List, typing.Dict, etc. (they are instances of type, not subclasses) + if isinstance(expected_type, type) and issubclass(expected_type, BaseModel): + # Recursively process the nested model + child_node = cls.from_pydantic(expected_type, key=field_name) + else: + child_node = cls(key=field_name, expected_type=expected_type, instruction=instruction, example=example) + + root_node.add_child(child_node) + + return root_node + + +class ToolUse(BaseModel): + tool_name: str = Field(default="a", description="tool name", examples=[]) + + +class Task(BaseModel): + task_id: int = Field(default="1", description="task id", examples=[1, 2, 3]) + name: str = Field(default="Get data from ...", description="task name", examples=[]) + dependent_task_ids: List[int] = Field(default=[], description="dependent task ids", examples=[1, 2, 3]) + tool: ToolUse = Field(default=ToolUse(), description="tool use", examples=[]) + + +class Tasks(BaseModel): + tasks: List[Task] = Field(default=[], description="tasks", examples=[]) + + +if __name__ == "__main__": + node = ActionNode.from_pydantic(Tasks) + print("Tasks") + print(Tasks.model_json_schema()) + print("Task") + print(Task.model_json_schema()) + print(node) + prompt = node.compile(context="") + node.create_children_class() + print(prompt) From ed54f6b86a7000de1487472a8900138933ac6c42 Mon Sep 17 00:00:00 2001 From: huzixia <528543747@qq.com> Date: Fri, 26 Jan 2024 22:59:10 +0800 Subject: [PATCH 492/637] To avoid JSONDecodeError: Remove comments in output json str, after json value content, maybe start with #, maybe start with //, particularly, it is not inside the string value Addtionly, if you do not want JSONDecodeError to occur, you can add 'Delete comments in json' after FORMAT_CONSTRAINT in action_node.py --- metagpt/actions/action_node.py | 2 ++ metagpt/utils/repair_llm_raw_output.py | 31 ++++++++++++++++++++++---- 2 files changed, 29 insertions(+), 4 deletions(-) diff --git a/metagpt/actions/action_node.py b/metagpt/actions/action_node.py index 6c65b33ef..0f441cfee 100644 --- a/metagpt/actions/action_node.py +++ b/metagpt/actions/action_node.py @@ -24,6 +24,8 @@ TAG = "CONTENT" LANGUAGE_CONSTRAINT = "Language: Please use the same language as Human INPUT." FORMAT_CONSTRAINT = f"Format: output wrapped inside [{TAG}][/{TAG}] like format example, nothing else." +# Delete comments in json +# If you don't want JSONDecodeError to occur, you can add Delete comments in json after FORMAT_CONSTRAINT SIMPLE_TEMPLATE = """ diff --git a/metagpt/utils/repair_llm_raw_output.py b/metagpt/utils/repair_llm_raw_output.py index b71def136..4995918c2 100644 --- a/metagpt/utils/repair_llm_raw_output.py +++ b/metagpt/utils/repair_llm_raw_output.py @@ -120,13 +120,21 @@ def repair_json_format(output: str) -> str: elif output.startswith("{") and output.endswith("]"): output = output[:-1] + "}" - # remove `#` in output json str, usually appeared in `glm-4` + # remove comments in output json str, after json value content, maybe start with #, maybe start with // arr = output.split("\n") new_arr = [] for line in arr: - idx = line.find("#") - if idx >= 0: - line = line[:idx] + # look for # or // comments and make sure they are not inside the string value + comment_index = -1 + for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): + if match.group(1): # if the string value + continue + if match.group(2): # if comments + comment_index = match.start(2) + break + # if comments, then delete them + if comment_index != -1: + line = line[:comment_index].rstrip() new_arr.append(line) output = "\n".join(new_arr) return output @@ -198,6 +206,21 @@ def repair_invalid_json(output: str, error: str) -> str: new_line = line.replace("}", "") elif line.endswith("},") and output.endswith("},"): new_line = line[:-1] + # remove comments in output json str, after json value content, maybe start with #, maybe start with // + elif rline[col_no] == "#" or rline[col_no] == "/": + new_line = rline[:col_no] + for i in range(line_no + 1, len(arr)): + # look for # or // comments and make sure they are not inside the string value + comment_index = -1 + for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): + if match.group(1): # if the string value + continue + if match.group(2): # if comments + comment_index = match.start(2) + break + # if comments, then delete them + if comment_index != -1: + arr[i] = arr[i][:comment_index].rstrip() elif (rline[col_no] in ["'", '"']) and (line.startswith('"') or line.startswith("'")) and "," not in line: # problem, `"""` or `'''` without `,` new_line = f",{line}" From 43b069f453d0ea351ba31b918b4fcb8bae5863e0 Mon Sep 17 00:00:00 2001 From: huzixia <528543747@qq.com> Date: Fri, 26 Jan 2024 23:20:16 +0800 Subject: [PATCH 493/637] Addtionly, if you do not want JSONDecodeError to occur, you can add 'Delete comments in json' after FORMAT_CONSTRAINT in action_node.py --- metagpt/actions/action_node.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/metagpt/actions/action_node.py b/metagpt/actions/action_node.py index 0f441cfee..ed0e27869 100644 --- a/metagpt/actions/action_node.py +++ b/metagpt/actions/action_node.py @@ -23,7 +23,8 @@ from metagpt.utils.common import OutputParser, general_after_log TAG = "CONTENT" LANGUAGE_CONSTRAINT = "Language: Please use the same language as Human INPUT." -FORMAT_CONSTRAINT = f"Format: output wrapped inside [{TAG}][/{TAG}] like format example, nothing else." +FORMAT_CONSTRAINT = (f"Format: output wrapped inside [{TAG}][/{TAG}] like format example, nothing else. " + f"Delete comments in json") # Delete comments in json # If you don't want JSONDecodeError to occur, you can add Delete comments in json after FORMAT_CONSTRAINT From f16b24758692bd10d89069c13a1260f9d15c968c Mon Sep 17 00:00:00 2001 From: huzixia <528543747@qq.com> Date: Sat, 27 Jan 2024 15:32:12 +0800 Subject: [PATCH 494/637] merge code with similar logic to avoid duplication --- ...move comments in output json str, after js | 12 + PR/action_node.py | 351 ++++++++++++++++++ PR/repair_llm_raw_output.py | 351 ++++++++++++++++++ metagpt/utils/repair_llm_raw_output.py | 43 ++- 4 files changed, 735 insertions(+), 22 deletions(-) create mode 100644 PR/# remove comments in output json str, after js create mode 100644 PR/action_node.py create mode 100644 PR/repair_llm_raw_output.py diff --git a/PR/# remove comments in output json str, after js b/PR/# remove comments in output json str, after js new file mode 100644 index 000000000..f795fefdb --- /dev/null +++ b/PR/# remove comments in output json str, after js @@ -0,0 +1,12 @@ + +git commit -m "To avoid JSONDecodeError: " -m "Remove comments in output json str, after json value content, maybe start with #, maybe start with //, particularly, it is not inside the string value" -m "Addtionly, if you do not want JSONDecodeError to occur, you can add 'Delete comments in json' after FORMAT_CONSTRAINT in action_node.py" + + + +git commit -m "Addtionly, if you do not want JSONDecodeError to occur, you can add 'Delete comments in json' after FORMAT_CONSTRAINT in action_node.py" + + + + + + diff --git a/PR/action_node.py b/PR/action_node.py new file mode 100644 index 000000000..0f441cfee --- /dev/null +++ b/PR/action_node.py @@ -0,0 +1,351 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2023/12/11 18:45 +@Author : alexanderwu +@File : action_node.py + +NOTE: You should use typing.List instead of list to do type annotation. Because in the markdown extraction process, + we can use typing to extract the type of the node, but we cannot use built-in list to extract. +""" +import json +from typing import Any, Dict, List, Optional, Tuple, Type + +from pydantic import BaseModel, create_model, model_validator +from tenacity import retry, stop_after_attempt, wait_random_exponential + +from metagpt.config import CONFIG +from metagpt.llm import BaseLLM +from metagpt.logs import logger +from metagpt.provider.postprocess.llm_output_postprocess import llm_output_postprocess +from metagpt.utils.common import OutputParser, general_after_log + +TAG = "CONTENT" + +LANGUAGE_CONSTRAINT = "Language: Please use the same language as Human INPUT." +FORMAT_CONSTRAINT = f"Format: output wrapped inside [{TAG}][/{TAG}] like format example, nothing else." +# Delete comments in json +# If you don't want JSONDecodeError to occur, you can add Delete comments in json after FORMAT_CONSTRAINT + + +SIMPLE_TEMPLATE = """ +## context +{context} + +----- + +## format example +{example} + +## nodes: ": # " +{instruction} + +## constraint +{constraint} + +## action +Follow instructions of nodes, generate output and make sure it follows the format example. +""" + + +def dict_to_markdown(d, prefix="- ", kv_sep="\n", postfix="\n"): + markdown_str = "" + for key, value in d.items(): + markdown_str += f"{prefix}{key}{kv_sep}{value}{postfix}" + return markdown_str + + +class ActionNode: + """ActionNode is a tree of nodes.""" + + schema: str # raw/json/markdown, default: "" + + # Action Context + context: str # all the context, including all necessary info + llm: BaseLLM # LLM with aask interface + children: dict[str, "ActionNode"] + + # Action Input + key: str # Product Requirement / File list / Code + expected_type: Type # such as str / int / float etc. + # context: str # everything in the history. + instruction: str # the instructions should be followed. + example: Any # example for In Context-Learning. + + # Action Output + content: str + instruct_content: BaseModel + + def __init__( + self, + key: str, + expected_type: Type, + instruction: str, + example: Any, + content: str = "", + children: dict[str, "ActionNode"] = None, + schema: str = "", + ): + self.key = key + self.expected_type = expected_type + self.instruction = instruction + self.example = example + self.content = content + self.children = children if children is not None else {} + self.schema = schema + + def __str__(self): + return ( + f"{self.key}, {repr(self.expected_type)}, {self.instruction}, {self.example}" + f", {self.content}, {self.children}" + ) + + def __repr__(self): + return self.__str__() + + def add_child(self, node: "ActionNode"): + """增加子ActionNode""" + self.children[node.key] = node + + def add_children(self, nodes: List["ActionNode"]): + """批量增加子ActionNode""" + for node in nodes: + self.add_child(node) + + @classmethod + def from_children(cls, key, nodes: List["ActionNode"]): + """直接从一系列的子nodes初始化""" + obj = cls(key, str, "", "") + obj.add_children(nodes) + return obj + + def get_children_mapping(self, exclude=None) -> Dict[str, Tuple[Type, Any]]: + """获得子ActionNode的字典,以key索引""" + exclude = exclude or [] + return {k: (v.expected_type, ...) for k, v in self.children.items() if k not in exclude} + + def get_self_mapping(self) -> Dict[str, Tuple[Type, Any]]: + """get self key: type mapping""" + return {self.key: (self.expected_type, ...)} + + def get_mapping(self, mode="children", exclude=None) -> Dict[str, Tuple[Type, Any]]: + """get key: type mapping under mode""" + if mode == "children" or (mode == "auto" and self.children): + return self.get_children_mapping(exclude=exclude) + return {} if exclude and self.key in exclude else self.get_self_mapping() + + @classmethod + def create_model_class(cls, class_name: str, mapping: Dict[str, Tuple[Type, Any]]): + """基于pydantic v1的模型动态生成,用来检验结果类型正确性""" + + def check_fields(cls, values): + required_fields = set(mapping.keys()) + missing_fields = required_fields - set(values.keys()) + if missing_fields: + raise ValueError(f"Missing fields: {missing_fields}") + + unrecognized_fields = set(values.keys()) - required_fields + if unrecognized_fields: + logger.warning(f"Unrecognized fields: {unrecognized_fields}") + return values + + validators = {"check_missing_fields_validator": model_validator(mode="before")(check_fields)} + + new_class = create_model(class_name, __validators__=validators, **mapping) + return new_class + + def create_children_class(self, exclude=None): + """使用object内有的字段直接生成model_class""" + class_name = f"{self.key}_AN" + mapping = self.get_children_mapping(exclude=exclude) + return self.create_model_class(class_name, mapping) + + def to_dict(self, format_func=None, mode="auto", exclude=None) -> Dict: + """将当前节点与子节点都按照node: format的格式组织成字典""" + + # 如果没有提供格式化函数,使用默认的格式化方式 + if format_func is None: + format_func = lambda node: f"{node.instruction}" + + # 使用提供的格式化函数来格式化当前节点的值 + formatted_value = format_func(self) + + # 创建当前节点的键值对 + if mode == "children" or (mode == "auto" and self.children): + node_dict = {} + else: + node_dict = {self.key: formatted_value} + + if mode == "root": + return node_dict + + # 遍历子节点并递归调用 to_dict 方法 + exclude = exclude or [] + for _, child_node in self.children.items(): + if child_node.key in exclude: + continue + node_dict.update(child_node.to_dict(format_func)) + + return node_dict + + def compile_to(self, i: Dict, schema, kv_sep) -> str: + if schema == "json": + return json.dumps(i, indent=4) + elif schema == "markdown": + return dict_to_markdown(i, kv_sep=kv_sep) + else: + return str(i) + + def tagging(self, text, schema, tag="") -> str: + if not tag: + return text + if schema == "json": + return f"[{tag}]\n" + text + f"\n[/{tag}]" + else: # markdown + return f"[{tag}]\n" + text + f"\n[/{tag}]" + + def _compile_f(self, schema, mode, tag, format_func, kv_sep, exclude=None) -> str: + nodes = self.to_dict(format_func=format_func, mode=mode, exclude=exclude) + text = self.compile_to(nodes, schema, kv_sep) + return self.tagging(text, schema, tag) + + def compile_instruction(self, schema="markdown", mode="children", tag="", exclude=None) -> str: + """compile to raw/json/markdown template with all/root/children nodes""" + format_func = lambda i: f"{i.expected_type} # {i.instruction}" + return self._compile_f(schema, mode, tag, format_func, kv_sep=": ", exclude=exclude) + + def compile_example(self, schema="json", mode="children", tag="", exclude=None) -> str: + """compile to raw/json/markdown examples with all/root/children nodes""" + + # 这里不能使用f-string,因为转译为str后再json.dumps会额外加上引号,无法作为有效的example + # 错误示例:"File list": "['main.py', 'const.py', 'game.py']", 注意这里值不是list,而是str + format_func = lambda i: i.example + return self._compile_f(schema, mode, tag, format_func, kv_sep="\n", exclude=exclude) + + def compile(self, context, schema="json", mode="children", template=SIMPLE_TEMPLATE, exclude=[]) -> str: + """ + mode: all/root/children + mode="children": 编译所有子节点为一个统一模板,包括instruction与example + mode="all": NotImplemented + mode="root": NotImplemented + schmea: raw/json/markdown + schema="raw": 不编译,context, lang_constaint, instruction + schema="json":编译context, example(json), instruction(markdown), constraint, action + schema="markdown": 编译context, example(markdown), instruction(markdown), constraint, action + """ + if schema == "raw": + return context + "\n\n## Actions\n" + LANGUAGE_CONSTRAINT + "\n" + self.instruction + + # FIXME: json instruction会带来格式问题,如:"Project name": "web_2048 # 项目名称使用下划线", + # compile example暂时不支持markdown + instruction = self.compile_instruction(schema="markdown", mode=mode, exclude=exclude) + example = self.compile_example(schema=schema, tag=TAG, mode=mode, exclude=exclude) + # nodes = ", ".join(self.to_dict(mode=mode).keys()) + constraints = [LANGUAGE_CONSTRAINT, FORMAT_CONSTRAINT] + constraint = "\n".join(constraints) + + prompt = template.format( + context=context, + example=example, + instruction=instruction, + constraint=constraint, + ) + return prompt + + @retry( + wait=wait_random_exponential(min=1, max=20), + stop=stop_after_attempt(6), + after=general_after_log(logger), + ) + async def _aask_v1( + self, + prompt: str, + output_class_name: str, + output_data_mapping: dict, + system_msgs: Optional[list[str]] = None, + schema="markdown", # compatible to original format + timeout=CONFIG.timeout, + ) -> (str, BaseModel): + """Use ActionOutput to wrap the output of aask""" + content = await self.llm.aask(prompt, system_msgs, timeout=timeout) + logger.debug(f"llm raw output:\n{content}") + output_class = self.create_model_class(output_class_name, output_data_mapping) + + if schema == "json": + parsed_data = llm_output_postprocess( + output=content, schema=output_class.model_json_schema(), req_key=f"[/{TAG}]" + ) + else: # using markdown parser + parsed_data = OutputParser.parse_data_with_mapping(content, output_data_mapping) + + logger.debug(f"parsed_data:\n{parsed_data}") + instruct_content = output_class(**parsed_data) + return content, instruct_content + + def get(self, key): + return self.instruct_content.model_dump()[key] + + def set_recursive(self, name, value): + setattr(self, name, value) + for _, i in self.children.items(): + i.set_recursive(name, value) + + def set_llm(self, llm): + self.set_recursive("llm", llm) + + def set_context(self, context): + self.set_recursive("context", context) + + async def simple_fill(self, schema, mode, timeout=CONFIG.timeout, exclude=None): + prompt = self.compile(context=self.context, schema=schema, mode=mode, exclude=exclude) + + if schema != "raw": + mapping = self.get_mapping(mode, exclude=exclude) + class_name = f"{self.key}_AN" + content, scontent = await self._aask_v1(prompt, class_name, mapping, schema=schema, timeout=timeout) + self.content = content + self.instruct_content = scontent + else: + self.content = await self.llm.aask(prompt) + self.instruct_content = None + + return self + + async def fill(self, context, llm, schema="json", mode="auto", strgy="simple", timeout=CONFIG.timeout, exclude=[]): + """Fill the node(s) with mode. + + :param context: Everything we should know when filling node. + :param llm: Large Language Model with pre-defined system message. + :param schema: json/markdown, determine example and output format. + - raw: free form text + - json: it's easy to open source LLM with json format + - markdown: when generating code, markdown is always better + :param mode: auto/children/root + - auto: automated fill children's nodes and gather outputs, if no children, fill itself + - children: fill children's nodes and gather outputs + - root: fill root's node and gather output + :param strgy: simple/complex + - simple: run only once + - complex: run each node + :param timeout: Timeout for llm invocation. + :param exclude: The keys of ActionNode to exclude. + :return: self + """ + self.set_llm(llm) + self.set_context(context) + if self.schema: + schema = self.schema + + if strgy == "simple": + return await self.simple_fill(schema=schema, mode=mode, timeout=timeout, exclude=exclude) + elif strgy == "complex": + # 这里隐式假设了拥有children + tmp = {} + for _, i in self.children.items(): + if exclude and i.key in exclude: + continue + child = await i.simple_fill(schema=schema, mode=mode, timeout=timeout, exclude=exclude) + tmp.update(child.instruct_content.dict()) + cls = self.create_children_class() + self.instruct_content = cls(**tmp) + return self diff --git a/PR/repair_llm_raw_output.py b/PR/repair_llm_raw_output.py new file mode 100644 index 000000000..4995918c2 --- /dev/null +++ b/PR/repair_llm_raw_output.py @@ -0,0 +1,351 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : repair llm raw output with particular conditions + +import copy +from enum import Enum +from typing import Callable, Union + +import regex as re +from tenacity import RetryCallState, retry, stop_after_attempt, wait_fixed + +from metagpt.config import CONFIG +from metagpt.logs import logger +from metagpt.utils.custom_decoder import CustomDecoder + + +class RepairType(Enum): + CS = "case sensitivity" + RKPM = "required key pair missing" # condition like `[key] xx` which lacks `[/key]` + SCM = "special character missing" # Usually the req_key appear in pairs like `[key] xx [/key]` + JSON = "json format" + + +def repair_case_sensitivity(output: str, req_key: str) -> str: + """ + usually, req_key is the key name of expected json or markdown content, it won't appear in the value part. + fix target string `"Shared Knowledge": ""` but `"Shared knowledge": ""` actually + """ + if req_key in output: + return output + + output_lower = output.lower() + req_key_lower = req_key.lower() + if req_key_lower in output_lower: + # find the sub-part index, and replace it with raw req_key + lidx = output_lower.find(req_key_lower) + source = output[lidx : lidx + len(req_key_lower)] + output = output.replace(source, req_key) + logger.info(f"repair_case_sensitivity: {req_key}") + + return output + + +def repair_special_character_missing(output: str, req_key: str = "[/CONTENT]") -> str: + """ + fix + 1. target string `[CONTENT] xx [CONTENT] xxx [CONTENT]` lacks `/` in the last `[CONTENT]` + 2. target string `xx [CONTENT] xxx [CONTENT] xxxx` lacks `/` in the last `[CONTENT]` + """ + sc_arr = ["/"] + + if req_key in output: + return output + + for sc in sc_arr: + req_key_pure = req_key.replace(sc, "") + appear_cnt = output.count(req_key_pure) + if req_key_pure in output and appear_cnt > 1: + # req_key with special_character usually in the tail side + ridx = output.rfind(req_key_pure) + output = f"{output[:ridx]}{req_key}{output[ridx + len(req_key_pure):]}" + logger.info(f"repair_special_character_missing: {sc} in {req_key_pure} as position {ridx}") + + return output + + +def repair_required_key_pair_missing(output: str, req_key: str = "[/CONTENT]") -> str: + """ + implement the req_key pair in the begin or end of the content + req_key format + 1. `[req_key]`, and its pair `[/req_key]` + 2. `[/req_key]`, and its pair `[req_key]` + """ + sc = "/" # special char + if req_key.startswith("[") and req_key.endswith("]"): + if sc in req_key: + left_key = req_key.replace(sc, "") # `[/req_key]` -> `[req_key]` + right_key = req_key + else: + left_key = req_key + right_key = f"{req_key[0]}{sc}{req_key[1:]}" # `[req_key]` -> `[/req_key]` + + if left_key not in output: + output = left_key + "\n" + output + if right_key not in output: + + def judge_potential_json(routput: str, left_key: str) -> Union[str, None]: + ridx = routput.rfind(left_key) + if ridx < 0: + return None + sub_output = routput[ridx:] + idx1 = sub_output.rfind("}") + idx2 = sub_output.rindex("]") + idx = idx1 if idx1 >= idx2 else idx2 + sub_output = sub_output[: idx + 1] + return sub_output + + if output.strip().endswith("}") or (output.strip().endswith("]") and not output.strip().endswith(left_key)): + # # avoid [req_key]xx[req_key] case to append [/req_key] + output = output + "\n" + right_key + elif judge_potential_json(output, left_key) and (not output.strip().endswith(left_key)): + sub_content = judge_potential_json(output, left_key) + output = sub_content + "\n" + right_key + + return output + + +def repair_json_format(output: str) -> str: + """ + fix extra `[` or `}` in the end + """ + output = output.strip() + + if output.startswith("[{"): + output = output[1:] + logger.info(f"repair_json_format: {'[{'}") + elif output.endswith("}]"): + output = output[:-1] + logger.info(f"repair_json_format: {'}]'}") + elif output.startswith("{") and output.endswith("]"): + output = output[:-1] + "}" + + # remove comments in output json str, after json value content, maybe start with #, maybe start with // + arr = output.split("\n") + new_arr = [] + for line in arr: + # look for # or // comments and make sure they are not inside the string value + comment_index = -1 + for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): + if match.group(1): # if the string value + continue + if match.group(2): # if comments + comment_index = match.start(2) + break + # if comments, then delete them + if comment_index != -1: + line = line[:comment_index].rstrip() + new_arr.append(line) + output = "\n".join(new_arr) + return output + + +def _repair_llm_raw_output(output: str, req_key: str, repair_type: RepairType = None) -> str: + repair_types = [repair_type] if repair_type else [item for item in RepairType if item not in [RepairType.JSON]] + for repair_type in repair_types: + if repair_type == RepairType.CS: + output = repair_case_sensitivity(output, req_key) + elif repair_type == RepairType.RKPM: + output = repair_required_key_pair_missing(output, req_key) + elif repair_type == RepairType.SCM: + output = repair_special_character_missing(output, req_key) + elif repair_type == RepairType.JSON: + output = repair_json_format(output) + return output + + +def repair_llm_raw_output(output: str, req_keys: list[str], repair_type: RepairType = None) -> str: + """ + in open-source llm model, it usually can't follow the instruction well, the output may be incomplete, + so here we try to repair it and use all repair methods by default. + typical case + 1. case sensitivity + target: "Original Requirements" + output: "Original requirements" + 2. special character missing + target: [/CONTENT] + output: [CONTENT] + 3. json format + target: { xxx } + output: { xxx }] + """ + if not CONFIG.repair_llm_output: + return output + + # do the repairation usually for non-openai models + for req_key in req_keys: + output = _repair_llm_raw_output(output=output, req_key=req_key, repair_type=repair_type) + return output + + +def repair_invalid_json(output: str, error: str) -> str: + """ + repair the situation like there are extra chars like + error examples + example 1. json.decoder.JSONDecodeError: Expecting ',' delimiter: line 154 column 1 (char 2765) + example 2. xxx.JSONDecodeError: Expecting property name enclosed in double quotes: line 14 column 1 (char 266) + """ + pattern = r"line ([0-9]+) column ([0-9]+)" + + matches = re.findall(pattern, error, re.DOTALL) + if len(matches) > 0: + line_no = int(matches[0][0]) - 1 + col_no = int(matches[0][1]) - 1 + + # due to CustomDecoder can handle `"": ''` or `'': ""`, so convert `"""` -> `"`, `'''` -> `'` + output = output.replace('"""', '"').replace("'''", '"') + arr = output.split("\n") + rline = arr[line_no] # raw line + line = arr[line_no].strip() + # different general problems + if line.endswith("],"): + # problem, redundant char `]` + new_line = line.replace("]", "") + elif line.endswith("},") and not output.endswith("},"): + # problem, redundant char `}` + new_line = line.replace("}", "") + elif line.endswith("},") and output.endswith("},"): + new_line = line[:-1] + # remove comments in output json str, after json value content, maybe start with #, maybe start with // + elif rline[col_no] == "#" or rline[col_no] == "/": + new_line = rline[:col_no] + for i in range(line_no + 1, len(arr)): + # look for # or // comments and make sure they are not inside the string value + comment_index = -1 + for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): + if match.group(1): # if the string value + continue + if match.group(2): # if comments + comment_index = match.start(2) + break + # if comments, then delete them + if comment_index != -1: + arr[i] = arr[i][:comment_index].rstrip() + elif (rline[col_no] in ["'", '"']) and (line.startswith('"') or line.startswith("'")) and "," not in line: + # problem, `"""` or `'''` without `,` + new_line = f",{line}" + elif '",' not in line and "," not in line and '"' not in line: + new_line = f'{line}",' + elif not line.endswith(","): + # problem, miss char `,` at the end. + new_line = f"{line}," + elif "," in line and len(line) == 1: + new_line = f'"{line}' + elif '",' in line: + new_line = line[:-2] + "'," + else: + new_line = line + + arr[line_no] = new_line + output = "\n".join(arr) + logger.info(f"repair_invalid_json, raw error: {error}") + + return output + + +def run_after_exp_and_passon_next_retry(logger: "loguru.Logger") -> Callable[["RetryCallState"], None]: + def run_and_passon(retry_state: RetryCallState) -> None: + """ + RetryCallState example + { + "start_time":143.098322024, + "retry_object":")>", + "fn":"", + "args":"(\"tag:[/CONTENT]\",)", # function input args + "kwargs":{}, # function input kwargs + "attempt_number":1, # retry number + "outcome":"", # type(outcome.result()) = "str", type(outcome.exception()) = "class" + "outcome_timestamp":143.098416904, + "idle_for":0, + "next_action":"None" + } + """ + if retry_state.outcome.failed: + if retry_state.args: + # # can't be used as args=retry_state.args + func_param_output = retry_state.args[0] + elif retry_state.kwargs: + func_param_output = retry_state.kwargs.get("output", "") + exp_str = str(retry_state.outcome.exception()) + + fix_str = "try to fix it, " if CONFIG.repair_llm_output else "" + logger.warning( + f"parse json from content inside [CONTENT][/CONTENT] failed at retry " + f"{retry_state.attempt_number}, {fix_str}exp: {exp_str}" + ) + + repaired_output = repair_invalid_json(func_param_output, exp_str) + retry_state.kwargs["output"] = repaired_output + + return run_and_passon + + +@retry( + stop=stop_after_attempt(3 if CONFIG.repair_llm_output else 0), + wait=wait_fixed(1), + after=run_after_exp_and_passon_next_retry(logger), +) +def retry_parse_json_text(output: str) -> Union[list, dict]: + """ + repair the json-text situation like there are extra chars like [']', '}'] + + Warning + if CONFIG.repair_llm_output is False, retry _aask_v1 {x=3} times, and the retry_parse_json_text's retry not work + if CONFIG.repair_llm_output is True, the _aask_v1 and the retry_parse_json_text will loop for {x=3*3} times. + it's a two-layer retry cycle + """ + # logger.debug(f"output to json decode:\n{output}") + + # if CONFIG.repair_llm_output is True, it will try to fix output until the retry break + parsed_data = CustomDecoder(strict=False).decode(output) + + return parsed_data + + +def extract_content_from_output(content: str, right_key: str = "[/CONTENT]"): + """extract xxx from [CONTENT](xxx)[/CONTENT] using regex pattern""" + + def re_extract_content(cont: str, pattern: str) -> str: + matches = re.findall(pattern, cont, re.DOTALL) + for match in matches: + if match: + cont = match + break + return cont.strip() + + # TODO construct the extract pattern with the `right_key` + raw_content = copy.deepcopy(content) + pattern = r"\[CONTENT\]([\s\S]*)\[/CONTENT\]" + new_content = re_extract_content(raw_content, pattern) + + if not new_content.startswith("{"): + # TODO find a more general pattern + # # for `[CONTENT]xxx[CONTENT]xxxx[/CONTENT] situation + logger.warning(f"extract_content try another pattern: {pattern}") + if right_key not in new_content: + raw_content = copy.deepcopy(new_content + "\n" + right_key) + # # pattern = r"\[CONTENT\](\s*\{.*?\}\s*)\[/CONTENT\]" + new_content = re_extract_content(raw_content, pattern) + else: + if right_key in new_content: + idx = new_content.find(right_key) + new_content = new_content[:idx] + new_content = new_content.strip() + + return new_content + + +def extract_state_value_from_output(content: str) -> str: + """ + For openai models, they will always return state number. But for open llm models, the instruction result maybe a + long text contain target number, so here add a extraction to improve success rate. + + Args: + content (str): llm's output from `Role._think` + """ + content = content.strip() # deal the output cases like " 0", "0\n" and so on. + pattern = r"([0-9])" # TODO find the number using a more proper method not just extract from content using pattern + matches = re.findall(pattern, content, re.DOTALL) + matches = list(set(matches)) + state = matches[0] if len(matches) > 0 else "-1" + return state diff --git a/metagpt/utils/repair_llm_raw_output.py b/metagpt/utils/repair_llm_raw_output.py index 4995918c2..ef3580750 100644 --- a/metagpt/utils/repair_llm_raw_output.py +++ b/metagpt/utils/repair_llm_raw_output.py @@ -105,6 +105,23 @@ def repair_required_key_pair_missing(output: str, req_key: str = "[/CONTENT]") - return output +def remove_comments_from_line(line): + """ + Remove comments from a single line of string. + Comments are assumed to start with '#' or '//' and are not inside string values. + """ + comment_index = -1 + for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): + if match.group(1): # if the string value + continue + if match.group(2): # if comments + comment_index = match.start(2) + break + if comment_index != -1: # if comments, then delete them + return line[:comment_index].rstrip() + return line + + def repair_json_format(output: str) -> str: """ fix extra `[` or `}` in the end @@ -125,17 +142,8 @@ def repair_json_format(output: str) -> str: new_arr = [] for line in arr: # look for # or // comments and make sure they are not inside the string value - comment_index = -1 - for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): - if match.group(1): # if the string value - continue - if match.group(2): # if comments - comment_index = match.start(2) - break - # if comments, then delete them - if comment_index != -1: - line = line[:comment_index].rstrip() - new_arr.append(line) + new_line = remove_comments_from_line(line) + new_arr.append(new_line) output = "\n".join(new_arr) return output @@ -209,18 +217,9 @@ def repair_invalid_json(output: str, error: str) -> str: # remove comments in output json str, after json value content, maybe start with #, maybe start with // elif rline[col_no] == "#" or rline[col_no] == "/": new_line = rline[:col_no] + # check the next line and remove the comments for i in range(line_no + 1, len(arr)): - # look for # or // comments and make sure they are not inside the string value - comment_index = -1 - for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): - if match.group(1): # if the string value - continue - if match.group(2): # if comments - comment_index = match.start(2) - break - # if comments, then delete them - if comment_index != -1: - arr[i] = arr[i][:comment_index].rstrip() + arr[i] = remove_comments_from_line(arr[i]) elif (rline[col_no] in ["'", '"']) and (line.startswith('"') or line.startswith("'")) and "," not in line: # problem, `"""` or `'''` without `,` new_line = f",{line}" From 8b5f7848fa6aa8c9e0703dcdbf6760ef1efa87eb Mon Sep 17 00:00:00 2001 From: huzixia <528543747@qq.com> Date: Sat, 27 Jan 2024 17:00:59 +0800 Subject: [PATCH 495/637] delete PR dir --- metagpt/utils/repair_llm_raw_output.py | 43 +++++++++++++------------- 1 file changed, 21 insertions(+), 22 deletions(-) diff --git a/metagpt/utils/repair_llm_raw_output.py b/metagpt/utils/repair_llm_raw_output.py index 4995918c2..ef3580750 100644 --- a/metagpt/utils/repair_llm_raw_output.py +++ b/metagpt/utils/repair_llm_raw_output.py @@ -105,6 +105,23 @@ def repair_required_key_pair_missing(output: str, req_key: str = "[/CONTENT]") - return output +def remove_comments_from_line(line): + """ + Remove comments from a single line of string. + Comments are assumed to start with '#' or '//' and are not inside string values. + """ + comment_index = -1 + for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): + if match.group(1): # if the string value + continue + if match.group(2): # if comments + comment_index = match.start(2) + break + if comment_index != -1: # if comments, then delete them + return line[:comment_index].rstrip() + return line + + def repair_json_format(output: str) -> str: """ fix extra `[` or `}` in the end @@ -125,17 +142,8 @@ def repair_json_format(output: str) -> str: new_arr = [] for line in arr: # look for # or // comments and make sure they are not inside the string value - comment_index = -1 - for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): - if match.group(1): # if the string value - continue - if match.group(2): # if comments - comment_index = match.start(2) - break - # if comments, then delete them - if comment_index != -1: - line = line[:comment_index].rstrip() - new_arr.append(line) + new_line = remove_comments_from_line(line) + new_arr.append(new_line) output = "\n".join(new_arr) return output @@ -209,18 +217,9 @@ def repair_invalid_json(output: str, error: str) -> str: # remove comments in output json str, after json value content, maybe start with #, maybe start with // elif rline[col_no] == "#" or rline[col_no] == "/": new_line = rline[:col_no] + # check the next line and remove the comments for i in range(line_no + 1, len(arr)): - # look for # or // comments and make sure they are not inside the string value - comment_index = -1 - for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): - if match.group(1): # if the string value - continue - if match.group(2): # if comments - comment_index = match.start(2) - break - # if comments, then delete them - if comment_index != -1: - arr[i] = arr[i][:comment_index].rstrip() + arr[i] = remove_comments_from_line(arr[i]) elif (rline[col_no] in ["'", '"']) and (line.startswith('"') or line.startswith("'")) and "," not in line: # problem, `"""` or `'''` without `,` new_line = f",{line}" From 2361c7e8aa2df55bd3562243e95b4d5f538188ff Mon Sep 17 00:00:00 2001 From: huzixia <528543747@qq.com> Date: Sat, 27 Jan 2024 17:07:21 +0800 Subject: [PATCH 496/637] delete PR dir --- ...move comments in output json str, after js | 12 - PR/action_node.py | 351 ------------------ PR/repair_llm_raw_output.py | 351 ------------------ 3 files changed, 714 deletions(-) delete mode 100644 PR/# remove comments in output json str, after js delete mode 100644 PR/action_node.py delete mode 100644 PR/repair_llm_raw_output.py diff --git a/PR/# remove comments in output json str, after js b/PR/# remove comments in output json str, after js deleted file mode 100644 index f795fefdb..000000000 --- a/PR/# remove comments in output json str, after js +++ /dev/null @@ -1,12 +0,0 @@ - -git commit -m "To avoid JSONDecodeError: " -m "Remove comments in output json str, after json value content, maybe start with #, maybe start with //, particularly, it is not inside the string value" -m "Addtionly, if you do not want JSONDecodeError to occur, you can add 'Delete comments in json' after FORMAT_CONSTRAINT in action_node.py" - - - -git commit -m "Addtionly, if you do not want JSONDecodeError to occur, you can add 'Delete comments in json' after FORMAT_CONSTRAINT in action_node.py" - - - - - - diff --git a/PR/action_node.py b/PR/action_node.py deleted file mode 100644 index 0f441cfee..000000000 --- a/PR/action_node.py +++ /dev/null @@ -1,351 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -""" -@Time : 2023/12/11 18:45 -@Author : alexanderwu -@File : action_node.py - -NOTE: You should use typing.List instead of list to do type annotation. Because in the markdown extraction process, - we can use typing to extract the type of the node, but we cannot use built-in list to extract. -""" -import json -from typing import Any, Dict, List, Optional, Tuple, Type - -from pydantic import BaseModel, create_model, model_validator -from tenacity import retry, stop_after_attempt, wait_random_exponential - -from metagpt.config import CONFIG -from metagpt.llm import BaseLLM -from metagpt.logs import logger -from metagpt.provider.postprocess.llm_output_postprocess import llm_output_postprocess -from metagpt.utils.common import OutputParser, general_after_log - -TAG = "CONTENT" - -LANGUAGE_CONSTRAINT = "Language: Please use the same language as Human INPUT." -FORMAT_CONSTRAINT = f"Format: output wrapped inside [{TAG}][/{TAG}] like format example, nothing else." -# Delete comments in json -# If you don't want JSONDecodeError to occur, you can add Delete comments in json after FORMAT_CONSTRAINT - - -SIMPLE_TEMPLATE = """ -## context -{context} - ------ - -## format example -{example} - -## nodes: ": # " -{instruction} - -## constraint -{constraint} - -## action -Follow instructions of nodes, generate output and make sure it follows the format example. -""" - - -def dict_to_markdown(d, prefix="- ", kv_sep="\n", postfix="\n"): - markdown_str = "" - for key, value in d.items(): - markdown_str += f"{prefix}{key}{kv_sep}{value}{postfix}" - return markdown_str - - -class ActionNode: - """ActionNode is a tree of nodes.""" - - schema: str # raw/json/markdown, default: "" - - # Action Context - context: str # all the context, including all necessary info - llm: BaseLLM # LLM with aask interface - children: dict[str, "ActionNode"] - - # Action Input - key: str # Product Requirement / File list / Code - expected_type: Type # such as str / int / float etc. - # context: str # everything in the history. - instruction: str # the instructions should be followed. - example: Any # example for In Context-Learning. - - # Action Output - content: str - instruct_content: BaseModel - - def __init__( - self, - key: str, - expected_type: Type, - instruction: str, - example: Any, - content: str = "", - children: dict[str, "ActionNode"] = None, - schema: str = "", - ): - self.key = key - self.expected_type = expected_type - self.instruction = instruction - self.example = example - self.content = content - self.children = children if children is not None else {} - self.schema = schema - - def __str__(self): - return ( - f"{self.key}, {repr(self.expected_type)}, {self.instruction}, {self.example}" - f", {self.content}, {self.children}" - ) - - def __repr__(self): - return self.__str__() - - def add_child(self, node: "ActionNode"): - """增加子ActionNode""" - self.children[node.key] = node - - def add_children(self, nodes: List["ActionNode"]): - """批量增加子ActionNode""" - for node in nodes: - self.add_child(node) - - @classmethod - def from_children(cls, key, nodes: List["ActionNode"]): - """直接从一系列的子nodes初始化""" - obj = cls(key, str, "", "") - obj.add_children(nodes) - return obj - - def get_children_mapping(self, exclude=None) -> Dict[str, Tuple[Type, Any]]: - """获得子ActionNode的字典,以key索引""" - exclude = exclude or [] - return {k: (v.expected_type, ...) for k, v in self.children.items() if k not in exclude} - - def get_self_mapping(self) -> Dict[str, Tuple[Type, Any]]: - """get self key: type mapping""" - return {self.key: (self.expected_type, ...)} - - def get_mapping(self, mode="children", exclude=None) -> Dict[str, Tuple[Type, Any]]: - """get key: type mapping under mode""" - if mode == "children" or (mode == "auto" and self.children): - return self.get_children_mapping(exclude=exclude) - return {} if exclude and self.key in exclude else self.get_self_mapping() - - @classmethod - def create_model_class(cls, class_name: str, mapping: Dict[str, Tuple[Type, Any]]): - """基于pydantic v1的模型动态生成,用来检验结果类型正确性""" - - def check_fields(cls, values): - required_fields = set(mapping.keys()) - missing_fields = required_fields - set(values.keys()) - if missing_fields: - raise ValueError(f"Missing fields: {missing_fields}") - - unrecognized_fields = set(values.keys()) - required_fields - if unrecognized_fields: - logger.warning(f"Unrecognized fields: {unrecognized_fields}") - return values - - validators = {"check_missing_fields_validator": model_validator(mode="before")(check_fields)} - - new_class = create_model(class_name, __validators__=validators, **mapping) - return new_class - - def create_children_class(self, exclude=None): - """使用object内有的字段直接生成model_class""" - class_name = f"{self.key}_AN" - mapping = self.get_children_mapping(exclude=exclude) - return self.create_model_class(class_name, mapping) - - def to_dict(self, format_func=None, mode="auto", exclude=None) -> Dict: - """将当前节点与子节点都按照node: format的格式组织成字典""" - - # 如果没有提供格式化函数,使用默认的格式化方式 - if format_func is None: - format_func = lambda node: f"{node.instruction}" - - # 使用提供的格式化函数来格式化当前节点的值 - formatted_value = format_func(self) - - # 创建当前节点的键值对 - if mode == "children" or (mode == "auto" and self.children): - node_dict = {} - else: - node_dict = {self.key: formatted_value} - - if mode == "root": - return node_dict - - # 遍历子节点并递归调用 to_dict 方法 - exclude = exclude or [] - for _, child_node in self.children.items(): - if child_node.key in exclude: - continue - node_dict.update(child_node.to_dict(format_func)) - - return node_dict - - def compile_to(self, i: Dict, schema, kv_sep) -> str: - if schema == "json": - return json.dumps(i, indent=4) - elif schema == "markdown": - return dict_to_markdown(i, kv_sep=kv_sep) - else: - return str(i) - - def tagging(self, text, schema, tag="") -> str: - if not tag: - return text - if schema == "json": - return f"[{tag}]\n" + text + f"\n[/{tag}]" - else: # markdown - return f"[{tag}]\n" + text + f"\n[/{tag}]" - - def _compile_f(self, schema, mode, tag, format_func, kv_sep, exclude=None) -> str: - nodes = self.to_dict(format_func=format_func, mode=mode, exclude=exclude) - text = self.compile_to(nodes, schema, kv_sep) - return self.tagging(text, schema, tag) - - def compile_instruction(self, schema="markdown", mode="children", tag="", exclude=None) -> str: - """compile to raw/json/markdown template with all/root/children nodes""" - format_func = lambda i: f"{i.expected_type} # {i.instruction}" - return self._compile_f(schema, mode, tag, format_func, kv_sep=": ", exclude=exclude) - - def compile_example(self, schema="json", mode="children", tag="", exclude=None) -> str: - """compile to raw/json/markdown examples with all/root/children nodes""" - - # 这里不能使用f-string,因为转译为str后再json.dumps会额外加上引号,无法作为有效的example - # 错误示例:"File list": "['main.py', 'const.py', 'game.py']", 注意这里值不是list,而是str - format_func = lambda i: i.example - return self._compile_f(schema, mode, tag, format_func, kv_sep="\n", exclude=exclude) - - def compile(self, context, schema="json", mode="children", template=SIMPLE_TEMPLATE, exclude=[]) -> str: - """ - mode: all/root/children - mode="children": 编译所有子节点为一个统一模板,包括instruction与example - mode="all": NotImplemented - mode="root": NotImplemented - schmea: raw/json/markdown - schema="raw": 不编译,context, lang_constaint, instruction - schema="json":编译context, example(json), instruction(markdown), constraint, action - schema="markdown": 编译context, example(markdown), instruction(markdown), constraint, action - """ - if schema == "raw": - return context + "\n\n## Actions\n" + LANGUAGE_CONSTRAINT + "\n" + self.instruction - - # FIXME: json instruction会带来格式问题,如:"Project name": "web_2048 # 项目名称使用下划线", - # compile example暂时不支持markdown - instruction = self.compile_instruction(schema="markdown", mode=mode, exclude=exclude) - example = self.compile_example(schema=schema, tag=TAG, mode=mode, exclude=exclude) - # nodes = ", ".join(self.to_dict(mode=mode).keys()) - constraints = [LANGUAGE_CONSTRAINT, FORMAT_CONSTRAINT] - constraint = "\n".join(constraints) - - prompt = template.format( - context=context, - example=example, - instruction=instruction, - constraint=constraint, - ) - return prompt - - @retry( - wait=wait_random_exponential(min=1, max=20), - stop=stop_after_attempt(6), - after=general_after_log(logger), - ) - async def _aask_v1( - self, - prompt: str, - output_class_name: str, - output_data_mapping: dict, - system_msgs: Optional[list[str]] = None, - schema="markdown", # compatible to original format - timeout=CONFIG.timeout, - ) -> (str, BaseModel): - """Use ActionOutput to wrap the output of aask""" - content = await self.llm.aask(prompt, system_msgs, timeout=timeout) - logger.debug(f"llm raw output:\n{content}") - output_class = self.create_model_class(output_class_name, output_data_mapping) - - if schema == "json": - parsed_data = llm_output_postprocess( - output=content, schema=output_class.model_json_schema(), req_key=f"[/{TAG}]" - ) - else: # using markdown parser - parsed_data = OutputParser.parse_data_with_mapping(content, output_data_mapping) - - logger.debug(f"parsed_data:\n{parsed_data}") - instruct_content = output_class(**parsed_data) - return content, instruct_content - - def get(self, key): - return self.instruct_content.model_dump()[key] - - def set_recursive(self, name, value): - setattr(self, name, value) - for _, i in self.children.items(): - i.set_recursive(name, value) - - def set_llm(self, llm): - self.set_recursive("llm", llm) - - def set_context(self, context): - self.set_recursive("context", context) - - async def simple_fill(self, schema, mode, timeout=CONFIG.timeout, exclude=None): - prompt = self.compile(context=self.context, schema=schema, mode=mode, exclude=exclude) - - if schema != "raw": - mapping = self.get_mapping(mode, exclude=exclude) - class_name = f"{self.key}_AN" - content, scontent = await self._aask_v1(prompt, class_name, mapping, schema=schema, timeout=timeout) - self.content = content - self.instruct_content = scontent - else: - self.content = await self.llm.aask(prompt) - self.instruct_content = None - - return self - - async def fill(self, context, llm, schema="json", mode="auto", strgy="simple", timeout=CONFIG.timeout, exclude=[]): - """Fill the node(s) with mode. - - :param context: Everything we should know when filling node. - :param llm: Large Language Model with pre-defined system message. - :param schema: json/markdown, determine example and output format. - - raw: free form text - - json: it's easy to open source LLM with json format - - markdown: when generating code, markdown is always better - :param mode: auto/children/root - - auto: automated fill children's nodes and gather outputs, if no children, fill itself - - children: fill children's nodes and gather outputs - - root: fill root's node and gather output - :param strgy: simple/complex - - simple: run only once - - complex: run each node - :param timeout: Timeout for llm invocation. - :param exclude: The keys of ActionNode to exclude. - :return: self - """ - self.set_llm(llm) - self.set_context(context) - if self.schema: - schema = self.schema - - if strgy == "simple": - return await self.simple_fill(schema=schema, mode=mode, timeout=timeout, exclude=exclude) - elif strgy == "complex": - # 这里隐式假设了拥有children - tmp = {} - for _, i in self.children.items(): - if exclude and i.key in exclude: - continue - child = await i.simple_fill(schema=schema, mode=mode, timeout=timeout, exclude=exclude) - tmp.update(child.instruct_content.dict()) - cls = self.create_children_class() - self.instruct_content = cls(**tmp) - return self diff --git a/PR/repair_llm_raw_output.py b/PR/repair_llm_raw_output.py deleted file mode 100644 index 4995918c2..000000000 --- a/PR/repair_llm_raw_output.py +++ /dev/null @@ -1,351 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Desc : repair llm raw output with particular conditions - -import copy -from enum import Enum -from typing import Callable, Union - -import regex as re -from tenacity import RetryCallState, retry, stop_after_attempt, wait_fixed - -from metagpt.config import CONFIG -from metagpt.logs import logger -from metagpt.utils.custom_decoder import CustomDecoder - - -class RepairType(Enum): - CS = "case sensitivity" - RKPM = "required key pair missing" # condition like `[key] xx` which lacks `[/key]` - SCM = "special character missing" # Usually the req_key appear in pairs like `[key] xx [/key]` - JSON = "json format" - - -def repair_case_sensitivity(output: str, req_key: str) -> str: - """ - usually, req_key is the key name of expected json or markdown content, it won't appear in the value part. - fix target string `"Shared Knowledge": ""` but `"Shared knowledge": ""` actually - """ - if req_key in output: - return output - - output_lower = output.lower() - req_key_lower = req_key.lower() - if req_key_lower in output_lower: - # find the sub-part index, and replace it with raw req_key - lidx = output_lower.find(req_key_lower) - source = output[lidx : lidx + len(req_key_lower)] - output = output.replace(source, req_key) - logger.info(f"repair_case_sensitivity: {req_key}") - - return output - - -def repair_special_character_missing(output: str, req_key: str = "[/CONTENT]") -> str: - """ - fix - 1. target string `[CONTENT] xx [CONTENT] xxx [CONTENT]` lacks `/` in the last `[CONTENT]` - 2. target string `xx [CONTENT] xxx [CONTENT] xxxx` lacks `/` in the last `[CONTENT]` - """ - sc_arr = ["/"] - - if req_key in output: - return output - - for sc in sc_arr: - req_key_pure = req_key.replace(sc, "") - appear_cnt = output.count(req_key_pure) - if req_key_pure in output and appear_cnt > 1: - # req_key with special_character usually in the tail side - ridx = output.rfind(req_key_pure) - output = f"{output[:ridx]}{req_key}{output[ridx + len(req_key_pure):]}" - logger.info(f"repair_special_character_missing: {sc} in {req_key_pure} as position {ridx}") - - return output - - -def repair_required_key_pair_missing(output: str, req_key: str = "[/CONTENT]") -> str: - """ - implement the req_key pair in the begin or end of the content - req_key format - 1. `[req_key]`, and its pair `[/req_key]` - 2. `[/req_key]`, and its pair `[req_key]` - """ - sc = "/" # special char - if req_key.startswith("[") and req_key.endswith("]"): - if sc in req_key: - left_key = req_key.replace(sc, "") # `[/req_key]` -> `[req_key]` - right_key = req_key - else: - left_key = req_key - right_key = f"{req_key[0]}{sc}{req_key[1:]}" # `[req_key]` -> `[/req_key]` - - if left_key not in output: - output = left_key + "\n" + output - if right_key not in output: - - def judge_potential_json(routput: str, left_key: str) -> Union[str, None]: - ridx = routput.rfind(left_key) - if ridx < 0: - return None - sub_output = routput[ridx:] - idx1 = sub_output.rfind("}") - idx2 = sub_output.rindex("]") - idx = idx1 if idx1 >= idx2 else idx2 - sub_output = sub_output[: idx + 1] - return sub_output - - if output.strip().endswith("}") or (output.strip().endswith("]") and not output.strip().endswith(left_key)): - # # avoid [req_key]xx[req_key] case to append [/req_key] - output = output + "\n" + right_key - elif judge_potential_json(output, left_key) and (not output.strip().endswith(left_key)): - sub_content = judge_potential_json(output, left_key) - output = sub_content + "\n" + right_key - - return output - - -def repair_json_format(output: str) -> str: - """ - fix extra `[` or `}` in the end - """ - output = output.strip() - - if output.startswith("[{"): - output = output[1:] - logger.info(f"repair_json_format: {'[{'}") - elif output.endswith("}]"): - output = output[:-1] - logger.info(f"repair_json_format: {'}]'}") - elif output.startswith("{") and output.endswith("]"): - output = output[:-1] + "}" - - # remove comments in output json str, after json value content, maybe start with #, maybe start with // - arr = output.split("\n") - new_arr = [] - for line in arr: - # look for # or // comments and make sure they are not inside the string value - comment_index = -1 - for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): - if match.group(1): # if the string value - continue - if match.group(2): # if comments - comment_index = match.start(2) - break - # if comments, then delete them - if comment_index != -1: - line = line[:comment_index].rstrip() - new_arr.append(line) - output = "\n".join(new_arr) - return output - - -def _repair_llm_raw_output(output: str, req_key: str, repair_type: RepairType = None) -> str: - repair_types = [repair_type] if repair_type else [item for item in RepairType if item not in [RepairType.JSON]] - for repair_type in repair_types: - if repair_type == RepairType.CS: - output = repair_case_sensitivity(output, req_key) - elif repair_type == RepairType.RKPM: - output = repair_required_key_pair_missing(output, req_key) - elif repair_type == RepairType.SCM: - output = repair_special_character_missing(output, req_key) - elif repair_type == RepairType.JSON: - output = repair_json_format(output) - return output - - -def repair_llm_raw_output(output: str, req_keys: list[str], repair_type: RepairType = None) -> str: - """ - in open-source llm model, it usually can't follow the instruction well, the output may be incomplete, - so here we try to repair it and use all repair methods by default. - typical case - 1. case sensitivity - target: "Original Requirements" - output: "Original requirements" - 2. special character missing - target: [/CONTENT] - output: [CONTENT] - 3. json format - target: { xxx } - output: { xxx }] - """ - if not CONFIG.repair_llm_output: - return output - - # do the repairation usually for non-openai models - for req_key in req_keys: - output = _repair_llm_raw_output(output=output, req_key=req_key, repair_type=repair_type) - return output - - -def repair_invalid_json(output: str, error: str) -> str: - """ - repair the situation like there are extra chars like - error examples - example 1. json.decoder.JSONDecodeError: Expecting ',' delimiter: line 154 column 1 (char 2765) - example 2. xxx.JSONDecodeError: Expecting property name enclosed in double quotes: line 14 column 1 (char 266) - """ - pattern = r"line ([0-9]+) column ([0-9]+)" - - matches = re.findall(pattern, error, re.DOTALL) - if len(matches) > 0: - line_no = int(matches[0][0]) - 1 - col_no = int(matches[0][1]) - 1 - - # due to CustomDecoder can handle `"": ''` or `'': ""`, so convert `"""` -> `"`, `'''` -> `'` - output = output.replace('"""', '"').replace("'''", '"') - arr = output.split("\n") - rline = arr[line_no] # raw line - line = arr[line_no].strip() - # different general problems - if line.endswith("],"): - # problem, redundant char `]` - new_line = line.replace("]", "") - elif line.endswith("},") and not output.endswith("},"): - # problem, redundant char `}` - new_line = line.replace("}", "") - elif line.endswith("},") and output.endswith("},"): - new_line = line[:-1] - # remove comments in output json str, after json value content, maybe start with #, maybe start with // - elif rline[col_no] == "#" or rline[col_no] == "/": - new_line = rline[:col_no] - for i in range(line_no + 1, len(arr)): - # look for # or // comments and make sure they are not inside the string value - comment_index = -1 - for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): - if match.group(1): # if the string value - continue - if match.group(2): # if comments - comment_index = match.start(2) - break - # if comments, then delete them - if comment_index != -1: - arr[i] = arr[i][:comment_index].rstrip() - elif (rline[col_no] in ["'", '"']) and (line.startswith('"') or line.startswith("'")) and "," not in line: - # problem, `"""` or `'''` without `,` - new_line = f",{line}" - elif '",' not in line and "," not in line and '"' not in line: - new_line = f'{line}",' - elif not line.endswith(","): - # problem, miss char `,` at the end. - new_line = f"{line}," - elif "," in line and len(line) == 1: - new_line = f'"{line}' - elif '",' in line: - new_line = line[:-2] + "'," - else: - new_line = line - - arr[line_no] = new_line - output = "\n".join(arr) - logger.info(f"repair_invalid_json, raw error: {error}") - - return output - - -def run_after_exp_and_passon_next_retry(logger: "loguru.Logger") -> Callable[["RetryCallState"], None]: - def run_and_passon(retry_state: RetryCallState) -> None: - """ - RetryCallState example - { - "start_time":143.098322024, - "retry_object":")>", - "fn":"", - "args":"(\"tag:[/CONTENT]\",)", # function input args - "kwargs":{}, # function input kwargs - "attempt_number":1, # retry number - "outcome":"", # type(outcome.result()) = "str", type(outcome.exception()) = "class" - "outcome_timestamp":143.098416904, - "idle_for":0, - "next_action":"None" - } - """ - if retry_state.outcome.failed: - if retry_state.args: - # # can't be used as args=retry_state.args - func_param_output = retry_state.args[0] - elif retry_state.kwargs: - func_param_output = retry_state.kwargs.get("output", "") - exp_str = str(retry_state.outcome.exception()) - - fix_str = "try to fix it, " if CONFIG.repair_llm_output else "" - logger.warning( - f"parse json from content inside [CONTENT][/CONTENT] failed at retry " - f"{retry_state.attempt_number}, {fix_str}exp: {exp_str}" - ) - - repaired_output = repair_invalid_json(func_param_output, exp_str) - retry_state.kwargs["output"] = repaired_output - - return run_and_passon - - -@retry( - stop=stop_after_attempt(3 if CONFIG.repair_llm_output else 0), - wait=wait_fixed(1), - after=run_after_exp_and_passon_next_retry(logger), -) -def retry_parse_json_text(output: str) -> Union[list, dict]: - """ - repair the json-text situation like there are extra chars like [']', '}'] - - Warning - if CONFIG.repair_llm_output is False, retry _aask_v1 {x=3} times, and the retry_parse_json_text's retry not work - if CONFIG.repair_llm_output is True, the _aask_v1 and the retry_parse_json_text will loop for {x=3*3} times. - it's a two-layer retry cycle - """ - # logger.debug(f"output to json decode:\n{output}") - - # if CONFIG.repair_llm_output is True, it will try to fix output until the retry break - parsed_data = CustomDecoder(strict=False).decode(output) - - return parsed_data - - -def extract_content_from_output(content: str, right_key: str = "[/CONTENT]"): - """extract xxx from [CONTENT](xxx)[/CONTENT] using regex pattern""" - - def re_extract_content(cont: str, pattern: str) -> str: - matches = re.findall(pattern, cont, re.DOTALL) - for match in matches: - if match: - cont = match - break - return cont.strip() - - # TODO construct the extract pattern with the `right_key` - raw_content = copy.deepcopy(content) - pattern = r"\[CONTENT\]([\s\S]*)\[/CONTENT\]" - new_content = re_extract_content(raw_content, pattern) - - if not new_content.startswith("{"): - # TODO find a more general pattern - # # for `[CONTENT]xxx[CONTENT]xxxx[/CONTENT] situation - logger.warning(f"extract_content try another pattern: {pattern}") - if right_key not in new_content: - raw_content = copy.deepcopy(new_content + "\n" + right_key) - # # pattern = r"\[CONTENT\](\s*\{.*?\}\s*)\[/CONTENT\]" - new_content = re_extract_content(raw_content, pattern) - else: - if right_key in new_content: - idx = new_content.find(right_key) - new_content = new_content[:idx] - new_content = new_content.strip() - - return new_content - - -def extract_state_value_from_output(content: str) -> str: - """ - For openai models, they will always return state number. But for open llm models, the instruction result maybe a - long text contain target number, so here add a extraction to improve success rate. - - Args: - content (str): llm's output from `Role._think` - """ - content = content.strip() # deal the output cases like " 0", "0\n" and so on. - pattern = r"([0-9])" # TODO find the number using a more proper method not just extract from content using pattern - matches = re.findall(pattern, content, re.DOTALL) - matches = list(set(matches)) - state = matches[0] if len(matches) > 0 else "-1" - return state From 11f70ca9b1714dc312a847505c9940f0c60a24b1 Mon Sep 17 00:00:00 2001 From: huzixia <528543747@qq.com> Date: Sat, 27 Jan 2024 18:06:52 +0800 Subject: [PATCH 497/637] modify code based on feedback of action_node.py and repair_llm_raw_output.py, add code in test_repair_llm_raw_output.py --- metagpt/actions/action_node.py | 5 +--- metagpt/utils/repair_llm_raw_output.py | 9 +------ .../utils/test_repair_llm_raw_output.py | 26 +++++++++++++++++++ 3 files changed, 28 insertions(+), 12 deletions(-) diff --git a/metagpt/actions/action_node.py b/metagpt/actions/action_node.py index ed0e27869..6c65b33ef 100644 --- a/metagpt/actions/action_node.py +++ b/metagpt/actions/action_node.py @@ -23,10 +23,7 @@ from metagpt.utils.common import OutputParser, general_after_log TAG = "CONTENT" LANGUAGE_CONSTRAINT = "Language: Please use the same language as Human INPUT." -FORMAT_CONSTRAINT = (f"Format: output wrapped inside [{TAG}][/{TAG}] like format example, nothing else. " - f"Delete comments in json") -# Delete comments in json -# If you don't want JSONDecodeError to occur, you can add Delete comments in json after FORMAT_CONSTRAINT +FORMAT_CONSTRAINT = f"Format: output wrapped inside [{TAG}][/{TAG}] like format example, nothing else." SIMPLE_TEMPLATE = """ diff --git a/metagpt/utils/repair_llm_raw_output.py b/metagpt/utils/repair_llm_raw_output.py index ef3580750..973cffb8a 100644 --- a/metagpt/utils/repair_llm_raw_output.py +++ b/metagpt/utils/repair_llm_raw_output.py @@ -137,11 +137,10 @@ def repair_json_format(output: str) -> str: elif output.startswith("{") and output.endswith("]"): output = output[:-1] + "}" - # remove comments in output json str, after json value content, maybe start with #, maybe start with // + # remove comments in output json string arr = output.split("\n") new_arr = [] for line in arr: - # look for # or // comments and make sure they are not inside the string value new_line = remove_comments_from_line(line) new_arr.append(new_line) output = "\n".join(new_arr) @@ -214,12 +213,6 @@ def repair_invalid_json(output: str, error: str) -> str: new_line = line.replace("}", "") elif line.endswith("},") and output.endswith("},"): new_line = line[:-1] - # remove comments in output json str, after json value content, maybe start with #, maybe start with // - elif rline[col_no] == "#" or rline[col_no] == "/": - new_line = rline[:col_no] - # check the next line and remove the comments - for i in range(line_no + 1, len(arr)): - arr[i] = remove_comments_from_line(arr[i]) elif (rline[col_no] in ["'", '"']) and (line.startswith('"') or line.startswith("'")) and "," not in line: # problem, `"""` or `'''` without `,` new_line = f",{line}" diff --git a/tests/metagpt/utils/test_repair_llm_raw_output.py b/tests/metagpt/utils/test_repair_llm_raw_output.py index 1f809a081..9d53b8243 100644 --- a/tests/metagpt/utils/test_repair_llm_raw_output.py +++ b/tests/metagpt/utils/test_repair_llm_raw_output.py @@ -141,6 +141,32 @@ def test_repair_json_format(): output = repair_llm_raw_output(output=raw_output, req_keys=[None], repair_type=RepairType.JSON) assert output == target_output + raw_output = """ +{ + "Language": "en_us", // define language + "Programming Language": "Python" # define code language +} +""" + target_output = """{ + "Language": "en_us", + "Programming Language": "Python" +}""" + output = repair_llm_raw_output(output=raw_output, req_keys=[None], repair_type=RepairType.JSON) + assert output == target_output + + raw_output = """ + { + "Language": "#en_us#", // define language + "Programming Language": "//Python # Code // Language//" # define code language + } + """ + target_output = """{ + "Language": "#en_us#", + "Programming Language": "//Python # Code // Language//" + }""" + output = repair_llm_raw_output(output=raw_output, req_keys=[None], repair_type=RepairType.JSON) + assert output == target_output + def test_repair_invalid_json(): from metagpt.utils.repair_llm_raw_output import repair_invalid_json From c3b4c698d80cba70e446cd6a97f375459c8c5595 Mon Sep 17 00:00:00 2001 From: huzixia <528543747@qq.com> Date: Sat, 27 Jan 2024 18:23:57 +0800 Subject: [PATCH 498/637] update repair_llm_raw_output.py --- metagpt/utils/repair_llm_raw_output.py | 36 ++++++++++---------------- 1 file changed, 14 insertions(+), 22 deletions(-) diff --git a/metagpt/utils/repair_llm_raw_output.py b/metagpt/utils/repair_llm_raw_output.py index 973cffb8a..6da974d96 100644 --- a/metagpt/utils/repair_llm_raw_output.py +++ b/metagpt/utils/repair_llm_raw_output.py @@ -105,23 +105,6 @@ def repair_required_key_pair_missing(output: str, req_key: str = "[/CONTENT]") - return output -def remove_comments_from_line(line): - """ - Remove comments from a single line of string. - Comments are assumed to start with '#' or '//' and are not inside string values. - """ - comment_index = -1 - for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", line): - if match.group(1): # if the string value - continue - if match.group(2): # if comments - comment_index = match.start(2) - break - if comment_index != -1: # if comments, then delete them - return line[:comment_index].rstrip() - return line - - def repair_json_format(output: str) -> str: """ fix extra `[` or `}` in the end @@ -136,13 +119,22 @@ def repair_json_format(output: str) -> str: logger.info(f"repair_json_format: {'}]'}") elif output.startswith("{") and output.endswith("]"): output = output[:-1] + "}" - - # remove comments in output json string + # remove comments in output json string, after json value content, maybe start with #, maybe start with // arr = output.split("\n") new_arr = [] - for line in arr: - new_line = remove_comments_from_line(line) - new_arr.append(new_line) + for json_line in arr: + # look for # or // comments and make sure they are not inside the string value + comment_index = -1 + for match in re.finditer(r"(\".*?\"|\'.*?\')|(#|//)", json_line): + if match.group(1): # if the string value + continue + if match.group(2): # if comments + comment_index = match.start(2) + break + # if comments, then delete them + if comment_index != -1: + json_line = json_line[:comment_index].rstrip() + new_arr.append(json_line) output = "\n".join(new_arr) return output From a68c3442bcff09864409ed47a02bbcc476657d23 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E7=A8=8B=E5=85=81=E6=9D=83?= Date: Sun, 28 Jan 2024 10:39:00 +0800 Subject: [PATCH 499/637] Refactor get_choice_delta_text for safer dict access --- metagpt/provider/base_llm.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/provider/base_llm.py b/metagpt/provider/base_llm.py index a50cdacd9..47c527b97 100644 --- a/metagpt/provider/base_llm.py +++ b/metagpt/provider/base_llm.py @@ -91,7 +91,7 @@ class BaseLLM(ABC): def get_choice_delta_text(self, rsp: dict) -> str: """Required to provide the first text of stream choice""" - return rsp.get("choices")[0]["delta"]["content"] + return rsp.get("choices", [{}])[0].get("delta", {}).get("content", "") def get_choice_function(self, rsp: dict) -> dict: """Required to provide the first function of choice From 06b4e4767a9ab7a74d8f293b215d644d1cda71a6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Mon, 29 Jan 2024 10:14:09 +0800 Subject: [PATCH 500/637] feat: generate_repo return ProjectRepo --- metagpt/startup.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/metagpt/startup.py b/metagpt/startup.py index 000b3c5d4..4a077cab7 100644 --- a/metagpt/startup.py +++ b/metagpt/startup.py @@ -1,5 +1,6 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- + import asyncio import shutil from pathlib import Path @@ -9,6 +10,7 @@ import typer from metagpt.config2 import config from metagpt.const import CONFIG_ROOT, METAGPT_ROOT from metagpt.context import Context +from metagpt.utils.project_repo import ProjectRepo app = typer.Typer(add_completion=False, pretty_exceptions_show_locals=False) @@ -26,7 +28,7 @@ def generate_repo( reqa_file, max_auto_summarize_code, recover_path, -): +) -> ProjectRepo: """Run the startup logic. Can be called from CLI or other Python scripts.""" from metagpt.roles import ( Architect, @@ -67,6 +69,8 @@ def generate_repo( company.run_project(idea) asyncio.run(company.run(n_round=n_round)) + return ctx.repo + @app.command("", help="Start a new project.") def startup( From 310687258eebb80abe04a949ae78d2a87fd0f2c2 Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 11:59:53 +0800 Subject: [PATCH 501/637] add gpt-4v support for aask and AN.fill --- examples/llm_hello_world.py | 11 +++++++++ metagpt/actions/action_node.py | 26 +++++++++++++++----- metagpt/provider/base_llm.py | 25 ++++++++++++++++--- metagpt/provider/openai_api.py | 2 +- metagpt/utils/common.py | 30 +++++++++++++++++++++++ metagpt/utils/token_counter.py | 11 ++++++++- tests/metagpt/actions/test_action_node.py | 14 +++++++++++ tests/mock/mock_llm.py | 8 +++--- 8 files changed, 113 insertions(+), 14 deletions(-) diff --git a/examples/llm_hello_world.py b/examples/llm_hello_world.py index 219a303c8..dfc2603aa 100644 --- a/examples/llm_hello_world.py +++ b/examples/llm_hello_world.py @@ -6,9 +6,11 @@ @File : llm_hello_world.py """ import asyncio +from pathlib import Path from metagpt.llm import LLM from metagpt.logs import logger +from metagpt.utils.common import encode_image async def main(): @@ -27,6 +29,15 @@ async def main(): if hasattr(llm, "completion"): logger.info(llm.completion(hello_msg)) + # check llm-vision capacity if it supports + invoice_path = Path(__file__).parent.joinpath("..", "tests", "data", "invoices", "invoice-2.png") + img_base64 = encode_image(invoice_path) + try: + res = await llm.aask(msg="if this is a invoice, just return True else return False", images=[img_base64]) + assert "true" in res.lower() + except Exception: + pass + if __name__ == "__main__": asyncio.run(main()) diff --git a/metagpt/actions/action_node.py b/metagpt/actions/action_node.py index 162ab90eb..bd2f0d11f 100644 --- a/metagpt/actions/action_node.py +++ b/metagpt/actions/action_node.py @@ -370,12 +370,13 @@ class ActionNode: prompt: str, output_class_name: str, output_data_mapping: dict, + images: Optional[Union[str, list[str]]] = None, system_msgs: Optional[list[str]] = None, schema="markdown", # compatible to original format timeout=3, ) -> (str, BaseModel): """Use ActionOutput to wrap the output of aask""" - content = await self.llm.aask(prompt, system_msgs, timeout=timeout) + content = await self.llm.aask(prompt, system_msgs, images=images, timeout=timeout) logger.debug(f"llm raw output:\n{content}") output_class = self.create_model_class(output_class_name, output_data_mapping) @@ -404,13 +405,15 @@ class ActionNode: def set_context(self, context): self.set_recursive("context", context) - async def simple_fill(self, schema, mode, timeout=3, exclude=None): + async def simple_fill(self, schema, mode, images: Optional[Union[str, list[str]]] = None, timeout=3, exclude=None): prompt = self.compile(context=self.context, schema=schema, mode=mode, exclude=exclude) if schema != "raw": mapping = self.get_mapping(mode, exclude=exclude) class_name = f"{self.key}_AN" - content, scontent = await self._aask_v1(prompt, class_name, mapping, schema=schema, timeout=timeout) + content, scontent = await self._aask_v1( + prompt, class_name, mapping, images=images, schema=schema, timeout=timeout + ) self.content = content self.instruct_content = scontent else: @@ -419,7 +422,17 @@ class ActionNode: return self - async def fill(self, context, llm, schema="json", mode="auto", strgy="simple", timeout=3, exclude=[]): + async def fill( + self, + context, + llm, + schema="json", + mode="auto", + strgy="simple", + images: Optional[Union[str, list[str]]] = None, + timeout=3, + exclude=[], + ): """Fill the node(s) with mode. :param context: Everything we should know when filling node. @@ -435,6 +448,7 @@ class ActionNode: :param strgy: simple/complex - simple: run only once - complex: run each node + :param images: the list of image url or base64 for gpt4-v :param timeout: Timeout for llm invocation. :param exclude: The keys of ActionNode to exclude. :return: self @@ -445,14 +459,14 @@ class ActionNode: schema = self.schema if strgy == "simple": - return await self.simple_fill(schema=schema, mode=mode, timeout=timeout, exclude=exclude) + return await self.simple_fill(schema=schema, mode=mode, images=images, timeout=timeout, exclude=exclude) elif strgy == "complex": # 这里隐式假设了拥有children tmp = {} for _, i in self.children.items(): if exclude and i.key in exclude: continue - child = await i.simple_fill(schema=schema, mode=mode, timeout=timeout, exclude=exclude) + child = await i.simple_fill(schema=schema, mode=mode, images=images, timeout=timeout, exclude=exclude) tmp.update(child.instruct_content.model_dump()) cls = self.create_children_class() self.instruct_content = cls(**tmp) diff --git a/metagpt/provider/base_llm.py b/metagpt/provider/base_llm.py index 5fe9d1c3a..7c5892018 100644 --- a/metagpt/provider/base_llm.py +++ b/metagpt/provider/base_llm.py @@ -34,8 +34,26 @@ class BaseLLM(ABC): def __init__(self, config: LLMConfig): pass - def _user_msg(self, msg: str) -> dict[str, str]: - return {"role": "user", "content": msg} + def _user_msg(self, msg: str, images: Optional[Union[str, list[str]]] = None) -> dict[str, Union[str, dict]]: + if images: + # as gpt-4v, chat with image + return self._user_msg_with_imgs(msg, images) + else: + return {"role": "user", "content": msg} + + def _user_msg_with_imgs(self, msg: str, images: Optional[Union[str, list[str]]]): + """ + images: can be list of http(s) url or base64 + """ + if isinstance(images, str): + images = [images] + content = [{"type": "text", "text": msg}] + for image in images: + # image url or image base64 + url = image if image.startswith("http") else f"data:image/jpeg;base64,{image}" + # it can with multiple-image inputs + content.append({"type": "image_url", "image_url": url}) + return {"role": "user", "content": content} def _assistant_msg(self, msg: str) -> dict[str, str]: return {"role": "assistant", "content": msg} @@ -54,6 +72,7 @@ class BaseLLM(ABC): msg: str, system_msgs: Optional[list[str]] = None, format_msgs: Optional[list[dict[str, str]]] = None, + images: Optional[Union[str, list[str]]] = None, timeout=3, stream=True, ) -> str: @@ -65,7 +84,7 @@ class BaseLLM(ABC): message = [] if format_msgs: message.extend(format_msgs) - message.append(self._user_msg(msg)) + message.append(self._user_msg(msg, images=images)) logger.debug(message) rsp = await self.acompletion_text(message, stream=stream, timeout=timeout) return rsp diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index d6944eae6..2ec78317a 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -99,7 +99,7 @@ class OpenAILLM(BaseLLM): "messages": messages, "max_tokens": self._get_max_tokens(messages), "n": 1, - "stop": None, + # "stop": None, # default it's None and gpt4-v can't have this one "temperature": 0.3, "model": self.model, "timeout": max(self.config.timeout, timeout), diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index f1bd1a8e5..73017cf77 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -12,7 +12,9 @@ from __future__ import annotations import ast +import base64 import contextlib +import csv import importlib import inspect import json @@ -465,6 +467,29 @@ def write_json_file(json_file: str, data: list, encoding=None): json.dump(data, fout, ensure_ascii=False, indent=4, default=to_jsonable_python) +def read_csv_to_list(curr_file: str, header=False, strip_trail=True): + """ + Reads in a csv file to a list of list. If header is True, it returns a + tuple with (header row, all rows) + ARGS: + curr_file: path to the current csv file. + RETURNS: + List of list where the component lists are the rows of the file. + """ + logger.debug(f"start read csv: {curr_file}") + analysis_list = [] + with open(curr_file) as f_analysis_file: + data_reader = csv.reader(f_analysis_file, delimiter=",") + for count, row in enumerate(data_reader): + if strip_trail: + row = [i.strip() for i in row] + analysis_list += [row] + if not header: + return analysis_list + else: + return analysis_list[0], analysis_list[1:] + + def import_class(class_name: str, module_name: str) -> type: module = importlib.import_module(module_name) a_class = getattr(module, class_name) @@ -573,3 +598,8 @@ def list_files(root: str | Path) -> List[Path]: except Exception as e: logger.error(f"Error: {e}") return files + + +def encode_image(image_path: Path, encoding: str = "utf-8") -> str: + with open(str(image_path), "rb") as image_file: + return base64.b64encode(image_file.read()).decode(encoding) diff --git a/metagpt/utils/token_counter.py b/metagpt/utils/token_counter.py index 94506e373..a0fb3b70d 100644 --- a/metagpt/utils/token_counter.py +++ b/metagpt/utils/token_counter.py @@ -29,6 +29,7 @@ TOKEN_COSTS = { "gpt-4-turbo-preview": {"prompt": 0.01, "completion": 0.03}, "gpt-4-0125-preview": {"prompt": 0.01, "completion": 0.03}, "gpt-4-1106-preview": {"prompt": 0.01, "completion": 0.03}, + "gpt-4-vision-preview": {"prompt": 0.01, "completion": 0.03}, # TODO add extra image price calculator "gpt-4-1106-vision-preview": {"prompt": 0.01, "completion": 0.03}, "text-embedding-ada-002": {"prompt": 0.0004, "completion": 0.0}, "glm-3-turbo": {"prompt": 0.0, "completion": 0.0007}, # 128k version, prompt + completion tokens=0.005¥/k-tokens @@ -54,6 +55,7 @@ TOKEN_MAX = { "gpt-4-turbo-preview": 128000, "gpt-4-0125-preview": 128000, "gpt-4-1106-preview": 128000, + "gpt-4-vision-preview": 128000, "gpt-4-1106-vision-preview": 128000, "text-embedding-ada-002": 8192, "chatglm_turbo": 32768, @@ -82,6 +84,7 @@ def count_message_tokens(messages, model="gpt-3.5-turbo-0613"): "gpt-4-turbo-preview", "gpt-4-0125-preview", "gpt-4-1106-preview", + "gpt-4-vision-preview", "gpt-4-1106-vision-preview", }: tokens_per_message = 3 # # every reply is primed with <|start|>assistant<|message|> @@ -112,7 +115,13 @@ def count_message_tokens(messages, model="gpt-3.5-turbo-0613"): for message in messages: num_tokens += tokens_per_message for key, value in message.items(): - num_tokens += len(encoding.encode(value)) + content = value + if isinstance(value, list): + # for gpt-4v + for item in value: + if isinstance(item, dict) and item.get("type") in ["text"]: + content = item.get("text", "") + num_tokens += len(encoding.encode(content)) if key == "name": num_tokens += tokens_per_name num_tokens += 3 # every reply is primed with <|start|>assistant<|message|> diff --git a/tests/metagpt/actions/test_action_node.py b/tests/metagpt/actions/test_action_node.py index 53de9cc75..8aee071d4 100644 --- a/tests/metagpt/actions/test_action_node.py +++ b/tests/metagpt/actions/test_action_node.py @@ -5,6 +5,7 @@ @Author : alexanderwu @File : test_action_node.py """ +from pathlib import Path from typing import List, Tuple import pytest @@ -17,6 +18,7 @@ from metagpt.llm import LLM from metagpt.roles import Role from metagpt.schema import Message from metagpt.team import Team +from metagpt.utils.common import encode_image @pytest.mark.asyncio @@ -241,6 +243,18 @@ def test_create_model_class_with_mapping(): assert value == ["game.py", "app.py", "static/css/styles.css", "static/js/script.js", "templates/index.html"] +@pytest.mark.asyncio +async def test_action_node_with_image(): + invoice = ActionNode( + key="invoice", expected_type=bool, instruction="if it's a invoice file, return True else False", example="False" + ) + + invoice_path = Path(__file__).parent.joinpath("..", "..", "data", "invoices", "invoice-2.png") + img_base64 = encode_image(invoice_path) + node = await invoice.fill(context="", llm=LLM(), images=[img_base64]) + assert node.instruct_content.invoice + + if __name__ == "__main__": test_create_model_class() test_create_model_class_with_mapping() diff --git a/tests/mock/mock_llm.py b/tests/mock/mock_llm.py index bef380c83..f093d9ce1 100644 --- a/tests/mock/mock_llm.py +++ b/tests/mock/mock_llm.py @@ -1,4 +1,4 @@ -from typing import Optional +from typing import Optional, Union from metagpt.config2 import config from metagpt.logs import log_llm_stream, logger @@ -35,6 +35,7 @@ class MockLLM(OpenAILLM): msg: str, system_msgs: Optional[list[str]] = None, format_msgs: Optional[list[dict[str, str]]] = None, + images: Optional[Union[str, list[str]]] = None, timeout=3, stream=True, ): @@ -47,7 +48,7 @@ class MockLLM(OpenAILLM): message = [] if format_msgs: message.extend(format_msgs) - message.append(self._user_msg(msg)) + message.append(self._user_msg(msg, images=images)) rsp = await self.acompletion_text(message, stream=stream, timeout=timeout) return rsp @@ -66,6 +67,7 @@ class MockLLM(OpenAILLM): msg: str, system_msgs: Optional[list[str]] = None, format_msgs: Optional[list[dict[str, str]]] = None, + images: Optional[Union[str, list[str]]] = None, timeout=3, stream=True, ) -> str: @@ -73,7 +75,7 @@ class MockLLM(OpenAILLM): if system_msgs: joined_system_msg = "#MSG_SEP#".join(system_msgs) + "#SYSTEM_MSG_END#" msg_key = joined_system_msg + msg_key - rsp = await self._mock_rsp(msg_key, self.original_aask, msg, system_msgs, format_msgs, timeout, stream) + rsp = await self._mock_rsp(msg_key, self.original_aask, msg, system_msgs, format_msgs, images, timeout, stream) return rsp async def aask_batch(self, msgs: list, timeout=3) -> str: From e7cd90f7f8c75b82145769ae22a70142aaca9939 Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 14:31:00 +0800 Subject: [PATCH 502/637] add openai dall-e support --- metagpt/provider/openai_api.py | 25 +++++++++++++++++++++ metagpt/utils/common.py | 32 ++++++++++++++++++++++++--- requirements.txt | 1 + tests/metagpt/provider/test_openai.py | 13 +++++++++++ 4 files changed, 68 insertions(+), 3 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 2ec78317a..8f9d91d6c 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -28,6 +28,7 @@ from metagpt.provider.base_llm import BaseLLM from metagpt.provider.constant import GENERAL_FUNCTION_SCHEMA, GENERAL_TOOL_CHOICE from metagpt.provider.llm_provider_registry import register_provider from metagpt.schema import Message +from metagpt.utils.common import decode_image from metagpt.utils.cost_manager import CostManager, Costs from metagpt.utils.exceptions import handle_exception from metagpt.utils.token_counter import ( @@ -243,3 +244,27 @@ class OpenAILLM(BaseLLM): async def aspeech_to_text(self, **kwargs): """speech to text""" return await self.aclient.audio.transcriptions.create(**kwargs) + + async def gen_image( + self, + prompt: str, + size: str = "1024x1024", + quality: str = "standard", + model: str = None, + resp_format: str = "url", + ) -> list["Image"]: + """image generate""" + assert resp_format in ["url", "b64_json"] + if not model: + model = self.model + res = await self.aclient.images.generate( + model=model, prompt=prompt, size=size, quality=quality, n=1, response_format=resp_format + ) + imgs = [] + for item in res.data: + if resp_format == "url": + img_url_or_b64 = item.url + else: + img_url_or_b64 = item.b64_json + imgs.append(decode_image(img_url_or_b64)) + return imgs diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 73017cf77..93921e983 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -24,11 +24,14 @@ import re import sys import traceback import typing +from io import BytesIO from pathlib import Path from typing import Any, List, Tuple, Union import aiofiles import loguru +import requests +from PIL import Image from pydantic_core import to_jsonable_python from tenacity import RetryCallState, RetryError, _utils @@ -600,6 +603,29 @@ def list_files(root: str | Path) -> List[Path]: return files -def encode_image(image_path: Path, encoding: str = "utf-8") -> str: - with open(str(image_path), "rb") as image_file: - return base64.b64encode(image_file.read()).decode(encoding) +def encode_image(image_path_or_pil: Union[Path, Image], encoding: str = "utf-8") -> str: + """encode image from file or PIL.Image into base64""" + if isinstance(image_path_or_pil, Image): + buffer = BytesIO() + image_path_or_pil.save(buffer, format="JPEG") + bytes_data = buffer.getvalue() + else: + if not image_path_or_pil.exists(): + raise FileNotFoundError(f"{image_path_or_pil} not exists") + with open(str(image_path_or_pil), "rb") as image_file: + bytes_data = image_file.read() + return base64.b64encode(bytes_data).decode(encoding) + + +def decode_image(img_url_or_b64: str) -> Image: + """decode image from url or base64 into PIL.Image""" + if img_url_or_b64.startswith("http"): + # image http(s) url + resp = requests.get(img_url_or_b64) + img = Image.open(BytesIO(resp.content)) + else: + # image b64_json + b64_data = re.sub("^data:image/.+;base64,", "", img_url_or_b64) + img_data = BytesIO(base64.b64decode(b64_data)) + img = Image.open(img_data) + return img diff --git a/requirements.txt b/requirements.txt index d54a1d22e..93091d137 100644 --- a/requirements.txt +++ b/requirements.txt @@ -59,3 +59,4 @@ networkx~=3.2.1 google-generativeai==0.3.2 # playwright==1.40.0 # playwright extras require anytree +Pillow \ No newline at end of file diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index bc7f92f33..82ab091c5 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -1,4 +1,5 @@ import pytest +from PIL import Image from metagpt.const import TEST_DATA_PATH from metagpt.llm import LLM @@ -62,6 +63,18 @@ async def test_speech_to_text(): assert "你好" == resp.text +@pytest.mark.asyncio +async def test_gen_image(): + llm = LLM() + model = "dall-e-3" + prompt = 'a logo with word "MetaGPT"' + images: list[Image] = await llm.gen_image(model=model, prompt=prompt) + assert images[0].size == (1024, 1024) + + images: list[Image] = await llm.gen_image(model=model, prompt=prompt, resp_format="b64_json") + assert images[0].size == (1024, 1024) + + class TestOpenAI: def test_make_client_kwargs_without_proxy(self): instance = OpenAILLM(mock_llm_config) From df728a034e0f20e128b58fb94c8c7ef94421cd6a Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 14:32:57 +0800 Subject: [PATCH 503/637] simplify code --- metagpt/provider/openai_api.py | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 8f9d91d6c..3d3251934 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -262,9 +262,6 @@ class OpenAILLM(BaseLLM): ) imgs = [] for item in res.data: - if resp_format == "url": - img_url_or_b64 = item.url - else: - img_url_or_b64 = item.b64_json + img_url_or_b64 = item.url if resp_format == "url" else item.b64_json imgs.append(decode_image(img_url_or_b64)) return imgs From 606f1b8f9cf60629f23c3ea8459a0a95c5b7103b Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 30 Jan 2024 16:40:13 +0800 Subject: [PATCH 504/637] accept goal during run; move more logic from role to planner --- metagpt/plan/planner.py | 64 ++++++++++++++------- metagpt/roles/role.py | 28 +++------ tests/metagpt/roles/run_code_interpreter.py | 3 +- 3 files changed, 50 insertions(+), 45 deletions(-) diff --git a/metagpt/plan/planner.py b/metagpt/plan/planner.py index 87492e455..fea5f0f8d 100644 --- a/metagpt/plan/planner.py +++ b/metagpt/plan/planner.py @@ -44,6 +44,48 @@ class Planner(BaseModel): def current_task_id(self): return self.plan.current_task_id + async def update_plan(self, goal: str = "", max_tasks: int = 3, max_retries: int = 3): + if goal: + self.plan = Plan(goal=goal) + + plan_confirmed = False + while not plan_confirmed: + context = self.get_useful_memories() + rsp = await WritePlan().run(context, max_tasks=max_tasks, use_tools=self.use_tools) + self.working_memory.add(Message(content=rsp, role="assistant", cause_by=WritePlan)) + + # precheck plan before asking reviews + is_plan_valid, error = precheck_update_plan_from_rsp(rsp, self.plan) + if not is_plan_valid and max_retries > 0: + error_msg = f"The generated plan is not valid with error: {error}, try regenerating, remember to generate either the whole plan or the single changed task only" + logger.warning(error_msg) + self.working_memory.add(Message(content=error_msg, role="assistant", cause_by=WritePlan)) + max_retries -= 1 + continue + + _, plan_confirmed = await self.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) + + update_plan_from_rsp(rsp=rsp, current_plan=self.plan) + + self.working_memory.clear() + + async def process_task_result(self, task_result: TaskResult): + # ask for acceptance, users can other refuse and change tasks in the plan + review, task_result_confirmed = await self.ask_review(task_result) + + if task_result_confirmed: + # tick off this task and record progress + await self.confirm_task(self.current_task, task_result, review) + + elif "redo" in review: + # Ask the Role to redo this task with help of review feedback, + # useful when the code run is successful but the procedure or result is not what we want + pass # simply pass, not confirming the result + + else: + # update plan according to user's feedback and to take on changed tasks + await self.update_plan() + async def ask_review( self, task_result: TaskResult = None, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER ): @@ -74,28 +116,6 @@ class Planner(BaseModel): self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) await self.update_plan(review) - async def update_plan(self, max_tasks: int = 3, max_retries: int = 3): - plan_confirmed = False - while not plan_confirmed: - context = self.get_useful_memories() - rsp = await WritePlan().run(context, max_tasks=max_tasks, use_tools=self.use_tools) - self.working_memory.add(Message(content=rsp, role="assistant", cause_by=WritePlan)) - - # precheck plan before asking reviews - is_plan_valid, error = precheck_update_plan_from_rsp(rsp, self.plan) - if not is_plan_valid and max_retries > 0: - error_msg = f"The generated plan is not valid with error: {error}, try regenerating, remember to generate either the whole plan or the single changed task only" - logger.warning(error_msg) - self.working_memory.add(Message(content=error_msg, role="assistant", cause_by=WritePlan)) - max_retries -= 1 - continue - - _, plan_confirmed = await self.ask_review(trigger=ReviewConst.TASK_REVIEW_TRIGGER) - - update_plan_from_rsp(rsp=rsp, current_plan=self.plan) - - self.working_memory.clear() - def get_useful_memories(self, task_exclude_field=None) -> list[Message]: """find useful memories only to reduce context length and improve performance""" # TODO dataset description , code steps diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 21e48a127..d176bbac3 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -452,10 +452,11 @@ class Role(SerializationMixin, is_polymorphic_base=True): async def _plan_and_act(self) -> Message: """first plan, then execute an action sequence, i.e. _think (of a plan) -> _act -> _act -> ... Use llm to come up with the plan dynamically.""" - ### Common Procedure in both single- and multi-agent setting ### - # create initial plan and update until confirmation - await self.planner.update_plan() + # create initial plan and update it until confirmation + goal = self.rc.memory.get()[-1].content # retreive latest user requirement + await self.planner.update_plan(goal=goal) + # take on tasks until all finished while self.planner.current_task: task = self.planner.current_task logger.info(f"ready to take on task {task}") @@ -463,25 +464,10 @@ class Role(SerializationMixin, is_polymorphic_base=True): # take on current task task_result = await self._act_on_task(task) - # ask for acceptance, users can other refuse and change tasks in the plan - review, task_result_confirmed = await self.planner.ask_review(task_result) + # process the result, such as reviewing, confirming, plan updating + await self.planner.process_task_result(task_result) - if task_result_confirmed: - # tick off this task and record progress - await self.planner.confirm_task(task, task_result, review) - - elif "redo" in review: - # Ask the Role to redo this task with help of review feedback, - # useful when the code run is successful but the procedure or result is not what we want - continue - - else: - # update plan according to user's feedback and to take on changed tasks - await self.planner.update_plan() - - completed_plan_memory = self.planner.get_useful_memories() # completed plan as a outcome - - rsp = completed_plan_memory[0] + rsp = self.planner.get_useful_memories()[0] # return the completed plan as a response self.rc.memory.add(rsp) # add to persistent memory diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py index e41507256..379194534 100644 --- a/tests/metagpt/roles/run_code_interpreter.py +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -23,10 +23,9 @@ async def run_code_interpreter(role_class, requirement, auto_run, use_tools, use """ if role_class == "ci": - role = CodeInterpreter(goal=requirement, auto_run=auto_run, use_tools=use_tools, tools=tools) + role = CodeInterpreter(auto_run=auto_run, use_tools=use_tools, tools=tools) else: role = MLEngineer( - goal=requirement, auto_run=auto_run, use_tools=use_tools, use_code_steps=use_code_steps, From 1d772e8eb59ee4af737ad060359d69b875f7b30f Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 19:06:49 +0800 Subject: [PATCH 505/637] update base env and android_env --- metagpt/const.py | 42 +++++ metagpt/environment/__init__.py | 13 ++ metagpt/environment/android_env/__init__.py | 3 + .../environment/android_env/android_env.py | 13 ++ .../android_env/android_ext_env.py | 157 ++++++++++++++++++ metagpt/environment/api/__init__.py | 3 + metagpt/environment/api/env_api.py | 45 +++++ .../base_env.py} | 90 ++++++++-- metagpt/logs.py | 5 +- metagpt/utils/common.py | 21 ++- .../environment/android_env/__init__.py | 3 + .../android_env/test_android_ext_env.py | 75 +++++++++ tests/metagpt/environment/api/__init__.py | 3 + tests/metagpt/environment/api/test_env_api.py | 15 ++ tests/metagpt/environment/test_base_env.py | 50 ++++++ 15 files changed, 521 insertions(+), 17 deletions(-) create mode 100644 metagpt/environment/__init__.py create mode 100644 metagpt/environment/android_env/__init__.py create mode 100644 metagpt/environment/android_env/android_env.py create mode 100644 metagpt/environment/android_env/android_ext_env.py create mode 100644 metagpt/environment/api/__init__.py create mode 100644 metagpt/environment/api/env_api.py rename metagpt/{environment.py => environment/base_env.py} (61%) create mode 100644 tests/metagpt/environment/android_env/__init__.py create mode 100644 tests/metagpt/environment/android_env/test_android_ext_env.py create mode 100644 tests/metagpt/environment/api/__init__.py create mode 100644 tests/metagpt/environment/api/test_env_api.py create mode 100644 tests/metagpt/environment/test_base_env.py diff --git a/metagpt/const.py b/metagpt/const.py index a1c650ce3..41a98356c 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -129,3 +129,45 @@ IGNORED_MESSAGE_ID = "0" GENERALIZATION = "Generalize" COMPOSITION = "Composite" AGGREGATION = "Aggregate" + +# For Android Assistant Agent +ADB_EXEC_FAIL = "FAILED" + +# For Mincraft Game Agent +MC_CKPT_DIR = METAGPT_ROOT / "data/mincraft/ckpt" +MC_LOG_DIR = METAGPT_ROOT / "logs" +MC_DEFAULT_WARMUP = { + "context": 15, + "biome": 10, + "time": 15, + "nearby_blocks": 0, + "other_blocks": 10, + "nearby_entities": 5, + "health": 15, + "hunger": 15, + "position": 0, + "equipment": 0, + "inventory": 0, + "optional_inventory_items": 7, + "chests": 0, + "completed_tasks": 0, + "failed_tasks": 0, +} +MC_CURRICULUM_OB = [ + "context", + "biome", + "time", + "nearby_blocks", + "other_blocks", + "nearby_entities", + "health", + "hunger", + "position", + "equipment", + "inventory", + "chests", + "completed_tasks", + "failed_tasks", +] +MC_CORE_INVENTORY_ITEMS = r".*_log|.*_planks|stick|crafting_table|furnace" +r"|cobblestone|dirt|coal|.*_pickaxe|.*_sword|.*_axe", # curriculum_agent: only show these items in inventory before optional_inventory_items reached in warm up diff --git a/metagpt/environment/__init__.py b/metagpt/environment/__init__.py new file mode 100644 index 000000000..692672fa7 --- /dev/null +++ b/metagpt/environment/__init__.py @@ -0,0 +1,13 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : + +from metagpt.environment.base_env import Environment +from metagpt.environment.android_env.android_env import AndroidEnv +from metagpt.environment.mincraft_env.mincraft_env import MincraftExtEnv +from metagpt.environment.werewolf_env.werewolf_env import WerewolfEnv +from metagpt.environment.stanford_town_env.stanford_town_env import StanfordTownEnv +from metagpt.environment.software_env.software_env import SoftwareEnv + + +__all__ = ["AndroidEnv", "MincraftExtEnv", "WerewolfEnv", "StanfordTownEnv", "SoftwareEnv", "Environment"] diff --git a/metagpt/environment/android_env/__init__.py b/metagpt/environment/android_env/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/metagpt/environment/android_env/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/metagpt/environment/android_env/android_env.py b/metagpt/environment/android_env/android_env.py new file mode 100644 index 000000000..c27e20541 --- /dev/null +++ b/metagpt/environment/android_env/android_env.py @@ -0,0 +1,13 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : MG Android Env + +from pydantic import Field + +from metagpt.environment.android_env.android_ext_env import AndroidExtEnv +from metagpt.environment.base_env import Environment + + +class AndroidEnv(Environment, AndroidExtEnv): + rows: int = Field(default=0, description="rows of a grid on the screenshot") + cols: int = Field(default=0, description="cols of a grid on the screenshot") diff --git a/metagpt/environment/android_env/android_ext_env.py b/metagpt/environment/android_env/android_ext_env.py new file mode 100644 index 000000000..7467d394c --- /dev/null +++ b/metagpt/environment/android_env/android_ext_env.py @@ -0,0 +1,157 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : The Android external environment to integrate with Android apps + +import subprocess +from pathlib import Path +from typing import Any, Optional + +from pydantic import Field + +from metagpt.const import ADB_EXEC_FAIL +from metagpt.environment.base_env import ExtEnv, mark_as_readable, mark_as_writeable + + +class AndroidExtEnv(ExtEnv): + device_id: Optional[str] = Field(default=None) + screenshot_dir: Optional[Path] = Field(default=None) + xml_dir: Optional[Path] = Field(default=None) + width: int = Field(default=720, description="device screen width") + height: int = Field(default=1080, description="device screen height") + + def __init__(self, **data: Any): + super().__init__(**data) + if data.get("device_id"): + (width, height) = self.device_shape + self.width = data.get("width", width) + self.height = data.get("height", height) + + @property + def adb_prefix_si(self): + """adb cmd prefix with `device_id` and `shell input`""" + return f"adb -s {self.device_id} shell input " + + @property + def adb_prefix_shell(self): + """adb cmd prefix with `device_id` and `shell`""" + return f"adb -s {self.device_id} shell " + + @property + def adb_prefix(self): + """adb cmd prefix with `device_id`""" + return f"adb -s {self.device_id} " + + def execute_adb_with_cmd(self, adb_cmd: str) -> str: + res = subprocess.run(adb_cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE, text=True) + exec_res = ADB_EXEC_FAIL + if not res.returncode: + exec_res = res.stdout.strip() + return exec_res + + @property + def device_shape(self) -> tuple[int, int]: + adb_cmd = f"{self.adb_prefix_shell} wm size" + shape = (0, 0) + shape_res = self.execute_adb_with_cmd(adb_cmd) + if shape_res != ADB_EXEC_FAIL: + shape = tuple(map(int, shape_res.split(": ")[1].split("x"))) + return shape + + def list_devices(self): + adb_cmd = "adb devices" + res = self.execute_adb_with_cmd(adb_cmd) + devices = [] + if res != ADB_EXEC_FAIL: + devices = res.split("\n")[1:] + devices = [device.split()[0] for device in devices] + return devices + + @mark_as_readable + def get_screenshot(self, ss_name: str, local_save_dir: Path) -> Path: + """ + ss_name: screenshot file name + local_save_dir: local dir to store image from virtual machine + """ + assert self.screenshot_dir + ss_remote_path = Path(self.screenshot_dir).joinpath(f"{ss_name}.png") + ss_cmd = f"{self.adb_prefix_shell} screencap -p {ss_remote_path}" + ss_res = self.execute_adb_with_cmd(ss_cmd) + + res = ADB_EXEC_FAIL + if ss_res != ADB_EXEC_FAIL: + ss_local_path = Path(local_save_dir).joinpath(f"{ss_name}.png") + pull_cmd = f"{self.adb_prefix} pull {ss_remote_path} {ss_local_path}" + pull_res = self.execute_adb_with_cmd(pull_cmd) + if pull_res != ADB_EXEC_FAIL: + res = ss_local_path + return Path(res) + + @mark_as_readable + def get_xml(self, xml_name: str, local_save_dir: Path) -> Path: + xml_remote_path = Path(self.xml_dir).joinpath(f"{xml_name}.xml") + dump_cmd = f"{self.adb_prefix_shell} uiautomator dump {xml_remote_path}" + xml_res = self.execute_adb_with_cmd(dump_cmd) + + res = ADB_EXEC_FAIL + if xml_res != ADB_EXEC_FAIL: + xml_local_path = Path(local_save_dir).joinpath(f"{xml_name}.xml") + pull_cmd = f"{self.adb_prefix} pull {xml_remote_path} {xml_local_path}" + pull_res = self.execute_adb_with_cmd(pull_cmd) + if pull_res != ADB_EXEC_FAIL: + res = xml_local_path + return Path(res) + + @mark_as_writeable + def system_back(self) -> str: + adb_cmd = f"{self.adb_prefix_si} keyevent KEYCODE_BACK" + back_res = self.execute_adb_with_cmd(adb_cmd) + return back_res + + @mark_as_writeable + def system_tap(self, x: int, y: int) -> str: + adb_cmd = f"{self.adb_prefix_si} tap {x} {y}" + tap_res = self.execute_adb_with_cmd(adb_cmd) + return tap_res + + @mark_as_writeable + def user_input(self, input_txt: str) -> str: + input_txt = input_txt.replace(" ", "%s").replace("'", "") + adb_cmd = f"{self.adb_prefix_si} text {input_txt}" + input_res = self.execute_adb_with_cmd(adb_cmd) + return input_res + + @mark_as_writeable + def user_longpress(self, x: int, y: int, duration: int = 500) -> str: + adb_cmd = f"{self.adb_prefix_si} swipe {x} {y} {x} {y} {duration}" + press_res = self.execute_adb_with_cmd(adb_cmd) + return press_res + + @mark_as_writeable + def user_swipe(self, x: int, y: int, orient: str = "up", dist: str = "medium", if_quick: bool = False) -> str: + dist_unit = int(self.width / 10) + if dist == "long": + dist_unit *= 3 + elif dist == "medium": + dist_unit *= 2 + + if orient == "up": + offset = 0, -2 * dist_unit + elif orient == "down": + offset = 0, 2 * dist_unit + elif orient == "left": + offset = -1 * dist_unit, 0 + elif orient == "right": + offset = dist_unit, 0 + else: + return ADB_EXEC_FAIL + + duration = 100 if if_quick else 400 + adb_cmd = f"{self.adb_prefix_si} swipe {x} {y} {x + offset[0]} {y + offset[1]} {duration}" + swipe_res = self.execute_adb_with_cmd(adb_cmd) + return swipe_res + + @mark_as_writeable + def user_swipe_to(self, start: tuple[int, int], end: tuple[int, int], duration: int = 400): + adb_cmd = f"{self.adb_prefix_si} swipe {start[0]} {start[1]} {end[0]} {end[1]} {duration}" + swipe_res = self.execute_adb_with_cmd(adb_cmd) + return swipe_res diff --git a/metagpt/environment/api/__init__.py b/metagpt/environment/api/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/metagpt/environment/api/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/metagpt/environment/api/env_api.py b/metagpt/environment/api/env_api.py new file mode 100644 index 000000000..6469e5b4c --- /dev/null +++ b/metagpt/environment/api/env_api.py @@ -0,0 +1,45 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : the environment api store + +from typing import Callable + +from pydantic import BaseModel, Field + + +class EnvAPIAbstract(BaseModel): + """api/interface summary description""" + + api_name: str = Field(default="", description="the api function name or id") + args: set = Field(default={}, description="the api function `args` params") + kwargs: dict = Field(default=dict(), description="the api function `kwargs` params") + + +class EnvAPIRegistry(BaseModel): + """the registry to store environment w&r api/interface""" + + registry: dict[str, Callable] = Field(default=dict(), exclude=True) + + def get(self, api_name: str): + return self.registry.get(api_name) + + def __getitem__(self, api_name: str) -> Callable: + return self.get(api_name) + + def __setitem__(self, api_name: str, func: Callable): + self.registry[api_name] = func + + def __len__(self): + return len(self.registry) + + +class WriteAPIRegistry(EnvAPIRegistry): + """just as a explicit class name""" + + pass + + +class ReadAPIRegistry(EnvAPIRegistry): + """just as a explicit class name""" + + pass diff --git a/metagpt/environment.py b/metagpt/environment/base_env.py similarity index 61% rename from metagpt/environment.py rename to metagpt/environment/base_env.py index 5a2dd339b..01582d8d8 100644 --- a/metagpt/environment.py +++ b/metagpt/environment/base_env.py @@ -1,29 +1,91 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- -""" -@Time : 2023/5/11 22:12 -@Author : alexanderwu -@File : environment.py -@Modified By: mashenquan, 2023-11-1. According to Chapter 2.2.2 of RFC 116: - 1. Remove the functionality of `Environment` class as a public message buffer. - 2. Standardize the message forwarding behavior of the `Environment` class. - 3. Add the `is_idle` property. -@Modified By: mashenquan, 2023-11-4. According to the routing feature plan in Chapter 2.2.3.2 of RFC 113, the routing - functionality is to be consolidated into the `Environment` class. -""" +# @Desc : base env of executing environment + +from enum import Enum +from typing import Iterable, Optional, Set, Union import asyncio -from typing import Iterable, Set from pydantic import BaseModel, ConfigDict, Field, SerializeAsAny, model_validator from metagpt.context import Context +from metagpt.environment.api.env_api import ( + EnvAPIAbstract, + ReadAPIRegistry, + WriteAPIRegistry, +) from metagpt.logs import logger from metagpt.roles.role import Role from metagpt.schema import Message -from metagpt.utils.common import is_send_to +from metagpt.utils.common import is_send_to, is_coroutine_func -class Environment(BaseModel): +class EnvType(Enum): + ANDROID = "Android" + GYM = "Gym" + WEREWOLF = "Werewolf" + MINCRAFT = "Mincraft" + STANFORDTOWN = "StanfordTown" + + +env_write_api_registry = WriteAPIRegistry() +env_read_api_registry = ReadAPIRegistry() + + +def mark_as_readable(func): + """mark functionn as a readable one in ExtEnv, it observes something from ExtEnv""" + env_read_api_registry[func.__name__] = func + return func + + +def mark_as_writeable(func): + """mark functionn as a writeable one in ExtEnv, it does something to ExtEnv""" + env_write_api_registry[func.__name__] = func + return func + + +class ExtEnv(BaseModel): + """External Env to intergate actual game environment""" + + def _check_api_exist(self, rw_api: Optional[str] = None): + if not rw_api: + raise ValueError(f"{rw_api} not exists") + + async def observe(self, env_action: Union[str, EnvAPIAbstract]): + """get observation from particular api of ExtEnv""" + if isinstance(env_action, str): + read_api = env_read_api_registry.get(api_name=env_action) + self._check_api_exist(read_api) + if is_coroutine_func(read_api): + res = await read_api(self) + else: + res = read_api(self) + elif isinstance(env_action, EnvAPIAbstract): + read_api = env_read_api_registry.get(api_name=env_action.api_name) + self._check_api_exist(read_api) + if is_coroutine_func(read_api): + res = await read_api(self, *env_action.args, **env_action.kwargs) + else: + res = read_api(self, *env_action.args, **env_action.kwargs) + return res + + async def step(self, env_action: Union[str, Message, EnvAPIAbstract, list[EnvAPIAbstract]]): + """execute through particular api of ExtEnv""" + res = None + if isinstance(env_action, Message): + self.publish_message(env_action) + elif isinstance(env_action, EnvAPIAbstract): + write_api = env_write_api_registry.get(env_action.api_name) + self._check_api_exist(write_api) + if is_coroutine_func(write_api): + res = await write_api(self, *env_action.args, **env_action.kwargs) + else: + res = write_api(self, *env_action.args, **env_action.kwargs) + + return res + + +class Environment(ExtEnv): """环境,承载一批角色,角色可以向环境发布消息,可以被其他角色观察到 Environment, hosting a batch of roles, roles can publish messages to the environment, and can be observed by other roles """ diff --git a/metagpt/logs.py b/metagpt/logs.py index fb0fdd553..90bac21aa 100644 --- a/metagpt/logs.py +++ b/metagpt/logs.py @@ -15,14 +15,15 @@ from loguru import logger as _logger from metagpt.const import METAGPT_ROOT -def define_log_level(print_level="INFO", logfile_level="DEBUG"): +def define_log_level(print_level="INFO", logfile_level="DEBUG", name: str = None): """Adjust the log level to above level""" current_date = datetime.now() formatted_date = current_date.strftime("%Y%m%d") + log_name = f"{name}_{formatted_date}" if name else formatted_date # name a log with prefix name _logger.remove() _logger.add(sys.stderr, level=print_level) - _logger.add(METAGPT_ROOT / f"logs/{formatted_date}.txt", level=logfile_level) + _logger.add(METAGPT_ROOT / f"logs/{log_name}.txt", level=logfile_level) return _logger diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 93921e983..4e7e25531 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -26,7 +26,7 @@ import traceback import typing from io import BytesIO from pathlib import Path -from typing import Any, List, Tuple, Union +from typing import Any, List, Tuple, Union, Callable import aiofiles import loguru @@ -603,6 +603,25 @@ def list_files(root: str | Path) -> List[Path]: return files +def is_coroutine_func(func: Callable) -> bool: + return inspect.iscoroutinefunction(func) + + +def load_mc_skills_code(skill_names: list[str] = None, skills_dir: Path = None) -> list[str]: + """load mincraft skill from js files""" + if not skills_dir: + skills_dir = Path(__file__).parent.absolute() + if skill_names is None: + skill_names = [ + skill[:-3] for skill in os.listdir(f"{skills_dir}") if skill.endswith(".js") + ] + skills = [ + skills_dir.joinpath(f"{skill_name}.js").read_text() + for skill_name in skill_names + ] + return skills + + def encode_image(image_path_or_pil: Union[Path, Image], encoding: str = "utf-8") -> str: """encode image from file or PIL.Image into base64""" if isinstance(image_path_or_pil, Image): diff --git a/tests/metagpt/environment/android_env/__init__.py b/tests/metagpt/environment/android_env/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/tests/metagpt/environment/android_env/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/tests/metagpt/environment/android_env/test_android_ext_env.py b/tests/metagpt/environment/android_env/test_android_ext_env.py new file mode 100644 index 000000000..955210868 --- /dev/null +++ b/tests/metagpt/environment/android_env/test_android_ext_env.py @@ -0,0 +1,75 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : the unittest of AndroidExtEnv + +from pathlib import Path + +from metagpt.const import ADB_EXEC_FAIL +from metagpt.environment.android_env.android_ext_env import AndroidExtEnv + + +def mock_device_shape(self, adb_cmd: str) -> str: + return "shape: 720x1080" + + +def mock_device_shape_invalid(self, adb_cmd: str) -> str: + return ADB_EXEC_FAIL + + +def mock_list_devices(self, adb_cmd: str) -> str: + return "devices\nemulator-5554" + + +def mock_get_screenshot(self, adb_cmd: str) -> str: + return "screenshot_xxxx-xx-xx" + + +def mock_get_xml(self, adb_cmd: str) -> str: + return "xml_xxxx-xx-xx" + + +def mock_write_read_operation(self, adb_cmd: str) -> str: + return "OK" + + +def test_android_ext_env(mocker): + device_id = "emulator-5554" + mocker.patch( + "metagpt.environment.android_env.android_ext_env.AndroidExtEnv.execute_adb_with_cmd", mock_device_shape + ) + + ext_env = AndroidExtEnv(device_id=device_id, screenshot_dir="/data2/", xml_dir="/data2/") + assert ext_env.adb_prefix == f"adb -s {device_id} " + assert ext_env.adb_prefix_shell == f"adb -s {device_id} shell " + assert ext_env.adb_prefix_si == f"adb -s {device_id} shell input " + + assert ext_env.device_shape == (720, 1080) + + mocker.patch( + "metagpt.environment.android_env.android_ext_env.AndroidExtEnv.execute_adb_with_cmd", mock_device_shape_invalid + ) + assert ext_env.device_shape == (0, 0) + + mocker.patch( + "metagpt.environment.android_env.android_ext_env.AndroidExtEnv.execute_adb_with_cmd", mock_list_devices + ) + assert ext_env.list_devices() == [device_id] + + mocker.patch( + "metagpt.environment.android_env.android_ext_env.AndroidExtEnv.execute_adb_with_cmd", mock_get_screenshot + ) + assert ext_env.get_screenshot("screenshot_xxxx-xx-xx", "/data/") == Path("/data/screenshot_xxxx-xx-xx.png") + + mocker.patch("metagpt.environment.android_env.android_ext_env.AndroidExtEnv.execute_adb_with_cmd", mock_get_xml) + assert ext_env.get_xml("xml_xxxx-xx-xx", "/data/") == Path("/data/xml_xxxx-xx-xx.xml") + + mocker.patch( + "metagpt.environment.android_env.android_ext_env.AndroidExtEnv.execute_adb_with_cmd", mock_write_read_operation + ) + res = "OK" + assert ext_env.system_back() == res + assert ext_env.system_tap(10, 10) == res + assert ext_env.user_input("test_input") == res + assert ext_env.user_longpress(10, 10) == res + assert ext_env.user_swipe(10, 10) == res + assert ext_env.user_swipe_to((10, 10), (20, 20)) == res diff --git a/tests/metagpt/environment/api/__init__.py b/tests/metagpt/environment/api/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/tests/metagpt/environment/api/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/tests/metagpt/environment/api/test_env_api.py b/tests/metagpt/environment/api/test_env_api.py new file mode 100644 index 000000000..53f98c0d3 --- /dev/null +++ b/tests/metagpt/environment/api/test_env_api.py @@ -0,0 +1,15 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : + +from metagpt.environment.api.env_api import EnvAPIRegistry + + +def test_env_api_registry(): + def test_func(): + pass + + env_api_registry = EnvAPIRegistry() + env_api_registry["test"] = test_func + + env_api_registry.get("test") == test_func diff --git a/tests/metagpt/environment/test_base_env.py b/tests/metagpt/environment/test_base_env.py new file mode 100644 index 000000000..7404f9bf6 --- /dev/null +++ b/tests/metagpt/environment/test_base_env.py @@ -0,0 +1,50 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : the unittest of ExtEnv&Env + +import pytest + +from metagpt.environment.api.env_api import EnvAPIAbstract +from metagpt.environment.base_env import ( + Environment, + mark_as_readable, + mark_as_writeable, + env_read_api_registry, + env_write_api_registry +) + + +class ForTestEnv(Environment): + value: int = 0 + + @mark_as_readable + def read_api_no_parms(self): + return self.value + + @mark_as_readable + def read_api(self, a: int, b: int): + return a + b + + @mark_as_writeable + def write_api(self, a: int, b: int): + self.value = a + b + + @mark_as_writeable + async def async_read_api(self, a: int, b: int): + return a + b + + +@pytest.mark.asyncio +async def test_ext_env(): + env = ForTestEnv() + assert len(env_read_api_registry) > 0 + assert len(env_write_api_registry) > 0 + + _ = await env.step(EnvAPIAbstract(api_name="write_api", kwargs={"a": 5, "b": 10})) + assert env.value == 15 + + with pytest.raises(ValueError): + await env.observe("not_exist_api") + + assert await env.observe("read_api_no_parms") == 15 + assert await env.observe(EnvAPIAbstract(api_name="read_api", kwargs={"a": 5, "b": 5})) == 10 From 210a00c1e7ccca6fe1aca0766d8ecb9bb7095b5c Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 19:08:55 +0800 Subject: [PATCH 506/637] add werewolf_env --- metagpt/environment/base_env.py | 4 +- metagpt/environment/werewolf_env/__init__.py | 3 + .../environment/werewolf_env/werewolf_env.py | 31 ++ .../werewolf_env/werewolf_ext_env.py | 274 ++++++++++++++++++ metagpt/utils/common.py | 11 +- tests/metagpt/environment/test_base_env.py | 4 +- .../environment/werewolf_env/__init__.py | 3 + .../werewolf_env/test_werewolf_ext_env.py | 29 ++ 8 files changed, 347 insertions(+), 12 deletions(-) create mode 100644 metagpt/environment/werewolf_env/__init__.py create mode 100644 metagpt/environment/werewolf_env/werewolf_env.py create mode 100644 metagpt/environment/werewolf_env/werewolf_ext_env.py create mode 100644 tests/metagpt/environment/werewolf_env/__init__.py create mode 100644 tests/metagpt/environment/werewolf_env/test_werewolf_ext_env.py diff --git a/metagpt/environment/base_env.py b/metagpt/environment/base_env.py index 01582d8d8..1bdcfe373 100644 --- a/metagpt/environment/base_env.py +++ b/metagpt/environment/base_env.py @@ -2,9 +2,9 @@ # -*- coding: utf-8 -*- # @Desc : base env of executing environment +import asyncio from enum import Enum from typing import Iterable, Optional, Set, Union -import asyncio from pydantic import BaseModel, ConfigDict, Field, SerializeAsAny, model_validator @@ -17,7 +17,7 @@ from metagpt.environment.api.env_api import ( from metagpt.logs import logger from metagpt.roles.role import Role from metagpt.schema import Message -from metagpt.utils.common import is_send_to, is_coroutine_func +from metagpt.utils.common import is_coroutine_func, is_send_to class EnvType(Enum): diff --git a/metagpt/environment/werewolf_env/__init__.py b/metagpt/environment/werewolf_env/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/metagpt/environment/werewolf_env/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/metagpt/environment/werewolf_env/werewolf_env.py b/metagpt/environment/werewolf_env/werewolf_env.py new file mode 100644 index 000000000..d174f322c --- /dev/null +++ b/metagpt/environment/werewolf_env/werewolf_env.py @@ -0,0 +1,31 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : MG Werewolf Env + +from pydantic import Field + +from metagpt.environment.base_env import Environment +from metagpt.environment.werewolf_env.werewolf_ext_env import WerewolfExtEnv +from metagpt.logs import logger +from metagpt.schema import Message + + +class WerewolfEnv(Environment, WerewolfExtEnv): + timestamp: int = Field(default=0) + + def publish_message(self, message: Message, add_timestamp: bool = True): + """Post information to the current environment""" + logger.debug(f"publish_message: {message.dump()}") + if add_timestamp: + # Because the content of the message may be repeated, for example, killing the same person in two nights + # Therefore, a unique timestamp prefix needs to be added so that the same message will not be automatically deduplicated when added to the memory. + message.content = f"{self.timestamp} | " + message.content + self.memory.add(message) + self.history += f"\n{message}" + + async def run(self, k=1): + """Process all Role runs by order""" + for _ in range(k): + for role in self.roles.values(): + await role.run() + self.timestamp += 1 diff --git a/metagpt/environment/werewolf_env/werewolf_ext_env.py b/metagpt/environment/werewolf_env/werewolf_ext_env.py new file mode 100644 index 000000000..c3d34b1aa --- /dev/null +++ b/metagpt/environment/werewolf_env/werewolf_ext_env.py @@ -0,0 +1,274 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : The werewolf game external environment to integrate with + +import random +import re +from collections import Counter +from enum import Enum +from typing import Callable, Optional + +from pydantic import ConfigDict, Field + +from metagpt.environment.base_env import ExtEnv, mark_as_readable, mark_as_writeable +from metagpt.logs import logger + + +class RoleState(Enum): + ALIVE = "alive" # the role is alive + KILLED = "killed" # the role is killed by werewolf or voting + POISONED = "poisoned" # the role is killed by posion + SAVED = "saved" # the role is saved by antidote + + +# the ordered rules by the moderator to announce to everyone each step +STEP_INSTRUCTIONS = { + 0: { + "content": "It’s dark, everyone close your eyes. I will talk with you/your team secretly at night.", + "send_to": "Moderator", # for moderator to continuen speaking + "restricted_to": "", + }, + 1: { + "content": "Guard, please open your eyes!", + "send_to": "Moderator", # for moderator to continuen speaking + "restricted_to": "", + }, + 2: { + "content": """Guard, now tell me who you protect tonight? + You only choose one from the following living options please: {living_players}. + Or you can pass. For example: Protect ...""", + "send_to": "Guard", + "restricted_to": "Moderator,Guard", + }, + 3: {"content": "Guard, close your eyes", "send_to": "Moderator", "restricted_to": ""}, + 4: {"content": "Werewolves, please open your eyes!", "send_to": "Moderator", "restricted_to": ""}, + 5: { + "content": """Werewolves, I secretly tell you that {werewolf_players} are + all of the 2 werewolves! Keep in mind you are teammates. The rest players are not werewolves. + choose one from the following living options please: + {living_players}. For example: Kill ...""", + "send_to": "Werewolf", + "restricted_to": "Moderator,Werewolf", + }, + 6: {"content": "Werewolves, close your eyes", "send_to": "Moderator", "restricted_to": ""}, + 7: {"content": "Witch, please open your eyes!", "send_to": "Moderator", "restricted_to": ""}, + 8: { + "content": """Witch, tonight {player_hunted} has been killed by the werewolves. + You have a bottle of antidote, would you like to save him/her? If so, say "Save", else, say "Pass".""", + "send_to": "Witch", + "restricted_to": "Moderator,Witch", + }, # 要先判断女巫是否有解药,再去询问女巫是否使用解药救人 + 9: { + "content": """Witch, you also have a bottle of poison, would you like to use it to kill one of the living players? + Choose one from the following living options: {living_players}. + If so, say ONLY "Poison PlayerX", replace PlayerX with the actual player name, else, say "Pass".""", + "send_to": "Witch", + "restricted_to": "Moderator,Witch", + }, # + 10: {"content": "Witch, close your eyes", "send_to": "Moderator", "restricted_to": ""}, + 11: {"content": "Seer, please open your eyes!", "send_to": "Moderator", "restricted_to": ""}, + 12: { + "content": """Seer, you can check one player's identity. Who are you going to verify its identity tonight? + Choose only one from the following living options:{living_players}.""", + "send_to": "Seer", + "restricted_to": "Moderator,Seer", + }, + 13: {"content": "Seer, close your eyes", "send_to": "Moderator", "restricted_to": ""}, + # The 1-st daytime + 14: { + "content": """It's daytime. Everyone woke up except those who had been killed.""", + "send_to": "Moderator", + "restricted_to": "", + }, + 15: {"content": "{player_current_dead} was killed last night!", "send_to": "Moderator", "restricted_to": ""}, + 16: { + "content": """Living players: {living_players}, now freely talk about the current situation based on your observation and + reflection with a few sentences. Decide whether to reveal your identity based on your reflection.""", + "send_to": "", # send to all to speak in daytime + "restricted_to": "", + }, + 17: { + "content": """Now vote and tell me who you think is the werewolf. Don’t mention your role. + You only choose one from the following living options please: + {living_players}. Say ONLY: I vote to eliminate ...""", + "send_to": "", + "restricted_to": "", + }, + 18: {"content": """{player_current_dead} was eliminated.""", "send_to": "Moderator", "restricted_to": ""}, +} + + +class WerewolfExtEnv(ExtEnv): + model_config = ConfigDict(arbitrary_types_allowed=True) + + roles_state: dict[str, RoleState] = Field(default=dict(), description="the role's current state by role_name") + + step_idx: int = Field(default=0) # the current step of current round + eval_step_idx: int = Field(default=0) + per_round_steps: int = Field(default=len(STEP_INSTRUCTIONS)) + + # game global states + game_setup: str = Field(default="", description="game setup including role and its num") + living_players: list[str] = Field(default=[]) + werewolf_players: list[str] = Field(default=[]) + villager_players: list[str] = Field(default=[]) + special_role_players: list[str] = Field(default=[]) + winner: Optional[str] = Field(default=None) + win_reason: Optional[str] = Field(default=None) + witch_poison_left: int = Field(default=1) + witch_antidote_left: int = Field(default=1) + + # game current round states, a round is from closing your eyes to the next time you close your eyes + player_hunted: Optional[str] = Field(default=None) + player_protected: Optional[str] = Field(default=None) + is_hunted_player_saved: bool = Field(default=False) + player_poisoned: Optional[str] = Field(default=None) + player_current_dead: list[str] = Field(default=[]) + + def parse_game_setup(self, game_setup: str): + self.game_setup = game_setup + self.living_players = re.findall(r"Player[0-9]+", game_setup) + self.werewolf_players = re.findall(r"Player[0-9]+: Werewolf", game_setup) + self.werewolf_players = [p.replace(": Werewolf", "") for p in self.werewolf_players] + self.villager_players = re.findall(r"Player[0-9]+: Villager", game_setup) + self.villager_players = [p.replace(": Villager", "") for p in self.villager_players] + self.special_role_players = [ + p for p in self.living_players if p not in self.werewolf_players + self.villager_players + ] + + # init role state + self.roles_state = {player_name: RoleState.ALIVE for player_name in self.living_players} + + @mark_as_readable + def init_game_setup( + self, + role_uniq_objs: list[object], + num_villager: int = 2, + num_werewolf: int = 2, + shuffle=True, + add_human=False, + use_reflection=True, + use_experience=False, + use_memory_selection=False, + new_experience_version="", + prepare_human_player=Callable, + ) -> tuple[str, list]: + role_objs = [] + for role_obj in role_uniq_objs: + if str(role_obj) == "Villager": + role_objs.extend([role_obj] * num_villager) + elif str(role_obj) == "Werewolf": + role_objs.extend([role_obj] * num_werewolf) + else: + role_objs.append(role_obj) + if shuffle: + random.shuffle(len(role_objs)) + if add_human: + assigned_role_idx = random.randint(0, len(role_objs) - 1) + assigned_role = role_objs[assigned_role_idx] + role_objs[assigned_role_idx] = prepare_human_player(assigned_role) # TODO + + players = [ + role( + name=f"Player{i + 1}", + use_reflection=use_reflection, + use_experience=use_experience, + use_memory_selection=use_memory_selection, + new_experience_version=new_experience_version, + ) + for i, role in enumerate(role_objs) + ] + + if add_human: + logger.info(f"You are assigned {players[assigned_role_idx].name}({players[assigned_role_idx].profile})") + + game_setup = ["Game setup:"] + [f"{player.name}: {player.profile}," for player in players] + game_setup = "\n".join(game_setup) + + return game_setup, players + + @mark_as_readable + def curr_step_instruction(self) -> dict: + step_idx = self.step_idx % len(STEP_INSTRUCTIONS) + instruction = STEP_INSTRUCTIONS[step_idx] + self.step_idx += 1 + return instruction + + @mark_as_writeable + def update_players_state(self, player_names: list[str], state: RoleState = RoleState.KILLED): + for player_name in player_names: + if player_name in self.roles_state: + self.roles_state[player_name] = state + + @mark_as_readable + def get_players_status(self, player_names: list[str]) -> dict[str, RoleState]: + roles_state = { + player_name: self.roles_state[player_name] + for player_name in player_names + if player_name in self.roles_state + } + return roles_state + + @mark_as_writeable + def wolf_kill_someone(self, player_name: str): + self.update_players_state([player_name], RoleState.KILLED) + + @mark_as_writeable + def witch_poison_someone(self, player_name: str = None): + self.update_players_state([player_name], RoleState.POISONED) + + @mark_as_writeable + def witch_save_someone(self, player_name: str = None): + self.update_players_state([player_name], RoleState.SAVED) + + @mark_as_writeable + def update_game_states(self, memories: list): + step_idx = self.step_idx % self.per_round_steps + if step_idx not in [15, 18] or self.step_idx in self.eval_step_idx: + return + else: + self.eval_step_idx.append(self.step_idx) # record evaluation, avoid repetitive evaluation at the same step + + if step_idx == 15: # step no + # night ends: after all special roles acted, process the whole night + self.player_current_dead = [] # reset + + if self.player_hunted != self.player_protected and not self.is_hunted_player_saved: + self.player_current_dead.append(self.player_hunted) + if self.player_poisoned: + self.player_current_dead.append(self.player_poisoned) + + self.living_players = [p for p in self.living_players if p not in self.player_current_dead] + self.update_player_status(self.player_current_dead) + # reset + self.player_hunted = None + self.player_protected = None + self.is_hunted_player_saved = False + self.player_poisoned = None + + elif step_idx == 18: # step no + # day ends: after all roles voted, process all votings + voting_msgs = memories[-len(self.living_players) :] + voted_all = [] + for msg in voting_msgs: + voted = re.search(r"Player[0-9]+", msg.content[-10:]) + if not voted: + continue + voted_all.append(voted.group(0)) + self.player_current_dead = [Counter(voted_all).most_common()[0][0]] # 平票时,杀最先被投的 + # print("*" * 10, "dead", self.player_current_dead) + self.living_players = [p for p in self.living_players if p not in self.player_current_dead] + self.update_player_status(self.player_current_dead) + + # game's termination condition + living_werewolf = [p for p in self.werewolf_players if p in self.living_players] + living_villagers = [p for p in self.villager_players if p in self.living_players] + living_special_roles = [p for p in self.special_role_players if p in self.living_players] + if not living_werewolf: + self.winner = "good guys" + self.win_reason = "werewolves all dead" + elif not living_villagers or not living_special_roles: + self.winner = "werewolf" + self.win_reason = "villagers all dead" if not living_villagers else "special roles all dead" + if self.winner is not None: + self._record_all_experiences() # TODO diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 4e7e25531..79e7de5b6 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -26,7 +26,7 @@ import traceback import typing from io import BytesIO from pathlib import Path -from typing import Any, List, Tuple, Union, Callable +from typing import Any, Callable, List, Tuple, Union import aiofiles import loguru @@ -612,13 +612,8 @@ def load_mc_skills_code(skill_names: list[str] = None, skills_dir: Path = None) if not skills_dir: skills_dir = Path(__file__).parent.absolute() if skill_names is None: - skill_names = [ - skill[:-3] for skill in os.listdir(f"{skills_dir}") if skill.endswith(".js") - ] - skills = [ - skills_dir.joinpath(f"{skill_name}.js").read_text() - for skill_name in skill_names - ] + skill_names = [skill[:-3] for skill in os.listdir(f"{skills_dir}") if skill.endswith(".js")] + skills = [skills_dir.joinpath(f"{skill_name}.js").read_text() for skill_name in skill_names] return skills diff --git a/tests/metagpt/environment/test_base_env.py b/tests/metagpt/environment/test_base_env.py index 7404f9bf6..59c68fc9e 100644 --- a/tests/metagpt/environment/test_base_env.py +++ b/tests/metagpt/environment/test_base_env.py @@ -7,10 +7,10 @@ import pytest from metagpt.environment.api.env_api import EnvAPIAbstract from metagpt.environment.base_env import ( Environment, + env_read_api_registry, + env_write_api_registry, mark_as_readable, mark_as_writeable, - env_read_api_registry, - env_write_api_registry ) diff --git a/tests/metagpt/environment/werewolf_env/__init__.py b/tests/metagpt/environment/werewolf_env/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/tests/metagpt/environment/werewolf_env/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/tests/metagpt/environment/werewolf_env/test_werewolf_ext_env.py b/tests/metagpt/environment/werewolf_env/test_werewolf_ext_env.py new file mode 100644 index 000000000..a24328143 --- /dev/null +++ b/tests/metagpt/environment/werewolf_env/test_werewolf_ext_env.py @@ -0,0 +1,29 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : the unittest of WerewolfExtEnv + + +from metagpt.environment.werewolf_env.werewolf_ext_env import RoleState, WerewolfExtEnv + + +def test_werewolf_ext_env(): + ext_env = WerewolfExtEnv() + + game_setup = """Game setup: + Player0: Werewolf, + Player1: Werewolf, + Player2: Villager, + Player3: Guard, + """ + ext_env.parse_game_setup(game_setup) + assert len(ext_env.living_players) == 4 + assert len(ext_env.special_role_players) == 1 + assert len(ext_env.werewolf_players) == 2 + + curr_instr = ext_env.curr_step_instruction() + assert ext_env.step_idx == 1 + assert "close your eyes" in curr_instr["content"] + + player_names = ["Player0", "Player2"] + ext_env.update_players_state(player_names, RoleState.KILLED) + assert ext_env.get_players_status(player_names) == dict(zip(player_names, [RoleState.KILLED] * 2)) From 29e8a076bd96c841c22fd5e606e39914b2f8652f Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 19:09:27 +0800 Subject: [PATCH 507/637] add stanford_town_env --- .../environment/stanford_town_env/__init__.py | 3 + .../stanford_town_env/stanford_town_env.py | 12 + .../stanford_town_ext_env.py | 379 ++++++++++++++++++ .../environment/stanford_town_env/__init__.py | 3 + .../test_stanford_town_ext_env.py | 40 ++ 5 files changed, 437 insertions(+) create mode 100644 metagpt/environment/stanford_town_env/__init__.py create mode 100644 metagpt/environment/stanford_town_env/stanford_town_env.py create mode 100644 metagpt/environment/stanford_town_env/stanford_town_ext_env.py create mode 100644 tests/metagpt/environment/stanford_town_env/__init__.py create mode 100644 tests/metagpt/environment/stanford_town_env/test_stanford_town_ext_env.py diff --git a/metagpt/environment/stanford_town_env/__init__.py b/metagpt/environment/stanford_town_env/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/metagpt/environment/stanford_town_env/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/metagpt/environment/stanford_town_env/stanford_town_env.py b/metagpt/environment/stanford_town_env/stanford_town_env.py new file mode 100644 index 000000000..8721d6cd1 --- /dev/null +++ b/metagpt/environment/stanford_town_env/stanford_town_env.py @@ -0,0 +1,12 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : MG StanfordTown Env + +from metagpt.environment.base_env import Environment +from metagpt.environment.stanford_town_env.stanford_town_ext_env import ( + StanfordTownExtEnv, +) + + +class StanfordTownEnv(Environment, StanfordTownExtEnv): + pass diff --git a/metagpt/environment/stanford_town_env/stanford_town_ext_env.py b/metagpt/environment/stanford_town_env/stanford_town_ext_env.py new file mode 100644 index 000000000..8a9a65965 --- /dev/null +++ b/metagpt/environment/stanford_town_env/stanford_town_ext_env.py @@ -0,0 +1,379 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : The StanfordTown external environment to interate with the web interface +# refs to `generative_agents maze.py` + +import math +from pathlib import Path +from typing import Optional, Tuple + +from pydantic import ConfigDict, Field, model_validator + +from metagpt.environment.base_env import ExtEnv, mark_as_readable, mark_as_writeable +from metagpt.utils.common import read_csv_to_list, read_json_file + + +class StanfordTownExtEnv(ExtEnv): + model_config = ConfigDict(arbitrary_types_allowed=True) + + maze_asset_path: Optional[Path] = Field(default=None, description="the path to store maze assets") + maze_width: int = Field(default=140, description="maze map width") + maze_height: int = Field(default=100, description="maze map height") + sq_tile_size: int = Field(default=32, description="the pixel height/width of a tile") + special_constraint: str = Field( + default="", description="a string description of any relevant special constraints " "the world might have" + ) + tiles: list[list[dict]] = Field(default=[]) + address_tiles: dict[str, set] = Field(default=dict()) + collision_maze: list[list] = Field(default=[]) + + @model_validator(mode="before") + @classmethod + def _init_maze(cls, values): + maze_asset_path = values["maze_asset_path"] + assert maze_asset_path + maze_asset_path = Path(maze_asset_path) + + maze_matrix_path = maze_asset_path.joinpath("matrix") + meta_info = read_json_file(maze_matrix_path.joinpath("maze_meta_info.json")) + + maze_width = int(meta_info["maze_width"]) + maze_height = int(meta_info["maze_height"]) + values["maze_width"] = maze_width + values["maze_height"] = maze_height + values["sq_tile_size"] = int(meta_info["sq_tile_size"]) + values["special_constraint"] = meta_info["special_constraint"] + + # READING IN SPECIAL BLOCKS + # Special blocks are those that are colored in the Tiled map. + # Here is an example row for the arena block file: + # e.g, "25331, Double Studio, Studio, Bedroom 2, Painting" + + blocks_folder = maze_matrix_path.joinpath("special_blocks") + + _wb = blocks_folder.joinpath("world_blocks.csv") + wb_rows = read_csv_to_list(_wb, header=False) + wb = wb_rows[0][-1] + + _sb = blocks_folder.joinpath("sector_blocks.csv") + sb_rows = read_csv_to_list(_sb, header=False) + sb_dict = dict() + for i in sb_rows: + sb_dict[i[0]] = i[-1] + + _ab = blocks_folder.joinpath("arena_blocks.csv") + ab_rows = read_csv_to_list(_ab, header=False) + ab_dict = dict() + for i in ab_rows: + ab_dict[i[0]] = i[-1] + + _gob = blocks_folder.joinpath("game_object_blocks.csv") + gob_rows = read_csv_to_list(_gob, header=False) + gob_dict = dict() + for i in gob_rows: + gob_dict[i[0]] = i[-1] + + _slb = blocks_folder.joinpath("spawning_location_blocks.csv") + slb_rows = read_csv_to_list(_slb, header=False) + slb_dict = dict() + for i in slb_rows: + slb_dict[i[0]] = i[-1] + + # [SECTION 3] Reading in the matrices + # This is your typical two dimensional matrices. It's made up of 0s and + # the number that represents the color block from the blocks folder. + maze_folder = maze_matrix_path.joinpath("maze") + + _cm = maze_folder.joinpath("collision_maze.csv") + collision_maze_raw = read_csv_to_list(_cm, header=False)[0] + _sm = maze_folder.joinpath("sector_maze.csv") + sector_maze_raw = read_csv_to_list(_sm, header=False)[0] + _am = maze_folder.joinpath("arena_maze.csv") + arena_maze_raw = read_csv_to_list(_am, header=False)[0] + _gom = maze_folder.joinpath("game_object_maze.csv") + game_object_maze_raw = read_csv_to_list(_gom, header=False)[0] + _slm = maze_folder.joinpath("spawning_location_maze.csv") + spawning_location_maze_raw = read_csv_to_list(_slm, header=False)[0] + + # Loading the maze. The mazes are taken directly from the json exports of + # Tiled maps. They should be in csv format. + # Importantly, they are "not" in a 2-d matrix format -- they are single + # row matrices with the length of width x height of the maze. So we need + # to convert here. + # example format: [['0', '0', ... '25309', '0',...], ['0',...]...] + # 25309 is the collision bar number right now. + collision_maze = [] + sector_maze = [] + arena_maze = [] + game_object_maze = [] + spawning_location_maze = [] + for i in range(0, len(collision_maze_raw), maze_width): + tw = maze_width + collision_maze += [collision_maze_raw[i : i + tw]] + sector_maze += [sector_maze_raw[i : i + tw]] + arena_maze += [arena_maze_raw[i : i + tw]] + game_object_maze += [game_object_maze_raw[i : i + tw]] + spawning_location_maze += [spawning_location_maze_raw[i : i + tw]] + values["collision_maze"] = collision_maze + + tiles = [] + for i in range(maze_height): + row = [] + for j in range(maze_width): + tile_details = dict() + tile_details["world"] = wb + + tile_details["sector"] = "" + if sector_maze[i][j] in sb_dict: + tile_details["sector"] = sb_dict[sector_maze[i][j]] + + tile_details["arena"] = "" + if arena_maze[i][j] in ab_dict: + tile_details["arena"] = ab_dict[arena_maze[i][j]] + + tile_details["game_object"] = "" + if game_object_maze[i][j] in gob_dict: + tile_details["game_object"] = gob_dict[game_object_maze[i][j]] + + tile_details["spawning_location"] = "" + if spawning_location_maze[i][j] in slb_dict: + tile_details["spawning_location"] = slb_dict[spawning_location_maze[i][j]] + + tile_details["collision"] = False + if collision_maze[i][j] != "0": + tile_details["collision"] = True + + tile_details["events"] = set() + + row += [tile_details] + tiles += [row] + values["tiles"] = tiles + + # Each game object occupies an event in the tile. We are setting up the + # default event value here. + for i in range(maze_height): + for j in range(maze_width): + if tiles[i][j]["game_object"]: + object_name = ":".join( + [tiles[i][j]["world"], tiles[i][j]["sector"], tiles[i][j]["arena"], tiles[i][j]["game_object"]] + ) + go_event = (object_name, None, None, None) + tiles[i][j]["events"].add(go_event) + + # Reverse tile access. + # -- given a string address, we return a set of all + # tile coordinates belonging to that address (this is opposite of + # tiles that give you the string address given a coordinate). This is + # an optimization component for finding paths for the personas' movement. + # address_tiles['bedroom-2-a'] == {(58, 9)} + # address_tiles['double studio:recreation:pool table'] + # == {(29, 14), (31, 11), (30, 14), (32, 11), ...}, + address_tiles = dict() + for i in range(maze_height): + for j in range(maze_width): + addresses = [] + if tiles[i][j]["sector"]: + add = f'{tiles[i][j]["world"]}:' + add += f'{tiles[i][j]["sector"]}' + addresses += [add] + if tiles[i][j]["arena"]: + add = f'{tiles[i][j]["world"]}:' + add += f'{tiles[i][j]["sector"]}:' + add += f'{tiles[i][j]["arena"]}' + addresses += [add] + if tiles[i][j]["game_object"]: + add = f'{tiles[i][j]["world"]}:' + add += f'{tiles[i][j]["sector"]}:' + add += f'{tiles[i][j]["arena"]}:' + add += f'{tiles[i][j]["game_object"]}' + addresses += [add] + if tiles[i][j]["spawning_location"]: + add = f'{tiles[i][j]["spawning_location"]}' + addresses += [add] + + for add in addresses: + if add in address_tiles: + address_tiles[add].add((j, i)) + else: + address_tiles[add] = set([(j, i)]) + values["address_tiles"] = address_tiles + return values + + def turn_coordinate_to_tile(self, px_coordinate: tuple[int, int]) -> tuple[int, int]: + """ + Turns a pixel coordinate to a tile coordinate. + """ + x = math.ceil(px_coordinate[0] / self.sq_tile_size) + y = math.ceil(px_coordinate[1] / self.sq_tile_size) + return (x, y) + + @mark_as_readable + def get_collision_maze(self) -> list: + return self.collision_maze + + @mark_as_readable + def get_address_tiles(self) -> dict: + return self.address_tiles + + @mark_as_readable + def access_tile(self, tile: tuple[int, int]) -> dict: + """ + Returns the tiles details dictionary that is stored in self.tiles of the + designated x, y location. + + INPUT + tile: The tile coordinate of our interest in (x, y) form. + OUTPUT + The tile detail dictionary for the designated tile. + EXAMPLE OUTPUT + Given (58, 9), + self.tiles[9][58] = {'world': 'double studio', + 'sector': 'double studio', 'arena': 'bedroom 2', + 'game_object': 'bed', 'spawning_location': 'bedroom-2-a', + 'collision': False, + 'events': {('double studio:double studio:bedroom 2:bed', + None, None)}} + """ + x = tile[0] + y = tile[1] + return self.tiles[y][x] + + @mark_as_readable + def get_tile_path(self, tile: tuple[int, int], level: str) -> str: + """ + Get the tile string address given its coordinate. You designate the level + by giving it a string level description. + + INPUT: + tile: The tile coordinate of our interest in (x, y) form. + level: world, sector, arena, or game object + OUTPUT + The string address for the tile. + EXAMPLE OUTPUT + Given tile=(58, 9), and level=arena, + "double studio:double studio:bedroom 2" + """ + x = tile[0] + y = tile[1] + tile = self.tiles[y][x] + + path = f"{tile['world']}" + if level == "world": + return path + else: + path += f":{tile['sector']}" + + if level == "sector": + return path + else: + path += f":{tile['arena']}" + + if level == "arena": + return path + else: + path += f":{tile['game_object']}" + + return path + + @mark_as_readable + def get_nearby_tiles(self, tile: tuple[int, int], vision_r: int) -> list[tuple[int, int]]: + """ + Given the current tile and vision_r, return a list of tiles that are + within the radius. Note that this implementation looks at a square + boundary when determining what is within the radius. + i.e., for vision_r, returns x's. + x x x x x + x x x x x + x x P x x + x x x x x + x x x x x + + INPUT: + tile: The tile coordinate of our interest in (x, y) form. + vision_r: The radius of the persona's vision. + OUTPUT: + nearby_tiles: a list of tiles that are within the radius. + """ + left_end = 0 + if tile[0] - vision_r > left_end: + left_end = tile[0] - vision_r + + right_end = self.maze_width - 1 + if tile[0] + vision_r + 1 < right_end: + right_end = tile[0] + vision_r + 1 + + bottom_end = self.maze_height - 1 + if tile[1] + vision_r + 1 < bottom_end: + bottom_end = tile[1] + vision_r + 1 + + top_end = 0 + if tile[1] - vision_r > top_end: + top_end = tile[1] - vision_r + + nearby_tiles = [] + for i in range(left_end, right_end): + for j in range(top_end, bottom_end): + nearby_tiles += [(i, j)] + return nearby_tiles + + @mark_as_writeable + def add_tiles_event(self, pt_y: int, pt_x: int, event: Tuple[str, str, str, str]): + self.tiles[pt_y][pt_x]["events"].add(event) + + @mark_as_writeable + def add_event_from_tile(self, curr_event: tuple[str], tile: tuple[int, int]) -> None: + """ + Add an event triple to a tile. + + INPUT: + curr_event: Current event triple. + e.g., ('double studio:double studio:bedroom 2:bed', None, + None) + tile: The tile coordinate of our interest in (x, y) form. + OUPUT: + None + """ + self.tiles[tile[1]][tile[0]]["events"].add(curr_event) + + @mark_as_writeable + def remove_event_from_tile(self, curr_event: tuple[str], tile: tuple[int, int]) -> None: + """dswaq + Remove an event triple from a tile. + + INPUT: + curr_event: Current event triple. + e.g., ('double studio:double studio:bedroom 2:bed', None, + None) + tile: The tile coordinate of our interest in (x, y) form. + OUPUT: + None + """ + curr_tile_ev_cp = self.tiles[tile[1]][tile[0]]["events"].copy() + for event in curr_tile_ev_cp: + if event == curr_event: + self.tiles[tile[1]][tile[0]]["events"].remove(event) + + @mark_as_writeable + def turn_event_from_tile_idle(self, curr_event: tuple[str], tile: tuple[int, int]) -> None: + curr_tile_ev_cp = self.tiles[tile[1]][tile[0]]["events"].copy() + for event in curr_tile_ev_cp: + if event == curr_event: + self.tiles[tile[1]][tile[0]]["events"].remove(event) + new_event = (event[0], None, None, None) + self.tiles[tile[1]][tile[0]]["events"].add(new_event) + + @mark_as_writeable + def remove_subject_events_from_tile(self, subject: str, tile: tuple[int, int]) -> None: + """ + Remove an event triple that has the input subject from a tile. + + INPUT: + subject: "Isabella Rodriguez" + tile: The tile coordinate of our interest in (x, y) form. + OUPUT: + None + """ + curr_tile_ev_cp = self.tiles[tile[1]][tile[0]]["events"].copy() + for event in curr_tile_ev_cp: + if event[0] == subject: + self.tiles[tile[1]][tile[0]]["events"].remove(event) diff --git a/tests/metagpt/environment/stanford_town_env/__init__.py b/tests/metagpt/environment/stanford_town_env/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/tests/metagpt/environment/stanford_town_env/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/tests/metagpt/environment/stanford_town_env/test_stanford_town_ext_env.py b/tests/metagpt/environment/stanford_town_env/test_stanford_town_ext_env.py new file mode 100644 index 000000000..3071f9deb --- /dev/null +++ b/tests/metagpt/environment/stanford_town_env/test_stanford_town_ext_env.py @@ -0,0 +1,40 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : the unittest of StanfordTownExtEnv + +from pathlib import Path + +from metagpt.environment.stanford_town_env.stanford_town_ext_env import ( + StanfordTownExtEnv, +) + +maze_asset_path = ( + Path(__file__).absolute().parent.joinpath("..", "..", "..", "data", "environment", "stanford_town", "the_ville") +) + + +def test_stanford_town_ext_env(): + ext_env = StanfordTownExtEnv(maze_asset_path=maze_asset_path) + + tile_coord = ext_env.turn_coordinate_to_tile((64, 64)) + assert tile_coord == (2, 2) + + tile = (58, 9) + assert len(ext_env.get_collision_maze()) == 100 + assert len(ext_env.get_address_tiles()) == 306 + assert ext_env.access_tile(tile=tile)["world"] == "the Ville" + assert ext_env.get_tile_path(tile=tile, level="world") == "the Ville" + assert len(ext_env.get_nearby_tiles(tile=tile, vision_r=5)) == 121 + + event = ("double studio:double studio:bedroom 2:bed", None, None, None) + ext_env.add_tiles_event(tile[1], tile[0], event=event) + ext_env.add_event_from_tile(event, tile) + assert len(ext_env.tiles[tile[1]][tile[0]]["events"]) == 1 + + ext_env.turn_event_from_tile_idle(event, tile) + + ext_env.remove_event_from_tile(event, tile) + assert len(ext_env.tiles[tile[1]][tile[0]]["events"]) == 0 + + ext_env.remove_subject_events_from_tile(subject=event[0], tile=tile) + assert len(ext_env.tiles[tile[1]][tile[0]]["events"]) == 0 From 37c3f3b85021d2940b7f97025e56d8ededac39d8 Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 19:14:24 +0800 Subject: [PATCH 508/637] add mincraft_env --- metagpt/environment/mincraft_env/__init__.py | 3 + .../environment/mincraft_env/mincraft_env.py | 390 +++++++++++++++ .../mincraft_env/mincraft_ext_env.py | 180 +++++++ .../mincraft_env/mineflayer/.gitignore | 1 + .../mincraft_env/mineflayer/.prettierignore | 3 + .../mincraft_env/mineflayer/.prettierrc.json | 3 + .../mincraft_env/mineflayer/index.js | 425 +++++++++++++++++ .../mineflayer/lib/observation/base.js | 45 ++ .../mineflayer/lib/observation/chests.js | 31 ++ .../mineflayer/lib/observation/inventory.js | 39 ++ .../mineflayer/lib/observation/onChat.js | 26 + .../mineflayer/lib/observation/onError.js | 22 + .../mineflayer/lib/observation/onSave.js | 22 + .../mineflayer/lib/observation/status.js | 103 ++++ .../mineflayer/lib/observation/voxels.js | 67 +++ .../mineflayer/lib/skillLoader.js | 79 +++ .../mincraft_env/mineflayer/lib/utils.js | 31 ++ .../mineflayer-collectblock/.gitignore | 107 +++++ .../mineflayer-collectblock/LICENSE | 21 + .../mineflayer-collectblock/README.md | 89 ++++ .../mineflayer-collectblock/_config.yml | 1 + .../mineflayer-collectblock/docs/api.md | 52 ++ .../examples/collector.js | 70 +++ .../examples/oreMiner.js | 59 +++ .../examples/storageBot.js | 107 +++++ .../mineflayer-collectblock/package.json | 44 ++ .../mineflayer-collectblock/src/BlockVeins.ts | 35 ++ .../src/CollectBlock.ts | 451 ++++++++++++++++++ .../mineflayer-collectblock/src/Inventory.ts | 87 ++++ .../mineflayer-collectblock/src/Targets.ts | 60 +++ .../mineflayer-collectblock/src/TaskQueue.ts | 77 +++ .../src/TemporarySubscriber.ts | 34 ++ .../mineflayer-collectblock/src/Util.ts | 13 + .../mineflayer-collectblock/src/index.ts | 25 + .../mineflayer-collectblock/tsconfig.json | 69 +++ .../mincraft_env/mineflayer/package.json | 38 ++ .../mincraft_env/process_monitor.py | 78 +++ .../environment/mincraft_env/__init__.py | 3 + .../mincraft_env/test_mincraft_ext_env.py | 14 + 39 files changed, 3004 insertions(+) create mode 100644 metagpt/environment/mincraft_env/__init__.py create mode 100644 metagpt/environment/mincraft_env/mincraft_env.py create mode 100644 metagpt/environment/mincraft_env/mincraft_ext_env.py create mode 100644 metagpt/environment/mincraft_env/mineflayer/.gitignore create mode 100644 metagpt/environment/mincraft_env/mineflayer/.prettierignore create mode 100644 metagpt/environment/mincraft_env/mineflayer/.prettierrc.json create mode 100644 metagpt/environment/mincraft_env/mineflayer/index.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/observation/base.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/observation/chests.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/observation/inventory.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/observation/onChat.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/observation/onError.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/observation/onSave.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/observation/status.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/observation/voxels.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/skillLoader.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/lib/utils.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/.gitignore create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/LICENSE create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/README.md create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/_config.yml create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/docs/api.md create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/collector.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/oreMiner.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/storageBot.js create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/package.json create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/BlockVeins.ts create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/CollectBlock.ts create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Inventory.ts create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Targets.ts create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/TaskQueue.ts create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/TemporarySubscriber.ts create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Util.ts create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/index.ts create mode 100644 metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/tsconfig.json create mode 100644 metagpt/environment/mincraft_env/mineflayer/package.json create mode 100644 metagpt/environment/mincraft_env/process_monitor.py create mode 100644 tests/metagpt/environment/mincraft_env/__init__.py create mode 100644 tests/metagpt/environment/mincraft_env/test_mincraft_ext_env.py diff --git a/metagpt/environment/mincraft_env/__init__.py b/metagpt/environment/mincraft_env/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/metagpt/environment/mincraft_env/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/metagpt/environment/mincraft_env/mincraft_env.py b/metagpt/environment/mincraft_env/mincraft_env.py new file mode 100644 index 000000000..bc093eb61 --- /dev/null +++ b/metagpt/environment/mincraft_env/mincraft_env.py @@ -0,0 +1,390 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : MG Mincraft Env + +import json +import re +import time +from typing import Any, Iterable + +from langchain.embeddings.openai import OpenAIEmbeddings +from langchain.vectorstores import Chroma +from pydantic import ConfigDict, Field + +from metagpt.config2 import config as CONFIG +from metagpt.const import MC_CKPT_DIR +from metagpt.environment.base_env import Environment +from metagpt.environment.mincraft_env.mincraft_ext_env import MincraftExtEnv +from metagpt.logs import logger +from metagpt.utils.common import load_mc_skills_code, read_json_file, write_json_file + + +class MincraftEnv(Environment, MincraftExtEnv): + """MincraftEnv, including shared memory of cache and infomation between roles""" + + model_config = ConfigDict(arbitrary_types_allowed=True) + + event: dict[str, Any] = Field(default_factory=dict) + current_task: str = Field(default="Mine 1 wood log") + task_execution_time: float = Field(default=float) + context: str = Field(default="You can mine one of oak, birch, spruce, jungle, acacia, dark oak, or mangrove logs.") + code: str = Field(default="") + program_code: str = Field(default="") # write in skill/code/*.js + program_name: str = Field(default="") + critique: str = Field(default="") + skills: dict = Field(default_factory=dict) # for skills.json + retrieve_skills: list[str] = Field(default_factory=list) + event_summary: str = Field(default="") + + qa_cache: dict[str, str] = Field(default_factory=dict) + completed_tasks: list[str] = Field(default_factory=list) # Critique things + failed_tasks: list[str] = Field(default_factory=list) + + skill_desp: str = Field(default="") + + chest_memory: dict[str, Any] = Field(default_factory=dict) # eg: {'(1344, 64, 1381)': 'Unknown'} + chest_observation: str = Field(default="") # eg: "Chests: None\n\n" + + runtime_status: bool = False # equal to action execution status: success or failed + + vectordb: Chroma = Field(default_factory=Chroma) + + qa_cache_questions_vectordb: Chroma = Field(default_factory=Chroma) + + @property + def progress(self): + # return len(self.completed_tasks) + 10 # Test only + return len(self.completed_tasks) + + @property + def programs(self): + programs = "" + if self.code == "": + return programs # TODO: maybe fix 10054 now, a better way is isolating env.step() like voyager + for skill_name, entry in self.skills.items(): + programs += f"{entry['code']}\n\n" + for primitives in load_mc_skills_code(): # TODO add skills_dir + programs += f"{primitives}\n\n" + return programs + + def set_mc_port(self, mc_port): + super().set_mc_port(mc_port) + self.set_mc_resume() + + def set_mc_resume(self): + self.qa_cache_questions_vectordb = Chroma( + collection_name="qa_cache_questions_vectordb", + embedding_function=OpenAIEmbeddings(), + persist_directory=f"{MC_CKPT_DIR}/curriculum/vectordb", + ) + + self.vectordb = Chroma( + collection_name="skill_vectordb", + embedding_function=OpenAIEmbeddings(), + persist_directory=f"{MC_CKPT_DIR}/skill/vectordb", + ) + + if CONFIG.resume: + logger.info(f"Loading Action Developer from {MC_CKPT_DIR}/action") + self.chest_memory = read_json_file(f"{MC_CKPT_DIR}/action/chest_memory.json") + + logger.info(f"Loading Curriculum Agent from {MC_CKPT_DIR}/curriculum") + self.completed_tasks = read_json_file(f"{MC_CKPT_DIR}/curriculum/completed_tasks.json") + self.failed_tasks = read_json_file(f"{MC_CKPT_DIR}/curriculum/failed_tasks.json") + + logger.info(f"Loading Skill Manager from {MC_CKPT_DIR}/skill\033[0m") + self.skills = read_json_file(f"{MC_CKPT_DIR}/skill/skills.json") + + logger.info(f"Loading Qa Cache from {MC_CKPT_DIR}/curriculum\033[0m") + self.qa_cache = read_json_file(f"{MC_CKPT_DIR}/curriculum/qa_cache.json") + + if self.vectordb._collection.count() == 0: + logger.info(self.vectordb._collection.count()) + # Set vdvs for skills & qa_cache + skill_desps = [skill["description"] for program_name, skill in self.skills.items()] + program_names = [program_name for program_name, skill in self.skills.items()] + metadatas = [{"name": program_name} for program_name in program_names] + # add vectordb from file + self.vectordb.add_texts( + texts=skill_desps, + ids=program_names, + metadatas=metadatas, + ) + self.vectordb.persist() + + logger.info(self.qa_cache_questions_vectordb._collection.count()) + if self.qa_cache_questions_vectordb._collection.count() == 0: + questions = [question for question, answer in self.qa_cache.items()] + + self.qa_cache_questions_vectordb.add_texts(texts=questions) + + self.qa_cache_questions_vectordb.persist() + + logger.info( + f"INIT_CHECK: There are {self.vectordb._collection.count()} skills in vectordb and {len(self.skills)} skills in skills.json." + ) + # Check if Skill Manager's vectordb right using + assert self.vectordb._collection.count() == len(self.skills), ( + f"Skill Manager's vectordb is not synced with skills.json.\n" + f"There are {self.vectordb._collection.count()} skills in vectordb but {len(self.skills)} skills in skills.json.\n" + f"Did you set resume=False when initializing the manager?\n" + f"You may need to manually delete the vectordb directory for running from scratch." + ) + + logger.info( + f"INIT_CHECK: There are {self.qa_cache_questions_vectordb._collection.count()} qa_cache in vectordb and {len(self.qa_cache)} questions in qa_cache.json." + ) + assert self.qa_cache_questions_vectordb._collection.count() == len(self.qa_cache), ( + f"Curriculum Agent's qa cache question vectordb is not synced with qa_cache.json.\n" + f"There are {self.qa_cache_questions_vectordb._collection.count()} questions in vectordb " + f"but {len(self.qa_cache)} questions in qa_cache.json.\n" + f"Did you set resume=False when initializing the agent?\n" + f"You may need to manually delete the qa cache question vectordb directory for running from scratch.\n" + ) + + def register_roles(self, roles: Iterable["Minecraft"]): + for role in roles: + role.set_memory(self) + + def update_event(self, event: dict): + if self.event == event: + return + self.event = event + self.update_chest_memory(event) + self.update_chest_observation() + # self.event_summary = self.summarize_chatlog(event) + + def update_task(self, task: str): + self.current_task = task + + def update_context(self, context: str): + self.context = context + + def update_program_code(self, program_code: str): + self.program_code = program_code + + def update_code(self, code: str): + self.code = code # action_developer.gen_action_code to HERE + + def update_program_name(self, program_name: str): + self.program_name = program_name + + def update_critique(self, critique: str): + self.critique = critique # critic_agent.check_task_success to HERE + + def append_skill(self, skill: dict): + self.skills[self.program_name] = skill # skill_manager.retrieve_skills to HERE + + def update_retrieve_skills(self, retrieve_skills: list): + self.retrieve_skills = retrieve_skills + + def update_skill_desp(self, skill_desp: str): + self.skill_desp = skill_desp + + async def update_qa_cache(self, qa_cache: dict): + self.qa_cache = qa_cache + + def update_chest_memory(self, events: dict): + """ + Input: events: Dict + Result: self.chest_memory update & save to json + """ + nearbyChests = events[-1][1]["nearbyChests"] + for position, chest in nearbyChests.items(): + if position in self.chest_memory: + if isinstance(chest, dict): + self.chest_memory[position] = chest + if chest == "Invalid": + logger.info(f"Action Developer removing chest {position}: {chest}") + self.chest_memory.pop(position) + else: + if chest != "Invalid": + logger.info(f"Action Developer saving chest {position}: {chest}") + self.chest_memory[position] = chest + + write_json_file(f"{MC_CKPT_DIR}/action/chest_memory.json", self.chest_memory) + + def update_chest_observation(self): + """ + update chest_memory to chest_observation. + Refer to @ https://github.com/MineDojo/Voyager/blob/main/voyager/agents/action.py + """ + + chests = [] + for chest_position, chest in self.chest_memory.items(): + if isinstance(chest, dict) and len(chest) > 0: + chests.append(f"{chest_position}: {chest}") + for chest_position, chest in self.chest_memory.items(): + if isinstance(chest, dict) and len(chest) == 0: + chests.append(f"{chest_position}: Empty") + for chest_position, chest in self.chest_memory.items(): + if isinstance(chest, str): + assert chest == "Unknown" + chests.append(f"{chest_position}: Unknown items inside") + assert len(chests) == len(self.chest_memory) + if chests: + chests = "\n".join(chests) + self.chest_observation = f"Chests:\n{chests}\n\n" + else: + self.chest_observation = "Chests: None\n\n" + + def summarize_chatlog(self, events): + def filter_item(message: str): + craft_pattern = r"I cannot make \w+ because I need: (.*)" + craft_pattern2 = r"I cannot make \w+ because there is no crafting table nearby" + mine_pattern = r"I need at least a (.*) to mine \w+!" + if re.match(craft_pattern, message): + self.event_summary = re.match(craft_pattern, message).groups()[0] + elif re.match(craft_pattern2, message): + self.event_summary = "a nearby crafting table" + elif re.match(mine_pattern, message): + self.event_summary = re.match(mine_pattern, message).groups()[0] + else: + self.event_summary = "" + return self.event_summary + + chatlog = set() + for event_type, event in events: + if event_type == "onChat": + item = filter_item(event["onChat"]) + if item: + chatlog.add(item) + self.event_summary = "I also need " + ", ".join(chatlog) + "." if chatlog else "" + + def reset_block_info(self): + # revert all the placing event in the last step + pass + + def update_exploration_progress(self, success: bool): + """ + Split task into completed_tasks or failed_tasks + Args: info = { + "task": self.task, + "success": success, + "conversations": self.conversations, + } + """ + self.runtime_status = success + task = self.current_task + if task.startswith("Deposit useless items into the chest at"): + return + if success: + logger.info(f"Completed task {task}.") + self.completed_tasks.append(task) + else: + logger.info(f"Failed to complete task {task}. Skipping to next task.") + self.failed_tasks.append(task) + # when not success, below to update event! + # revert all the placing event in the last step + blocks = [] + positions = [] + for event_type, event in self.event: + if event_type == "onSave" and event["onSave"].endswith("_placed"): + block = event["onSave"].split("_placed")[0] + position = event["status"]["position"] + blocks.append(block) + positions.append(position) + new_events = self.step( + f"await givePlacedItemBack(bot, {json.dumps(blocks)}, {json.dumps(positions)})", + programs=self.programs, + ) + self.event[-1][1]["inventory"] = new_events[-1][1]["inventory"] + self.event[-1][1]["voxels"] = new_events[-1][1]["voxels"] + + self.save_sorted_tasks() + + def save_sorted_tasks(self): + updated_completed_tasks = [] + # record repeated failed tasks + updated_failed_tasks = self.failed_tasks + # dedup but keep order + for task in self.completed_tasks: + if task not in updated_completed_tasks: + updated_completed_tasks.append(task) + + # remove completed tasks from failed tasks + for task in updated_completed_tasks: + while task in updated_failed_tasks: + updated_failed_tasks.remove(task) + + self.completed_tasks = updated_completed_tasks + self.failed_tasks = updated_failed_tasks + + # dump to json + write_json_file(f"{MC_CKPT_DIR}/curriculum/completed_tasks.json", self.completed_tasks) + write_json_file(f"{MC_CKPT_DIR}/curriculum/failed_tasks.json", self.failed_tasks) + + async def on_event_retrieve(self, *args): + """ + Retrieve Minecraft events. + + Returns: + list: A list of Minecraft events. + + Raises: + Exception: If there is an issue retrieving events. + """ + try: + self.reset( + options={ + "mode": "soft", + "wait_ticks": 20, + } + ) + # difficulty = "easy" if len(self.completed_tasks) > 15 else "peaceful" + difficulty = "peaceful" + + events = self.step("bot.chat(`/time set ${getNextTime()}`);\n" + f"bot.chat('/difficulty {difficulty}');") + self.update_event(events) + return events + except Exception as e: + time.sleep(3) # wait for mineflayer to exit + # reset bot status here + events = self.reset( + options={ + "mode": "hard", + "wait_ticks": 20, + "inventory": self.event[-1][1]["inventory"], + "equipment": self.event[-1][1]["status"]["equipment"], + "position": self.event[-1][1]["status"]["position"], + } + ) + self.update_event(events) + logger.error(f"Failed to retrieve Minecraft events: {str(e)}") + return events + + async def on_event_execute(self, *args): + """ + Execute Minecraft events. + + This function is used to obtain events from the Minecraft environment. Check the implementation in + the 'voyager/env/bridge.py step()' function to capture events generated within the game. + + Returns: + list: A list of Minecraft events. + + Raises: + Exception: If there is an issue retrieving events. + """ + try: + events = self.step( + code=self.code, + programs=self.programs, + ) + self.update_event(events) + return events + except Exception as e: + time.sleep(3) # wait for mineflayer to exit + # reset bot status here + events = self.reset( + options={ + "mode": "hard", + "wait_ticks": 20, + "inventory": self.event[-1][1]["inventory"], + "equipment": self.event[-1][1]["status"]["equipment"], + "position": self.event[-1][1]["status"]["position"], + } + ) + self.update_event(events) + logger.error(f"Failed to execute Minecraft events: {str(e)}") + return events diff --git a/metagpt/environment/mincraft_env/mincraft_ext_env.py b/metagpt/environment/mincraft_env/mincraft_ext_env.py new file mode 100644 index 000000000..4934e34b2 --- /dev/null +++ b/metagpt/environment/mincraft_env/mincraft_ext_env.py @@ -0,0 +1,180 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : The Mincraft external environment to integrate with Mincraft game +# refs to `voyager bridge.py` + +import json +import time +from typing import Optional + +import requests +from pydantic import ConfigDict, Field, model_validator + +from metagpt.const import ( + MC_CKPT_DIR, + MC_CORE_INVENTORY_ITEMS, + MC_CURRICULUM_OB, + MC_DEFAULT_WARMUP, + METAGPT_ROOT, +) +from metagpt.environment.base_env import ExtEnv, mark_as_writeable +from metagpt.environment.mincraft_env.process_monitor import SubprocessMonitor +from metagpt.logs import logger + + +class MincraftExtEnv(ExtEnv): + model_config = ConfigDict(arbitrary_types_allowed=True) + + mc_port: Optional[int] = Field(default=None) + server_host: str = Field(default="http://127.0.0.1") + server_port: str = Field(default=3000) + request_timeout: int = Field(default=600) + + mineflayer: Optional[SubprocessMonitor] = Field(default=None, validate_default=True) + + has_reset: bool = Field(default=False) + reset_options: Optional[dict] = Field(default=None) + connected: bool = Field(default=False) + server_paused: bool = Field(default=False) + warm_up: dict = Field(default=dict()) + + @property + def server(self) -> str: + return f"{self.server_host}:{self.server_port}" + + @model_validator(mode="after") + def _post_init_ext_env(self): + if not self.mineflayer: + self.mineflayer = SubprocessMonitor( + commands=[ + "node", + METAGPT_ROOT.joinpath("metagpt", "environment", "mincraft_env", "mineflayer", "index.js"), + str(self.server_port), + ], + name="mineflayer", + ready_match=r"Server started on port (\d+)", + ) + if not self.warm_up: + warm_up = MC_DEFAULT_WARMUP + if "optional_inventory_items" in warm_up: + assert MC_CORE_INVENTORY_ITEMS is not None + # self.core_inv_items_regex = re.compile(MC_CORE_INVENTORY_ITEMS) + self.warm_up["optional_inventory_items"] = warm_up["optional_inventory_items"] + else: + self.warm_up["optional_inventory_items"] = 0 + for key in MC_CURRICULUM_OB: + self.warm_up[key] = warm_up.get(key, MC_DEFAULT_WARMUP[key]) + self.warm_up["nearby_blocks"] = 0 + self.warm_up["inventory"] = 0 + self.warm_up["completed_tasks"] = 0 + self.warm_up["failed_tasks"] = 0 + + # init ckpt sub-forders + MC_CKPT_DIR.joinpath("curriculum/vectordb").mkdir(parents=True, exist_ok=True) + MC_CKPT_DIR.joinpath("action").mkdir(exist_ok=True) + MC_CKPT_DIR.joinpath("skill/code").mkdir(parents=True, exist_ok=True) + MC_CKPT_DIR.joinpath("skill/description").mkdir(exist_ok=True) + MC_CKPT_DIR.joinpath("skill/vectordb").mkdir(exist_ok=True) + + def set_mc_port(self, mc_port: int): + self.mc_port = mc_port + + @mark_as_writeable + def close(self) -> bool: + self.unpause() + if self.connected: + res = requests.post(f"{self.server}/stop") + if res.status_code == 200: + self.connected = False + self.mineflayer.stop() + return not self.connected + + @mark_as_writeable + def check_process(self) -> dict: + retry = 0 + while not self.mineflayer.is_running: + logger.info("Mineflayer process has exited, restarting") + self.mineflayer.run() + if not self.mineflayer.is_running: + if retry > 3: + logger.error("Mineflayer process failed to start") + raise {} + else: + retry += 1 + continue + logger.info(self.mineflayer.ready_line) + res = requests.post( + f"{self.server}/start", + json=self.reset_options, + timeout=self.request_timeout, + ) + if res.status_code != 200: + self.mineflayer.stop() + logger.error(f"Minecraft server reply with code {res.status_code}") + raise {} + return res.json() + + @mark_as_writeable + def reset(self, *, seed=None, options=None) -> dict: + if options is None: + options = {} + if options.get("inventory", {}) and options.get("mode", "hard") != "hard": + logger.error("inventory can only be set when options is hard") + raise {} + + self.reset_options = { + "port": self.mc_port, + "reset": options.get("mode", "hard"), + "inventory": options.get("inventory", {}), + "equipment": options.get("equipment", []), + "spread": options.get("spread", False), + "waitTicks": options.get("wait_ticks", 5), + "position": options.get("position", None), + } + + self.unpause() + self.mineflayer.stop() + time.sleep(1) # wait for mineflayer to exit + + returned_data = self.check_process() + self.has_reset = True + self.connected = True + # All the reset in step will be soft + self.reset_options["reset"] = "soft" + self.pause() + return json.loads(returned_data) + + @mark_as_writeable + def step(self, code: str, programs: str = "") -> dict: + if not self.has_reset: + raise RuntimeError("Environment has not been reset yet") + self.check_process() + self.unpause() + data = { + "code": code, + "programs": programs, + } + res = requests.post(f"{self.server}/step", json=data, timeout=self.request_timeout) + if res.status_code != 200: + raise RuntimeError("Failed to step Minecraft server") + returned_data = res.json() + self.pause() + return json.loads(returned_data) + + @mark_as_writeable + def pause(self) -> bool: + if self.mineflayer.is_running and not self.server_paused: + res = requests.post(f"{self.server}/pause") + if res.status_code == 200: + self.server_paused = True + return self.server_paused + + @mark_as_writeable + def unpause(self) -> bool: + if self.mineflayer.is_running and self.server_paused: + res = requests.post(f"{self.server}/pause") + if res.status_code == 200: + self.server_paused = False + else: + logger.info(f"mineflayer pause result: {res.json()}") + return self.server_paused diff --git a/metagpt/environment/mincraft_env/mineflayer/.gitignore b/metagpt/environment/mincraft_env/mineflayer/.gitignore new file mode 100644 index 000000000..0fd468410 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/.gitignore @@ -0,0 +1 @@ +!/lib \ No newline at end of file diff --git a/metagpt/environment/mincraft_env/mineflayer/.prettierignore b/metagpt/environment/mincraft_env/mineflayer/.prettierignore new file mode 100644 index 000000000..1b07c39e9 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/.prettierignore @@ -0,0 +1,3 @@ +# Ignore artifacts: +build +coverage \ No newline at end of file diff --git a/metagpt/environment/mincraft_env/mineflayer/.prettierrc.json b/metagpt/environment/mincraft_env/mineflayer/.prettierrc.json new file mode 100644 index 000000000..0a02bcefd --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/.prettierrc.json @@ -0,0 +1,3 @@ +{ + "tabWidth": 4 +} diff --git a/metagpt/environment/mincraft_env/mineflayer/index.js b/metagpt/environment/mincraft_env/mineflayer/index.js new file mode 100644 index 000000000..7fb0a8787 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/index.js @@ -0,0 +1,425 @@ +const fs = require("fs"); +const express = require("express"); +const bodyParser = require("body-parser"); +const mineflayer = require("mineflayer"); + +const skills = require("./lib/skillLoader"); +const { initCounter, getNextTime } = require("./lib/utils"); +const obs = require("./lib/observation/base"); +const OnChat = require("./lib/observation/onChat"); +const OnError = require("./lib/observation/onError"); +const { Voxels, BlockRecords } = require("./lib/observation/voxels"); +const Status = require("./lib/observation/status"); +const Inventory = require("./lib/observation/inventory"); +const OnSave = require("./lib/observation/onSave"); +const Chests = require("./lib/observation/chests"); +const { plugin: tool } = require("mineflayer-tool"); + +let bot = null; + +const app = express(); + +app.use(bodyParser.json({ limit: "50mb" })); +app.use(bodyParser.urlencoded({ limit: "50mb", extended: false })); + +app.post("/start", (req, res) => { + if (bot) onDisconnect("Restarting bot"); + bot = null; + console.log(req.body); + bot = mineflayer.createBot({ + host: "localhost", // minecraft server ip + port: req.body.port, // minecraft server port + username: "bot", + disableChatSigning: true, + checkTimeoutInterval: 60 * 60 * 1000, + }); + bot.once("error", onConnectionFailed); + + // Event subscriptions + bot.waitTicks = req.body.waitTicks; + bot.globalTickCounter = 0; + bot.stuckTickCounter = 0; + bot.stuckPosList = []; + bot.iron_pickaxe = false; + + bot.on("kicked", onDisconnect); + + // mounting will cause physicsTick to stop + bot.on("mount", () => { + bot.dismount(); + }); + + bot.once("spawn", async () => { + bot.removeListener("error", onConnectionFailed); + let itemTicks = 1; + if (req.body.reset === "hard") { + bot.chat("/clear @s"); + bot.chat("/kill @s"); + const inventory = req.body.inventory ? req.body.inventory : {}; + const equipment = req.body.equipment + ? req.body.equipment + : [null, null, null, null, null, null]; + for (let key in inventory) { + bot.chat(`/give @s minecraft:${key} ${inventory[key]}`); + itemTicks += 1; + } + const equipmentNames = [ + "armor.head", + "armor.chest", + "armor.legs", + "armor.feet", + "weapon.mainhand", + "weapon.offhand", + ]; + for (let i = 0; i < 6; i++) { + if (i === 4) continue; + if (equipment[i]) { + bot.chat( + `/item replace entity @s ${equipmentNames[i]} with minecraft:${equipment[i]}` + ); + itemTicks += 1; + } + } + } + + if (req.body.position) { + bot.chat( + `/tp @s ${req.body.position.x} ${req.body.position.y} ${req.body.position.z}` + ); + } + + // if iron_pickaxe is in bot's inventory + if ( + bot.inventory.items().find((item) => item.name === "iron_pickaxe") + ) { + bot.iron_pickaxe = true; + } + + const { pathfinder } = require("mineflayer-pathfinder"); + const tool = require("mineflayer-tool").plugin; + const collectBlock = require("mineflayer-collectblock").plugin; + const pvp = require("mineflayer-pvp").plugin; + const minecraftHawkEye = require("minecrafthawkeye"); + bot.loadPlugin(pathfinder); + bot.loadPlugin(tool); + bot.loadPlugin(collectBlock); + bot.loadPlugin(pvp); + bot.loadPlugin(minecraftHawkEye); + + // bot.collectBlock.movements.digCost = 0; + // bot.collectBlock.movements.placeCost = 0; + + obs.inject(bot, [ + OnChat, + OnError, + Voxels, + Status, + Inventory, + OnSave, + Chests, + BlockRecords, + ]); + skills.inject(bot); + + if (req.body.spread) { + bot.chat(`/spreadplayers ~ ~ 0 300 under 80 false @s`); + await bot.waitForTicks(bot.waitTicks); + } + + await bot.waitForTicks(bot.waitTicks * itemTicks); + res.json(bot.observe()); + + initCounter(bot); + bot.chat("/gamerule keepInventory true"); + bot.chat("/gamerule doDaylightCycle false"); + }); + + function onConnectionFailed(e) { + console.log(e); + bot = null; + res.status(400).json({ error: e }); + } + function onDisconnect(message) { + if (bot.viewer) { + bot.viewer.close(); + } + bot.end(); + console.log(message); + bot = null; + } +}); + +app.post("/step", async (req, res) => { + // import useful package + let response_sent = false; + function otherError(err) { + console.log("Uncaught Error"); + bot.emit("error", handleError(err)); + bot.waitForTicks(bot.waitTicks).then(() => { + if (!response_sent) { + response_sent = true; + res.json(bot.observe()); + } + }); + } + + process.on("uncaughtException", otherError); + + const mcData = require("minecraft-data")(bot.version); + mcData.itemsByName["leather_cap"] = mcData.itemsByName["leather_helmet"]; + mcData.itemsByName["leather_tunic"] = + mcData.itemsByName["leather_chestplate"]; + mcData.itemsByName["leather_pants"] = + mcData.itemsByName["leather_leggings"]; + mcData.itemsByName["leather_boots"] = mcData.itemsByName["leather_boots"]; + mcData.itemsByName["lapis_lazuli_ore"] = mcData.itemsByName["lapis_ore"]; + mcData.blocksByName["lapis_lazuli_ore"] = mcData.blocksByName["lapis_ore"]; + const { + Movements, + goals: { + Goal, + GoalBlock, + GoalNear, + GoalXZ, + GoalNearXZ, + GoalY, + GoalGetToBlock, + GoalLookAtBlock, + GoalBreakBlock, + GoalCompositeAny, + GoalCompositeAll, + GoalInvert, + GoalFollow, + GoalPlaceBlock, + }, + pathfinder, + Move, + ComputedPath, + PartiallyComputedPath, + XZCoordinates, + XYZCoordinates, + SafeBlock, + GoalPlaceBlockOptions, + } = require("mineflayer-pathfinder"); + const { Vec3 } = require("vec3"); + + // Set up pathfinder + const movements = new Movements(bot, mcData); + bot.pathfinder.setMovements(movements); + + bot.globalTickCounter = 0; + bot.stuckTickCounter = 0; + bot.stuckPosList = []; + + function onTick() { + bot.globalTickCounter++; + if (bot.pathfinder.isMoving()) { + bot.stuckTickCounter++; + if (bot.stuckTickCounter >= 100) { + onStuck(1.5); + bot.stuckTickCounter = 0; + } + } + } + + bot.on("physicTick", onTick); + + // initialize fail count + let _craftItemFailCount = 0; + let _killMobFailCount = 0; + let _mineBlockFailCount = 0; + let _placeItemFailCount = 0; + let _smeltItemFailCount = 0; + + // Retrieve array form post bod + const code = req.body.code; + const programs = req.body.programs; + bot.cumulativeObs = []; + await bot.waitForTicks(bot.waitTicks); + const r = await evaluateCode(code, programs); + process.off("uncaughtException", otherError); + if (r !== "success") { + bot.emit("error", handleError(r)); + } + await returnItems(); + // wait for last message + await bot.waitForTicks(bot.waitTicks); + if (!response_sent) { + response_sent = true; + res.json(bot.observe()); + } + bot.removeListener("physicTick", onTick); + + async function evaluateCode(code, programs) { + // Echo the code produced for players to see it. Don't echo when the bot code is already producing dialog or it will double echo + try { + await eval("(async () => {" + programs + "\n" + code + "})()"); + return "success"; + } catch (err) { + return err; + } + } + + function onStuck(posThreshold) { + const currentPos = bot.entity.position; + bot.stuckPosList.push(currentPos); + + // Check if the list is full + if (bot.stuckPosList.length === 5) { + const oldestPos = bot.stuckPosList[0]; + const posDifference = currentPos.distanceTo(oldestPos); + + if (posDifference < posThreshold) { + teleportBot(); // execute the function + } + + // Remove the oldest time from the list + bot.stuckPosList.shift(); + } + } + + function teleportBot() { + const blocks = bot.findBlocks({ + matching: (block) => { + return block.type === 0; + }, + maxDistance: 1, + count: 27, + }); + + if (blocks) { + // console.log(blocks.length); + const randomIndex = Math.floor(Math.random() * blocks.length); + const block = blocks[randomIndex]; + bot.chat(`/tp @s ${block.x} ${block.y} ${block.z}`); + } else { + bot.chat("/tp @s ~ ~1.25 ~"); + } + } + + function returnItems() { + bot.chat("/gamerule doTileDrops false"); + const crafting_table = bot.findBlock({ + matching: mcData.blocksByName.crafting_table.id, + maxDistance: 128, + }); + if (crafting_table) { + bot.chat( + `/setblock ${crafting_table.position.x} ${crafting_table.position.y} ${crafting_table.position.z} air destroy` + ); + bot.chat("/give @s crafting_table"); + } + const furnace = bot.findBlock({ + matching: mcData.blocksByName.furnace.id, + maxDistance: 128, + }); + if (furnace) { + bot.chat( + `/setblock ${furnace.position.x} ${furnace.position.y} ${furnace.position.z} air destroy` + ); + bot.chat("/give @s furnace"); + } + if (bot.inventoryUsed() >= 32) { + // if chest is not in bot's inventory + if (!bot.inventory.items().find((item) => item.name === "chest")) { + bot.chat("/give @s chest"); + } + } + // if iron_pickaxe not in bot's inventory and bot.iron_pickaxe + if ( + bot.iron_pickaxe && + !bot.inventory.items().find((item) => item.name === "iron_pickaxe") + ) { + bot.chat("/give @s iron_pickaxe"); + } + bot.chat("/gamerule doTileDrops true"); + } + + function handleError(err) { + let stack = err.stack; + if (!stack) { + return err; + } + console.log(stack); + const final_line = stack.split("\n")[1]; + const regex = /:(\d+):\d+\)/; + + const programs_length = programs.split("\n").length; + let match_line = null; + for (const line of stack.split("\n")) { + const match = regex.exec(line); + if (match) { + const line_num = parseInt(match[1]); + if (line_num >= programs_length) { + match_line = line_num - programs_length; + break; + } + } + } + if (!match_line) { + return err.message; + } + let f_line = final_line.match( + /\((?.*):(?\d+):(?\d+)\)/ + ); + if (f_line && f_line.groups && fs.existsSync(f_line.groups.file)) { + const { file, line, pos } = f_line.groups; + const f = fs.readFileSync(file, "utf8").split("\n"); + // let filename = file.match(/(?<=node_modules\\)(.*)/)[1]; + let source = file + `:${line}\n${f[line - 1].trim()}\n `; + + const code_source = + "at " + + code.split("\n")[match_line - 1].trim() + + " in your code"; + return source + err.message + "\n" + code_source; + } else if ( + f_line && + f_line.groups && + f_line.groups.file.includes("") + ) { + const { file, line, pos } = f_line.groups; + let source = + "Your code" + + `:${match_line}\n${code.split("\n")[match_line - 1].trim()}\n `; + let code_source = ""; + if (line < programs_length) { + source = + "In your program code: " + + programs.split("\n")[line - 1].trim() + + "\n"; + code_source = `at line ${match_line}:${code + .split("\n") + [match_line - 1].trim()} in your code`; + } + return source + err.message + "\n" + code_source; + } + return err.message; + } +}); + +app.post("/stop", (req, res) => { + bot.end(); + res.json({ + message: "Bot stopped", + }); +}); + +app.post("/pause", (req, res) => { + if (!bot) { + res.status(400).json({ error: "Bot not spawned" }); + return; + } + bot.chat("/pause"); + bot.waitForTicks(bot.waitTicks).then(() => { + res.json({ message: "Success" }); + }); +}); + +// Server listening to PORT 3000 + +const DEFAULT_PORT = 3000; +const PORT = process.argv[2] || DEFAULT_PORT; +app.listen(PORT, () => { + console.log(`Server started on port ${PORT}`); +}); diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/observation/base.js b/metagpt/environment/mincraft_env/mineflayer/lib/observation/base.js new file mode 100644 index 000000000..b661a24b5 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/observation/base.js @@ -0,0 +1,45 @@ +class Observation { + constructor(bot) { + if (new.target === Observation) { + throw new TypeError( + "Cannot instantiate abstract class Observation" + ); + } + + this.bot = bot; + this.name = "Observation"; + } + + observe() { + throw new TypeError("Method 'observe()' must be implemented."); + } + + reset() {} +} + +function inject(bot, obs_list) { + bot.obsList = []; + bot.cumulativeObs = []; + bot.eventMemory = {}; + obs_list.forEach((obs) => { + bot.obsList.push(new obs(bot)); + }); + bot.event = function (event_name) { + let result = {}; + bot.obsList.forEach((obs) => { + if (obs.name.startsWith("on") && obs.name !== event_name) { + return; + } + result[obs.name] = obs.observe(); + }); + bot.cumulativeObs.push([event_name, result]); + }; + bot.observe = function () { + bot.event("observe"); + const result = bot.cumulativeObs; + bot.cumulativeObs = []; + return JSON.stringify(result); + }; +} + +module.exports = { Observation, inject }; diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/observation/chests.js b/metagpt/environment/mincraft_env/mineflayer/lib/observation/chests.js new file mode 100644 index 000000000..842bd171d --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/observation/chests.js @@ -0,0 +1,31 @@ +const { Observation } = require("./base"); + +class Chests extends Observation { + constructor(bot) { + super(bot); + this.name = "nearbyChests"; + this.chestsItems = {}; + bot.on("closeChest", (chestItems, position) => { + this.chestsItems[position] = chestItems; + }); + bot.on("removeChest", (chestPosition) => { + this.chestsItems[chestPosition] = "Invalid"; + }); + } + + observe() { + const chests = this.bot.findBlocks({ + matching: this.bot.registry.blocksByName.chest.id, + maxDistance: 16, + count: 999, + }); + chests.forEach((chest) => { + if (!this.chestsItems.hasOwnProperty(chest)) { + this.chestsItems[chest] = "Unknown"; + } + }); + return this.chestsItems; + } +} + +module.exports = Chests; diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/observation/inventory.js b/metagpt/environment/mincraft_env/mineflayer/lib/observation/inventory.js new file mode 100644 index 000000000..0645d1bfa --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/observation/inventory.js @@ -0,0 +1,39 @@ +const { Observation } = require("./base"); + +class Inventory extends Observation { + constructor(bot) { + super(bot); + this.name = "inventory"; + } + + observe() { + return listItems(this.bot); + } +} + +function listItems(bot) { + const items = getInventoryItems(bot); + return items.reduce(itemToDict, {}); +} + +function getInventoryItems(bot) { + const inventory = bot.currentWindow || bot.inventory; + return inventory.items(); +} + +function itemToDict(acc, cur) { + if (cur.name && cur.count) { + //if both name and count property are defined + if (acc[cur.name]) { + //if the item is already in the dict + acc[cur.name] += cur.count; + } else { + //if the item is not in the dict + acc[cur.name] = cur.count; + } + } + return acc; +} + +//export modules +module.exports = Inventory; diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/observation/onChat.js b/metagpt/environment/mincraft_env/mineflayer/lib/observation/onChat.js new file mode 100644 index 000000000..54b411e2a --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/observation/onChat.js @@ -0,0 +1,26 @@ +const Observation = require("./base.js").Observation; + +class onChat extends Observation { + constructor(bot) { + super(bot); + this.name = "onChat"; + this.obs = ""; + bot.on("chatEvent", (username, message) => { + // Save entity status to local variable + if (message.startsWith("/")) { + return; + } + + this.obs += message; + this.bot.event(this.name); + }); + } + + observe() { + const result = this.obs; + this.obs = ""; + return result; + } +} + +module.exports = onChat; diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/observation/onError.js b/metagpt/environment/mincraft_env/mineflayer/lib/observation/onError.js new file mode 100644 index 000000000..ac8fed9e5 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/observation/onError.js @@ -0,0 +1,22 @@ +const Observation = require("./base.js").Observation; + +class onError extends Observation { + constructor(bot) { + super(bot); + this.name = "onError"; + this.obs = null; + bot.on("error", (err) => { + // Save entity status to local variable + this.obs = err; + this.bot.event(this.name); + }); + } + + observe() { + const result = this.obs; + this.obs = null; + return result; + } +} + +module.exports = onError; diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/observation/onSave.js b/metagpt/environment/mincraft_env/mineflayer/lib/observation/onSave.js new file mode 100644 index 000000000..e5983590f --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/observation/onSave.js @@ -0,0 +1,22 @@ +const Observation = require("./base.js").Observation; + +class onSave extends Observation { + constructor(bot) { + super(bot); + this.name = "onSave"; + this.obs = null; + bot.on("save", (eventName) => { + // Save entity status to local variable + this.obs = eventName; + this.bot.event(this.name); + }); + } + + observe() { + const result = this.obs; + this.obs = null; + return result; + } +} + +module.exports = onSave; diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/observation/status.js b/metagpt/environment/mincraft_env/mineflayer/lib/observation/status.js new file mode 100644 index 000000000..b031fbcf2 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/observation/status.js @@ -0,0 +1,103 @@ +const Observation = require("./base.js").Observation; + +class Status extends Observation { + constructor(bot) { + super(bot); + this.name = "status"; + } + + observe() { + return { + health: this.bot.health, + food: this.bot.food, + saturation: this.bot.foodSaturation, + oxygen: this.bot.oxygenLevel, + position: this.bot.entity.position, + velocity: this.bot.entity.velocity, + yaw: this.bot.entity.yaw, + pitch: this.bot.entity.pitch, + onGround: this.bot.entity.onGround, + equipment: this.getEquipment(), + name: this.bot.entity.username, + timeSinceOnGround: this.bot.entity.timeSinceOnGround, + isInWater: this.bot.entity.isInWater, + isInLava: this.bot.entity.isInLava, + isInWeb: this.bot.entity.isInWeb, + isCollidedHorizontally: this.bot.entity.isCollidedHorizontally, + isCollidedVertically: this.bot.entity.isCollidedVertically, + biome: this.bot.blockAt(this.bot.entity.position) + ? this.bot.blockAt(this.bot.entity.position).biome.name + : "None", + entities: this.getEntities(), + timeOfDay: this.getTime(), + inventoryUsed: this.bot.inventoryUsed(), + elapsedTime: this.bot.globalTickCounter, + }; + } + + itemToObs(item) { + if (!item) return null; + return item.name; + } + + getTime() { + const timeOfDay = this.bot.time.timeOfDay; + let time = ""; + if (timeOfDay < 1000) { + time = "sunrise"; + } else if (timeOfDay < 6000) { + time = "day"; + } else if (timeOfDay < 12000) { + time = "noon"; + } else if (timeOfDay < 13000) { + time = "sunset"; + } else if (timeOfDay < 18000) { + time = "night"; + } else if (timeOfDay < 22000) { + time = "midnight"; + } else { + time = "sunrise"; + } + return time; + } + + // For each item in equipment, if it exists, return the name of the item + // otherwise return null + getEquipment() { + const slots = this.bot.inventory.slots; + const mainHand = this.bot.heldItem; + return slots + .slice(5, 9) + .concat(mainHand, slots[45]) + .map(this.itemToObs); + } + + getEntities() { + const entities = this.bot.entities; + if (!entities) return {}; + // keep all monsters in one list, keep other mobs in another list + const mobs = {}; + for (const id in entities) { + const entity = entities[id]; + if (!entity.displayName) continue; + if (entity.name === "player" || entity.name === "item") continue; + if (entity.position.distanceTo(this.bot.entity.position) < 32) { + if (!mobs[entity.name]) { + mobs[entity.name] = entity.position.distanceTo( + this.bot.entity.position + ); + } else if ( + mobs[entity.name] > + entity.position.distanceTo(this.bot.entity.position) + ) { + mobs[entity.name] = entity.position.distanceTo( + this.bot.entity.position + ); + } + } + } + return mobs; + } +} + +module.exports = Status; diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/observation/voxels.js b/metagpt/environment/mincraft_env/mineflayer/lib/observation/voxels.js new file mode 100644 index 000000000..ecb0c14b7 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/observation/voxels.js @@ -0,0 +1,67 @@ +// Blocks = require("./blocks") +const { Observation } = require("./base"); + +class Voxels extends Observation { + constructor(bot) { + super(bot); + this.name = "voxels"; + } + + observe() { + return Array.from(getSurroundingBlocks(this.bot, 8, 2, 8)); + } +} + +class BlockRecords extends Observation { + constructor(bot) { + super(bot); + this.name = "blockRecords"; + this.records = new Set(); + this.tick = 0; + bot.on("physicsTick", () => { + this.tick++; + if (this.tick >= 100) { + const items = getInventoryItems(this.bot); + getSurroundingBlocks(this.bot, 8, 2, 8).forEach((block) => { + if (!items.has(block)) this.records.add(block); + }); + this.tick = 0; + } + }); + } + + observe() { + return Array.from(this.records); + } + + reset() { + this.records = new Set(); + } +} + +function getSurroundingBlocks(bot, x_distance, y_distance, z_distance) { + const surroundingBlocks = new Set(); + + for (let x = -x_distance; x <= x_distance; x++) { + for (let y = -y_distance; y <= y_distance; y++) { + for (let z = -z_distance; z <= z_distance; z++) { + const block = bot.blockAt(bot.entity.position.offset(x, y, z)); + if (block && block.type !== 0) { + surroundingBlocks.add(block.name); + } + } + } + } + // console.log(surroundingBlocks); + return surroundingBlocks; +} + +function getInventoryItems(bot) { + const items = new Set(); + bot.inventory.items().forEach((item) => { + if (item) items.add(item.name); + }); + return items; +} + +module.exports = { Voxels, BlockRecords }; diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/skillLoader.js b/metagpt/environment/mincraft_env/mineflayer/lib/skillLoader.js new file mode 100644 index 000000000..d78cf7820 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/skillLoader.js @@ -0,0 +1,79 @@ +function inject(bot) { + bot._sleep = bot.sleep; + bot.sleep = async (bedBlock) => { + await bot.waitForTicks(20); + await bot._sleep(bedBlock); + await bot.waitForTicks(135); + }; + + bot._fish = bot.fish; + bot.fish = async () => { + if (bot.heldItem?.name !== "fishing_rod") { + bot.chat("I'm not holding a fishing rod!"); + return; + } + let timeout = null; + await Promise.race([ + bot._fish(), + new Promise( + (resolve, reject) => + (timeout = setTimeout(() => { + bot.activateItem(); + reject( + new Error( + "Finishing timeout, make sure you get to and look at a water block!" + ) + ); + }, 60000)) + ), + ]); + clearTimeout(timeout); + await bot.waitForTicks(20); + }; + + bot._consume = bot.consume; + bot.consume = async () => { + // action_count.activateItem++; + await bot._consume(); + await bot.waitForTicks(20); + }; + + bot._useOn = bot.useOn; + bot.useOn = async (entity) => { + if (entity.position.distanceTo(bot.entity.position) > 6) { + bot.chat("Please goto a place near the entity first!"); + return; + } + await bot._useOn(entity); + await bot.waitForTicks(20); + }; + + bot._activateBlock = bot.activateBlock; + bot.activateBlock = async (block) => { + if (block.position.distanceTo(bot.entity.position) > 6) { + bot.chat("Please goto a place near the block first!"); + return; + } + // action_count.activateBlock++; + await bot._activateBlock(block); + }; + + bot._chat = bot.chat; + bot.chat = (message) => { + // action_count.chat++; + bot.emit("chatEvent", "bot", message); + bot._chat(message); + }; + + bot.inventoryUsed = () => { + return bot.inventory.slots.slice(9, 45).filter((item) => item !== null) + .length; + }; + + bot.save = function (eventName) { + bot.emit("save", eventName); + }; +} + +// export all control_primitives +module.exports = { inject }; diff --git a/metagpt/environment/mincraft_env/mineflayer/lib/utils.js b/metagpt/environment/mincraft_env/mineflayer/lib/utils.js new file mode 100644 index 000000000..68af30796 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/lib/utils.js @@ -0,0 +1,31 @@ +let gameTimeCounter = 0; +let gameTimeList = []; +const initCounter = (bot) => { + gameTimeList = []; + for (let i = 0; i < 13000; i += 1000) { + gameTimeList.push(i); + } + for (let i = 13000; i < 24000; i += 2000) { + gameTimeList.push(i); + } + const timeOfDay = bot.time.timeOfDay; + for (let i = 0; i < gameTimeList.length; i++) { + if (gameTimeList[i] > timeOfDay) { + gameTimeCounter = i - 1; + break; + } + } +}; + +const getNextTime = () => { + gameTimeCounter++; + if (gameTimeCounter >= gameTimeList.length) { + gameTimeCounter = 0; + } + return gameTimeList[gameTimeCounter]; +}; + +module.exports = { + initCounter, + getNextTime, +}; diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/.gitignore b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/.gitignore new file mode 100644 index 000000000..0578fdca3 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/.gitignore @@ -0,0 +1,107 @@ +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +lerna-debug.log* + +# Diagnostic reports (https://nodejs.org/api/report.html) +report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json + +# Runtime data +pids +*.pid +*.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover +lib-cov + +# Coverage directory used by tools like istanbul +coverage +*.lcov + +# nyc test coverage +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) +.grunt + +# Bower dependency directory (https://bower.io/) +bower_components + +# node-waf configuration +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) +build/Release + +# Dependency directories +node_modules/ +jspm_packages/ + +# TypeScript v1 declaration files +typings/ + +# TypeScript cache +*.tsbuildinfo + +# Optional npm cache directory +.npm + +# Optional eslint cache +.eslintcache + +# Microbundle cache +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history +.node_repl_history + +# Output of 'npm pack' +*.tgz + +# Yarn Integrity file +.yarn-integrity + +# dotenv environment variables file +.env +.env.test + +# parcel-bundler cache (https://parceljs.org/) +.cache + +# Next.js build output +.next + +# Nuxt.js build / generate output +.nuxt +dist + +# Gatsby files +.cache/ +# Comment in the public line in if your project uses Gatsby and *not* Next.js +# https://nextjs.org/blog/next-9-1#public-directory-support +# public + +# vuepress build output +.vuepress/dist + +# Serverless directories +.serverless/ + +# FuseBox cache +.fusebox/ + +# DynamoDB Local files +.dynamodb/ + +# TernJS port file +.tern-port + +lib/ +package-lock.json diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/LICENSE b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/LICENSE new file mode 100644 index 000000000..f2896b56e --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2020 TheDudeFromCI + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/README.md b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/README.md new file mode 100644 index 000000000..555acb761 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/README.md @@ -0,0 +1,89 @@ +

mineflayer-collectblock

+

A small utility plugin for allowing users to collect blocks using a higher level API.

+ +

+ + + + + + +

+ +--- +## This is a modified version to better support Voyager + +## Showcase + +You can see a video of the plugin in action, [here.](https://youtu.be/5T_rcCnNnf4) +The source code of the bot in the video can be seen in the examples folder, [here.](https://github.com/TheDudeFromCI/mineflayer-collectblock/blob/master/examples/collector.js) + +### Description + +This plugin is a wrapper for mineflayer that allows for easier API usage when collecting blocks or item drops. This plugin is designed to reduce some of the boilerplate code based around the act of pathfinding to a block _(handled by_ ***mineflayer-pathfinder***_)_, selecting the best tool to mine that block _(handled by_ ***mineflayer-tool***_)_, actually mining it, then moving to collect the item drops from that block. This plugin allows for all of that basic concept to be wrapped up into a single API function. + +In addition to the usage above, some additional quality of life features are available in this plugin. These include the ability to automatically deposit items into a chest when the bot's inventory is full, collecting new tools from a chest if the bot doesn't currently have a required tool _(also handled by_ ***mineflayer-tool***_)_, and allowing for queueing of multiple blocks or item drops to the collection task, so they can be processed later. + +### Getting Started + +This plugin is built using Node and can be installed using: +```bash +npm install --save mineflayer-collectblock +``` + +### Simple Bot + +The brief description goes here. + +```js +// Create your bot +const mineflayer = require("mineflayer") +const bot = mineflayer.createBot({ + host: 'localhost', + username: 'Player', +}) +let mcData + +// Load collect block +bot.loadPlugin(require('mineflayer-collectblock').plugin) + +async function collectGrass() { + // Find a nearby grass block + const grass = bot.findBlock({ + matching: mcData.blocksByName.grass_block.id, + maxDistance: 64 + }) + + if (grass) { + // If we found one, collect it. + try { + await bot.collectBlock.collect(grass) + collectGrass() // Collect another grass block + } catch (err) { + console.log(err) // Handle errors, if any + } + } +} + +// On spawn, start collecting all nearby grass +bot.once('spawn', () => { + mcData = require('minecraft-data')(bot.version) + collectGrass() +}) +``` + +### Documentation + +[API](https://github.com/TheDudeFromCI/mineflayer-collectblock/blob/master/docs/api.md) + +[Examples](https://github.com/TheDudeFromCI/mineflayer-collectblock/tree/master/examples) + +### License + +This project uses the [MIT](https://github.com/TheDudeFromCI/mineflayer-collectblock/blob/master/LICENSE) license. + +### Contributions + +This project is accepting PRs and Issues. See something you think can be improved? Go for it! Any and all help is highly appreciated! + +For larger changes, it is recommended to discuss these changes in the issues tab before writing any code. It's also preferred to make many smaller PRs than one large one, where applicable. diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/_config.yml b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/_config.yml new file mode 100644 index 000000000..c4192631f --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/_config.yml @@ -0,0 +1 @@ +theme: jekyll-theme-cayman \ No newline at end of file diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/docs/api.md b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/docs/api.md new file mode 100644 index 000000000..66d8a3ecc --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/docs/api.md @@ -0,0 +1,52 @@ +# API + +Welcome to the *mineflayer-collectblock* API documentation page. + +## Table of Contents + +- [1. Summary](#1-summary) +- [Properties](#properties) + - [`bot.collectblock.movements: Movements`](#botcollectblockmovements-movements) +- [Functions](#functions) + - [collect](#collect) + - [Options:](#options) + +## 1. Summary + +The collect block plugin is a utility plugin that can be used to help make collecting blocks and item drops very easy, using only a single API call. No need to worry about pathfinding to the block, selecting the right tool, or moving to pick up the item drop after mining. + +## Properties + +### `bot.collectblock.movements: Movements` + +The movements object used by the pathfinder plugin to define the movement configuration. This object is passed to the pathfinder plugin when any API from this plugin is called in order to control how pathfinding should work when collecting the given blocks or item. + +If set to null, the pathfinder plugin movements is not updated. + +Defaults to a new movements object instance. + +## Functions + +### collect + +Usage: `bot.collectblock.collect(target: Collectable | Collectable[], options?: CollectOptions, cb: (err?: Error) => void): void` + +Causes the bot to collect the given block, item drop, or list of those. If the target is a block, the bot will move to the block, mine it, and pick up the item drop. If the target is an item drop, the bot will move to the item drop and pick it up. If the target is a list of collectables, the bot will move from target to target in order of closest to furthest and collect each target in turn. + +#### Options: + + * `append: boolean` + + If true, the target(s) will be appended to the existing target list instead of starting a new task. Defaults to false. + + * `ignoreNoPath: boolean` + + If true, errors will not be thrown when a path to the target block cannot be found. The bot will attempt to choose the best available position it can find, instead. Errors are still thrown if the bot cannot interact with the block from it's final location. Defaults to false. + + * `chestLocations: Vec3[]` + + Gets the list of chest locations to use when storing items after the bot's inventory becomes full. If undefined, it defaults to the chest location list on the bot.collectBlock plugin. + + * `itemFilter: ItemFilter` + + When transferring items to a chest, this filter is used to determine what items are allowed to be moved, and what items aren't allowed to be moved. Defaults to the item filter specified on the bot.collectBlock plugin. \ No newline at end of file diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/collector.js b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/collector.js new file mode 100644 index 000000000..b9bb8faf9 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/collector.js @@ -0,0 +1,70 @@ +/** + * This bot example show how to direct a bot to collect a specific block type + * or a group of nearby blocks of that type. + */ + +const mineflayer = require('mineflayer') +const collectBlock = require('mineflayer-collectblock').plugin + +if (process.argv.length < 4 || process.argv.length > 6) { + console.log('Usage : node collector.js [] []') + process.exit(1) +} + +const bot = mineflayer.createBot({ + host: process.argv[2], + port: process.argv[3], + username: process.argv[4] || 'collector', + password: process.argv[5] +}) + +bot.loadPlugin(collectBlock) + +let mcData +bot.once('spawn', () => { + mcData = require('minecraft-data')(bot.version) +}) + +bot.on('chat', async (username, message) => { + const args = message.split(' ') + if (args[0] !== 'collect') return + + let count = 1 + if (args.length === 3) count = parseInt(args[1]) + + let type = args[1] + if (args.length === 3) type = args[2] + + const blockType = mcData.blocksByName[type] + if (!blockType) { + return + } + + const blocks = bot.findBlocks({ + matching: blockType.id, + maxDistance: 64, + count: count + }) + + if (blocks.length === 0) { + bot.chat("I don't see that block nearby.") + return + } + + const targets = [] + for (let i = 0; i < Math.min(blocks.length, count); i++) { + targets.push(bot.blockAt(blocks[i])) + } + + bot.chat(`Found ${targets.length} ${type}(s)`) + + try { + await bot.collectBlock.collect(targets) + // All blocks have been collected. + bot.chat('Done') + } catch (err) { + // An error occurred, report it. + bot.chat(err.message) + console.log(err) + } +}) diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/oreMiner.js b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/oreMiner.js new file mode 100644 index 000000000..6accac88f --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/oreMiner.js @@ -0,0 +1,59 @@ +/** + * This bot example shows how to collect a vein of ores quickly after only finding a single block. + * This makes it easy to collect a vein of ores or mine a tree without looking for every block in the + * area. + */ + +const mineflayer = require('mineflayer') +const collectBlock = require('mineflayer-collectblock').plugin + +if (process.argv.length < 4 || process.argv.length > 6) { + console.log('Usage : node oreMiner.js [] []') + process.exit(1) +} + +const bot = mineflayer.createBot({ + host: process.argv[2], + port: process.argv[3], + username: process.argv[4] || 'oreMiner', + password: process.argv[5] +}) + +bot.loadPlugin(collectBlock) + +let mcData +bot.once('spawn', () => { + mcData = require('minecraft-data')(bot.version) +}) + +bot.on('chat', async (username, message) => { + const args = message.split(' ') + if (args[0] !== 'collect') return + + const blockType = mcData.blocksByName[args[1]] + if (!blockType) { + bot.chat(`I don't know any blocks named ${args[1]}.`) + return + } + + const block = bot.findBlock({ + matching: blockType.id, + maxDistance: 64 + }) + + if (!block) { + bot.chat("I don't see that block nearby.") + return + } + + const targets = bot.collectBlock.findFromVein(block) + try { + await bot.collectBlock.collect(targets) + // All blocks have been collected. + bot.chat('Done') + } catch (err) { + // An error occurred, report it. + bot.chat(err.message) + console.log(err) + } +}) diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/storageBot.js b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/storageBot.js new file mode 100644 index 000000000..b6f9971f2 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/examples/storageBot.js @@ -0,0 +1,107 @@ +/** + * This bot example shows how to use the chest filling mechanic of the plugin. + * Simply provide a given storage chest, and the bot will automatically try and + * store it's inventory in that chest when the bot's inventory becomes full. + */ + +if (process.argv.length < 4 || process.argv.length > 6) { + console.log('Usage : node storageBot.js [] []') + process.exit(1) +} + +// Load your libraries +const mineflayer = require('mineflayer') +const collectBlock = require('mineflayer-collectblock').plugin + +// Create your bot +const bot = mineflayer.createBot({ + host: process.argv[2], + port: parseInt(process.argv[3]), + username: process.argv[4] ? process.argv[4] : 'storageBot', + password: process.argv[5] +}) + +// Load the collect block plugin +bot.loadPlugin(collectBlock) + +// Load mcData on login +let mcData +bot.once('login', () => { + mcData = require('minecraft-data')(bot.version) +}) + +// On spawn, try to find any nearby chests and save those as storage locations. +// When the bot's inventory becomes too full, it will empty it's inventory into +// these chests before collecting more resources. If a chest gets full, it moves +// to the next one in order until it's inventory is empty or it runs out of chests. +bot.once('spawn', () => { + bot.collectBlock.chestLocations = bot.findBlocks({ + matching: mcData.blocksByName.chest.id, + maxDistance: 16, + count: 999999 // Get as many chests as we can + }) + + if (bot.collectBlock.chestLocations.length === 0) { + bot.chat("I don't see any chests nearby.") + } else { + for (const chestPos of bot.collectBlock.chestLocations) { + bot.chat(`I found a chest at ${chestPos}`) + } + } +}) + +// Wait for someone to say something +bot.on('chat', async (username, message) => { + // If the player says something start starts with "collect" + // Otherwise, do nothing + const args = message.split(' ') + if (args[0] !== 'collect') return + + // If the player specifies a number, collect that many. Otherwise, default to 1. + let count = 1 + if (args.length === 3) count = parseInt(args[1]) + + // If a number was given the item number is the 3rd arg, not the 2nd. + let type = args[1] + if (args.length === 3) type = args[2] + + // Get the id of that block type for this version of Minecraft. + const blockType = mcData.blocksByName[type] + if (!blockType) { + bot.chat(`I don't know any blocks named ${type}.`) + return + } + + // Find all nearby blocks of that type, up to the given count, within 64 blocks. + const blocks = bot.findBlocks({ + matching: blockType.id, + maxDistance: 64, + count: count + }) + + // Complain if we can't find any nearby blocks of that type. + if (blocks.length === 0) { + bot.chat("I don't see that block nearby.") + return + } + + // Convert the block position array into a block array to pass to collect block. + const targets = [] + for (let i = 0; i < Math.min(blocks.length, count); i++) { + targets.push(bot.blockAt(blocks[i])) + } + + // Announce what we found. + bot.chat(`Found ${targets.length} ${type}(s)`) + + // Tell the bot to collect all of the given blocks in the block list. + try { + await bot.collectBlock.collect(targets) + // All blocks have been collected. + bot.chat('Done') + } catch (err) { + // An error occurred, report it. + bot.chat(err.message) + console.log(err) + } +}) diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/package.json b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/package.json new file mode 100644 index 000000000..0f59e7aa6 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/package.json @@ -0,0 +1,44 @@ +{ + "name": "mineflayer-collectblock", + "version": "1.4.1", + "description": "A simple utility plugin for Mineflayer that add a higher level API for collecting blocks.", + "main": "lib/index.js", + "types": "lib/index.d.ts", + "scripts": { + "build": "ts-standard && tsc && require-self", + "clean": "rm -rf lib", + "test": "test" + }, + "repository": { + "type": "git", + "url": "git+https://github.com/TheDudeFromCI/mineflayer-collectblock.git" + }, + "keywords": [ + "mineflayer", + "plugin", + "api", + "utility", + "helper", + "collect" + ], + "author": "TheDudeFromCI", + "license": "MIT", + "bugs": { + "url": "https://github.com/TheDudeFromCI/mineflayer-collectblock/issues" + }, + "homepage": "https://github.com/TheDudeFromCI/mineflayer-collectblock#readme", + "dependencies": { + "mineflayer": "^4.0.0", + "mineflayer-pathfinder": "^2.1.1", + "mineflayer-tool": "^1.1.0" + }, + "devDependencies": { + "@types/node": "^18.6.4", + "require-self": "^0.2.3", + "ts-standard": "^11.0.0", + "typescript": "^4.1.3" + }, + "files": [ + "lib/**/*" + ] +} diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/BlockVeins.ts b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/BlockVeins.ts new file mode 100644 index 000000000..ae5542ce3 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/BlockVeins.ts @@ -0,0 +1,35 @@ +import { Bot } from 'mineflayer' +import { Block } from 'prismarine-block' + +export function findFromVein (bot: Bot, block: Block, maxBlocks: number, maxDistance: number, floodRadius: number): Block[] { + const targets: Block[] = [] + const open: Block[] = [block] + const type = block.type + const center = block.position + + for (let i = 0; i < maxBlocks; i++) { + const next = open.pop() + if (next == null) break + + targets.push(next) + + for (let x = -floodRadius; x <= floodRadius; x++) { + for (let y = -floodRadius; y <= floodRadius; y++) { + for (let z = -floodRadius; z <= floodRadius; z++) { + const neighborPos = next.position.offset(x, y, z) + if (neighborPos.manhattanDistanceTo(center) > maxDistance) continue + + const neighbor = bot.blockAt(neighborPos) + if (neighbor == null || neighbor.type !== type) continue + + if (targets.includes(neighbor)) continue + if (open.includes(neighbor)) continue + + open.push(neighbor) + } + } + } + } + + return targets +} diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/CollectBlock.ts b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/CollectBlock.ts new file mode 100644 index 000000000..d2be87822 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/CollectBlock.ts @@ -0,0 +1,451 @@ +import { Bot } from "mineflayer"; +import { Block } from "prismarine-block"; +import { Movements, goals } from "mineflayer-pathfinder"; +import { TemporarySubscriber } from "./TemporarySubscriber"; +import { Entity } from "prismarine-entity"; +import { error } from "./Util"; +import { Vec3 } from "vec3"; +import { emptyInventoryIfFull, ItemFilter } from "./Inventory"; +import { findFromVein } from "./BlockVeins"; +import { Collectable, Targets } from "./Targets"; +import { Item } from "prismarine-item"; +import mcDataLoader from "minecraft-data"; +import { once } from "events"; +import { callbackify } from "util"; + +export type Callback = (err?: Error) => void; + +async function collectAll( + bot: Bot, + options: CollectOptionsFull +): Promise { + let success_count = 0; + while (!options.targets.empty) { + await emptyInventoryIfFull( + bot, + options.chestLocations, + options.itemFilter + ); + const closest = options.targets.getClosest(); + if (closest == null) break; + switch (closest.constructor.name) { + case "Block": { + try { + if (success_count >= options.count) { + break; + } + await bot.tool.equipForBlock( + closest as Block, + equipToolOptions + ); + const goal = new goals.GoalLookAtBlock( + closest.position, + bot.world + ); + await bot.pathfinder.goto(goal); + await mineBlock(bot, closest as Block, options); + success_count++; + // TODO: options.ignoreNoPath + } catch (err) { + // @ts-ignore + // console.log(err.stack) + // bot.pathfinder.stop() + // bot.waitForTicks(10) + try { + bot.pathfinder.setGoal(null); + } catch (err) {} + if (options.ignoreNoPath) { + // @ts-ignore + if (err.name === "Invalid block") { + console.log( + `Block ${closest.name} at ${closest.position} is not valid! Skip it!` + ); + } // @ts-ignore + else if (err.name === "Unsafe block") { + console.log( + `${closest.name} at ${closest.position} is not safe to break! Skip it!` + ); + // @ts-ignore + } else if (err.name === "NoItem") { + const properties = + bot.registry.blocksByName[closest.name]; + const leastTool = Object.keys( + properties.harvestTools + )[0]; + const item = bot.registry.items[leastTool]; + bot.chat( + `I need at least a ${item.name} to mine ${closest.name}! Skip it!` + ); + return; + } else if ( + // @ts-ignore + err.name === "NoPath" || + // @ts-ignore + err.name === "Timeout" + ) { + if ( + bot.entity.position.distanceTo( + closest.position + ) < 0.5 + ) { + await mineBlock(bot, closest as Block, options); + break; + } + console.log( + `No path to ${closest.name} at ${closest.position}! Skip it!` + ); + // @ts-ignore + } else if (err.message === "Digging aborted") { + console.log(`Digging aborted! Skip it!`); + } else { + // @ts-ignore + bot.chat(`Error: ${err.message}`); + } + break; + } + throw err; + } + break; + } + case "Entity": { + // Don't collect any entities that are marked as 'invalid' + if (!(closest as Entity).isValid) break; + try { + const tempEvents = new TemporarySubscriber(bot); + const waitForPickup = new Promise( + (resolve, reject) => { + const timeout = setTimeout(() => { + // After 10 seconds, reject the promise + clearTimeout(timeout); + tempEvents.cleanup(); + reject(new Error("Failed to pickup item")); + }, 10000); + tempEvents.subscribeTo( + "entityGone", + (entity: Entity) => { + if (entity === closest) { + clearTimeout(timeout); + tempEvents.cleanup(); + resolve(); + } + } + ); + } + ); + bot.pathfinder.setGoal( + new goals.GoalFollow(closest as Entity, 0) + ); + // await bot.pathfinder.goto(new goals.GoalBlock(closest.position.x, closest.position.y, closest.position.z)) + await waitForPickup; + } catch (err) { + // @ts-ignore + console.log(err.stack); + try { + bot.pathfinder.setGoal(null); + } catch (err) {} + if (options.ignoreNoPath) { + // @ts-ignore + if (err.message === "Failed to pickup item") { + bot.chat(`Failed to pickup item! Skip it!`); + } + break; + } + throw err; + } + break; + } + default: { + throw error( + "UnknownType", + `Target ${closest.constructor.name} is not a Block or Entity!` + ); + } + } + options.targets.removeTarget(closest); + } + bot.chat(`Collect finish!`); +} + +const equipToolOptions = { + requireHarvest: true, + getFromChest: false, + maxTools: 2, +}; + +async function mineBlock( + bot: Bot, + block: Block, + options: CollectOptionsFull +): Promise { + if ( + bot.blockAt(block.position)?.type !== block.type || + bot.blockAt(block.position)?.type === 0 + ) { + options.targets.removeTarget(block); + throw error("Invalid block", "Block is not valid!"); + // @ts-expect-error + } else if (!bot.pathfinder.movements.safeToBreak(block)) { + options.targets.removeTarget(block); + throw error("Unsafe block", "Block is not safe to break!"); + } + + await bot.tool.equipForBlock(block, equipToolOptions); + + if (!block.canHarvest(bot.heldItem ? bot.heldItem.type : bot.heldItem)) { + options.targets.removeTarget(block); + throw error("NoItem", "Bot does not have a harvestable tool!"); + } + + const tempEvents = new TemporarySubscriber(bot); + tempEvents.subscribeTo("itemDrop", (entity: Entity) => { + if ( + entity.position.distanceTo(block.position.offset(0.5, 0.5, 0.5)) <= + 0.5 + ) { + options.targets.appendTarget(entity); + } + }); + try { + await bot.dig(block); + // Waiting for items to drop + await new Promise((resolve) => { + let remainingTicks = 10; + tempEvents.subscribeTo("physicTick", () => { + remainingTicks--; + if (remainingTicks <= 0) { + tempEvents.cleanup(); + resolve(); + } + }); + }); + } finally { + tempEvents.cleanup(); + } +} + +/** + * A set of options to apply when collecting the given targets. + */ +export interface CollectOptions { + /** + * If true, the target(s) will be appended to the existing target list instead of + * starting a new task. Defaults to false. + */ + append?: boolean; + + /** + * If true, errors will not be thrown when a path to the target block cannot + * be found. The bot will attempt to choose the best available position it + * can find, instead. Errors are still thrown if the bot cannot interact with + * the block from it's final location. Defaults to false. + */ + ignoreNoPath?: boolean; + + /** + * Gets the list of chest locations to use when storing items after the bot's + * inventory becomes full. If undefined, it defaults to the chest location + * list on the bot.collectBlock plugin. + */ + chestLocations?: Vec3[]; + + /** + * When transferring items to a chest, this filter is used to determine what + * items are allowed to be moved, and what items aren't allowed to be moved. + * Defaults to the item filter specified on the bot.collectBlock plugin. + */ + itemFilter?: ItemFilter; + + /** + * The total number of items to collect + */ + count?: number; +} + +/** + * A version of collect options where all values are assigned. + */ +interface CollectOptionsFull { + append: boolean; + ignoreNoPath: boolean; + chestLocations: Vec3[]; + itemFilter: ItemFilter; + targets: Targets; + count: number; +} + +/** + * The collect block plugin. + */ +export class CollectBlock { + /** + * The bot. + */ + private readonly bot: Bot; + + /** + * The list of active targets being collected. + */ + private readonly targets: Targets; + + /** + * The movements configuration to be sent to the pathfinder plugin. + */ + movements?: Movements; + + /** + * A list of chest locations which the bot is allowed to empty their inventory into + * if it becomes full while the bot is collecting resources. + */ + chestLocations: Vec3[] = []; + + /** + * When collecting items, this filter is used to determine what items should be placed + * into a chest if the bot's inventory becomes full. By default, returns true for all + * items except for tools, weapons, and armor. + * + * @param item - The item stack in the bot's inventory to check. + * + * @returns True if the item should be moved into the chest. False otherwise. + */ + itemFilter: ItemFilter = (item: Item) => { + if (item.name.includes("helmet")) return false; + if (item.name.includes("chestplate")) return false; + if (item.name.includes("leggings")) return false; + if (item.name.includes("boots")) return false; + if (item.name.includes("shield")) return false; + if (item.name.includes("sword")) return false; + if (item.name.includes("pickaxe")) return false; + if (item.name.includes("axe")) return false; + if (item.name.includes("shovel")) return false; + if (item.name.includes("hoe")) return false; + return true; + }; + + /** + * Creates a new instance of the create block plugin. + * + * @param bot - The bot this plugin is acting on. + */ + constructor(bot: Bot) { + this.bot = bot; + this.targets = new Targets(bot); + // @ts-ignore + this.movements = new Movements(bot, mcDataLoader(bot.version)); + } + + /** + * If target is a block: + * Causes the bot to break and collect the target block. + * + * If target is an item drop: + * Causes the bot to collect the item drop. + * + * If target is an array containing items or blocks, preforms the correct action for + * all targets in that array sorting dynamically by distance. + * + * @param target - The block(s) or item(s) to collect. + * @param options - The set of options to use when handling these targets + * @param cb - The callback that is called finished. + */ + async collect( + target: Collectable | Collectable[], + options: CollectOptions | Callback = {}, + cb?: Callback + ): Promise { + if (typeof options === "function") { + cb = options; + options = {}; + } + // @ts-expect-error + if (cb != null) return callbackify(this.collect)(target, options, cb); + + const optionsFull: CollectOptionsFull = { + append: options.append ?? false, + ignoreNoPath: options.ignoreNoPath ?? false, + chestLocations: options.chestLocations ?? this.chestLocations, + itemFilter: options.itemFilter ?? this.itemFilter, + targets: this.targets, + count: options.count ?? Infinity, + }; + + if (this.bot.pathfinder == null) { + throw error( + "UnresolvedDependency", + "The mineflayer-collectblock plugin relies on the mineflayer-pathfinder plugin to run!" + ); + } + + if (this.bot.tool == null) { + throw error( + "UnresolvedDependency", + "The mineflayer-collectblock plugin relies on the mineflayer-tool plugin to run!" + ); + } + + if (this.movements != null) { + this.bot.pathfinder.setMovements(this.movements); + } + + if (!optionsFull.append) await this.cancelTask(); + if (Array.isArray(target)) { + this.targets.appendTargets(target); + } else { + this.targets.appendTarget(target); + } + + try { + await collectAll(this.bot, optionsFull); + this.targets.clear(); + } catch (err) { + this.targets.clear(); + // Ignore path stopped error for cancelTask to work properly (imo we shouldn't throw any pathing errors) + // @ts-expect-error + if (err.name !== "PathStopped") throw err; + } finally { + // @ts-expect-error + this.bot.emit("collectBlock_finished"); + } + } + + /** + * Loads all touching blocks of the same type to the given block and returns them as an array. + * This effectively acts as a flood fill algorithm to retrieve blocks in the same ore vein and similar. + * + * @param block - The starting block. + * @param maxBlocks - The maximum number of blocks to look for before stopping. + * @param maxDistance - The max distance from the starting block to look. + * @param floodRadius - The max distance distance from block A to block B to be considered "touching" + */ + findFromVein( + block: Block, + maxBlocks = 100, + maxDistance = 16, + floodRadius = 1 + ): Block[] { + return findFromVein( + this.bot, + block, + maxBlocks, + maxDistance, + floodRadius + ); + } + + /** + * Cancels the current collection task, if still active. + * + * @param cb - The callback to use when the task is stopped. + */ + async cancelTask(cb?: Callback): Promise { + if (this.targets.empty) { + if (cb != null) cb(); + return await Promise.resolve(); + } + this.bot.pathfinder.stop(); + if (cb != null) { + // @ts-expect-error + this.bot.once("collectBlock_finished", cb); + } + await once(this.bot, "collectBlock_finished"); + } +} diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Inventory.ts b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Inventory.ts new file mode 100644 index 000000000..6a17d0cc5 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Inventory.ts @@ -0,0 +1,87 @@ +import { Bot } from 'mineflayer' +import { Callback } from './CollectBlock' +import { Vec3 } from 'vec3' +import { error } from './Util' +import { Item } from 'prismarine-item' +import { goals } from 'mineflayer-pathfinder' +import { callbackify } from 'util' + +export type ItemFilter = (item: Item) => boolean + +function getClosestChest (bot: Bot, chestLocations: Vec3[]): Vec3 | null { + let chest = null + let distance = 0 + + for (const c of chestLocations) { + const dist = c.distanceTo(bot.entity.position) + if (chest == null || dist < distance) { + chest = c + distance = dist + } + } + + if (chest != null) { + chestLocations.splice(chestLocations.indexOf(chest), 1) + } + + return chest +} + +export async function emptyInventoryIfFull (bot: Bot, chestLocations: Vec3[], itemFilter: ItemFilter, cb?: Callback): Promise { + // @ts-expect-error + if (cb != null) return callbackify(emptyInventoryIfFull)(bot, chestLocations, cb) + if (bot.inventory.emptySlotCount() > 0) return + return await emptyInventory(bot, chestLocations, itemFilter) +} + +export async function emptyInventory (bot: Bot, chestLocations: Vec3[], itemFilter: ItemFilter, cb?: Callback): Promise { + // @ts-expect-error + if (cb != null) return callbackify(emptyInventory)(bot, chestLocations, cb) + if (chestLocations.length === 0) { + throw error('NoChests', 'There are no defined chest locations!') + } + + // Shallow clone so we can safely remove chests from the list that are full. + chestLocations = [...chestLocations] + + while (true) { + const chest = getClosestChest(bot, chestLocations) + if (chest == null) { + throw error('NoChests', 'All chests are full.') + } + const hasRemaining = await tryEmptyInventory(bot, chest, itemFilter) + if (!hasRemaining) return + } +} + +async function tryEmptyInventory (bot: Bot, chestLocation: Vec3, itemFilter: ItemFilter, cb?: (err: Error | undefined, hasRemaining: boolean) => void): Promise { + // @ts-expect-error + if (cb != null) return callbackify(tryEmptyInventory)(bot, chestLocation, itemFilter, cb) + await gotoChest(bot, chestLocation) + return await placeItems(bot, chestLocation, itemFilter) +} + +async function gotoChest (bot: Bot, location: Vec3, cb?: Callback): Promise { + // @ts-expect-error + if (cb != null) return callbackify(gotoChest)(bot, location) + await bot.pathfinder.goto(new goals.GoalGetToBlock(location.x, location.y, location.z)) +} + +async function placeItems (bot: Bot, chestPos: Vec3, itemFilter: ItemFilter, cb?: (err: Error | undefined, hasRemaining: boolean) => void): Promise { + // @ts-expect-error + if (cb != null) return callbackify(placeItems)(bot, chestPos, itemFilter, cb) + const chestBlock = bot.blockAt(chestPos) + if (chestBlock == null) { + throw error('UnloadedChunk', 'Chest is in an unloaded chunk!') + } + const chest = await bot.openChest(chestBlock) + for (const item of bot.inventory.items()) { + if (!itemFilter(item)) continue + if (chest.firstEmptyContainerSlot() === null) { + // We have items that didn't fit. + return true + } + await chest.deposit(item.type, item.metadata, item.count) + } + return false +} diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Targets.ts b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Targets.ts new file mode 100644 index 000000000..568d07ad9 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Targets.ts @@ -0,0 +1,60 @@ +import { Bot } from 'mineflayer' +import { Block } from 'prismarine-block' +import { Entity } from 'prismarine-entity' + +export type Collectable = Block | Entity + +export class Targets { + private readonly bot: Bot + private targets: Collectable[] = [] + + constructor (bot: Bot) { + this.bot = bot + } + + appendTargets (targets: Collectable[]): void { + for (const target of targets) { + this.appendTarget(target) + } + } + + appendTarget (target: Collectable): void { + if (this.targets.includes(target)) return + this.targets.push(target) + } + + /** + * Gets the closest target to the bot in this list. + * + * @returns The closest target, or null if there are no targets. + */ + getClosest (): Collectable | null { + let closest: Collectable | null = null + let distance: number = 0 + + for (const target of this.targets) { + const dist = target.position.distanceTo(this.bot.entity.position) + + if (closest == null || dist < distance) { + closest = target + distance = dist + } + } + + return closest + } + + get empty (): boolean { + return this.targets.length === 0 + } + + clear (): void { + this.targets.length = 0 + } + + removeTarget (target: Collectable): void { + const index = this.targets.indexOf(target) + if (index < 0) return + this.targets.splice(index, 1) + } +} diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/TaskQueue.ts b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/TaskQueue.ts new file mode 100644 index 000000000..81fe3bc5a --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/TaskQueue.ts @@ -0,0 +1,77 @@ +import type { Callback } from './index' +export type Task = (cb: Callback) => void +export type SyncTask = () => void + +/** + * A simple utility class for queuing up a series of async tasks to execute. + */ +export class TaskQueue { + private tasks: Task[] = [] + + /** + * If true, the task list will stop executing if one of the tasks throws an error. + */ + readonly stopOnError: boolean = true + + /** + * Adds a new async task to this queue. The provided callback should be executed when + * the async task is complete. + * + * @param task - The async task to add. + */ + add (task: Task): void { + this.tasks.push(task) + } + + /** + * Adds a synchronous task toi this queue. + * + * @param task - The sync task to add. + */ + addSync (task: SyncTask): void { + this.add((cb) => { + try { + task() + cb() + } catch (err: any) { + cb(err) + } + }) + } + + /** + * Runs all tasks currently in this queue and empties the queue. + * + * @param cb - The optional callback to be executed when all tasks in this queue have + * finished executing. + */ + runAll (cb?: Callback): void { + const taskList = this.tasks + this.tasks = [] + + let index = -1 + const runNext: () => void = () => { + index++ + if (index >= taskList.length) { + if (cb !== undefined) cb() + return + } + + try { + taskList[index]((err) => { + if (err !== undefined) { + if (cb !== undefined) cb(err) + + if (this.stopOnError) return + } + + runNext() + }) + } catch (err: any) { + if (cb !== undefined) cb(err) + } + } + + runNext() + } +} diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/TemporarySubscriber.ts b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/TemporarySubscriber.ts new file mode 100644 index 000000000..3f14a607d --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/TemporarySubscriber.ts @@ -0,0 +1,34 @@ +import { Bot } from 'mineflayer' + +class Subscription { + constructor (readonly eventName: string, readonly callback: Function) {} +} + +export class TemporarySubscriber { + private readonly subscriptions: Subscription[] = [] + + constructor (readonly bot: Bot) {} + + /** + * Adds a new temporary event listener to the bot. + * + * @param event - The event to subscribe to. + * @param callback - The function to execute. + */ + subscribeTo (event: string, callback: Function): void { + this.subscriptions.push(new Subscription(event, callback)) + + // @ts-expect-error + this.bot.on(event, callback) + } + + /** + * Removes all attached event listeners from the bot. + */ + cleanup (): void { + for (const sub of this.subscriptions) { + // @ts-expect-error + this.bot.removeListener(sub.eventName, sub.callback) + } + } +} diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Util.ts b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Util.ts new file mode 100644 index 000000000..ee0f29e0c --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/Util.ts @@ -0,0 +1,13 @@ +/** + * Creates a new error object with the given type and message. + * + * @param type - The error type. + * @param message - The error message. + * + * @returns The error object. + */ +export function error (type: string, message: string): Error { + const e = new Error(message) + e.name = type + return e +} diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/index.ts b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/index.ts new file mode 100644 index 000000000..45c9a8508 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/src/index.ts @@ -0,0 +1,25 @@ +import { Bot } from 'mineflayer' +import { CollectBlock } from './CollectBlock' +import { pathfinder as pathfinderPlugin } from 'mineflayer-pathfinder' +import { plugin as toolPlugin } from 'mineflayer-tool' + +export function plugin (bot: Bot): void { + // @ts-expect-error + bot.collectBlock = new CollectBlock(bot) + + // Load plugins if not loaded manually. + setTimeout(() => loadPathfinderPlugin(bot), 0) + setTimeout(() => loadToolPlugin(bot), 0) +} + +function loadPathfinderPlugin (bot: Bot): void { + if (bot.pathfinder != null) return + bot.loadPlugin(pathfinderPlugin) +} + +function loadToolPlugin (bot: Bot): void { + if (bot.tool != null) return + bot.loadPlugin(toolPlugin) +} + +export { CollectBlock, Callback, CollectOptions } from './CollectBlock' diff --git a/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/tsconfig.json b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/tsconfig.json new file mode 100644 index 000000000..a6076bc0c --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/mineflayer-collectblock/tsconfig.json @@ -0,0 +1,69 @@ +{ + "compilerOptions": { + /* Visit https://aka.ms/tsconfig.json to read more about this file */ + /* Basic Options */ + // "incremental": true, /* Enable incremental compilation */ + "target": "ES2015", /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', 'ES2018', 'ES2019', 'ES2020', or 'ESNEXT'. */ + "module": "commonjs", /* Specify module code generation: 'none', 'commonjs', 'amd', 'system', 'umd', 'es2015', 'es2020', or 'ESNext'. */ + // "lib": [], /* Specify library files to be included in the compilation. */ + "allowJs": true, /* Allow javascript files to be compiled. */ + "checkJs": true, /* Report errors in .js files. */ + // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ + "declaration": true, + // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ + // "sourceMap": true, /* Generates corresponding '.map' file. */ + // "outFile": "./", /* Concatenate and emit output to single file. */ + "outDir": "./lib", + // "rootDir": "./", /* Specify the root directory of input files. Use to control the output directory structure with --outDir. */ + // "composite": true, /* Enable project compilation */ + // "tsBuildInfoFile": "./", /* Specify file to store incremental compilation information */ + // "removeComments": true, /* Do not emit comments to output. */ + // "noEmit": true, /* Do not emit outputs. */ + // "importHelpers": true, /* Import emit helpers from 'tslib'. */ + // "downlevelIteration": true, /* Provide full support for iterables in 'for-of', spread, and destructuring when targeting 'ES5' or 'ES3'. */ + // "isolatedModules": true, /* Transpile each file as a separate module (similar to 'ts.transpileModule'). */ + /* Strict Type-Checking Options */ + "strict": true, /* Enable all strict type-checking options. */ + // "noImplicitAny": true, /* Raise error on expressions and declarations with an implied 'any' type. */ + "strictNullChecks": true, /* Enable strict null checks. */ + // "strictFunctionTypes": true, /* Enable strict checking of function types. */ + // "strictBindCallApply": true, /* Enable strict 'bind', 'call', and 'apply' methods on functions. */ + // "strictPropertyInitialization": true, /* Enable strict checking of property initialization in classes. */ + // "noImplicitThis": true, /* Raise error on 'this' expressions with an implied 'any' type. */ + "alwaysStrict": true, /* Parse in strict mode and emit "use strict" for each source file. */ + /* Additional Checks */ + "noUnusedLocals": true, /* Report errors on unused locals. */ + // "noUnusedParameters": true, /* Report errors on unused parameters. */ + "noImplicitReturns": true, /* Report error when not all code paths in function return a value. */ + // "noFallthroughCasesInSwitch": true, /* Report errors for fallthrough cases in switch statement. */ + /* Module Resolution Options */ + // "moduleResolution": "node", /* Specify module resolution strategy: 'node' (Node.js) or 'classic' (TypeScript pre-1.6). */ + // "baseUrl": "./", /* Base directory to resolve non-absolute module names. */ + // "paths": {}, /* A series of entries which re-map imports to lookup locations relative to the 'baseUrl'. */ + // "rootDirs": [], /* List of root folders whose combined content represents the structure of the project at runtime. */ + // "typeRoots": [], /* List of folders to include type definitions from. */ + // "types": [], /* Type declaration files to be included in compilation. */ + // "allowSyntheticDefaultImports": true, /* Allow default imports from modules with no default export. This does not affect code emit, just typechecking. */ + "esModuleInterop": true, /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */ + // "preserveSymlinks": true, /* Do not resolve the real path of symlinks. */ + // "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */ + /* Source Map Options */ + // "sourceRoot": "", /* Specify the location where debugger should locate TypeScript files instead of source locations. */ + // "mapRoot": "", /* Specify the location where debugger should locate map files instead of generated locations. */ + // "inlineSourceMap": true, /* Emit a single file with source maps instead of having a separate file. */ + // "inlineSources": true, /* Emit the source alongside the sourcemaps within a single file; requires '--inlineSourceMap' or '--sourceMap' to be set. */ + /* Experimental Options */ + // "experimentalDecorators": true, /* Enables experimental support for ES7 decorators. */ + // "emitDecoratorMetadata": true, /* Enables experimental support for emitting type metadata for decorators. */ + /* Advanced Options */ + "skipLibCheck": true, /* Skip type checking of declaration files. */ + "forceConsistentCasingInFileNames": true /* Disallow inconsistently-cased references to the same file. */ + }, + "include": [ + "src" + ], + "exclude": [ + "node_modules", + "**/__tests__/*" + ] +} \ No newline at end of file diff --git a/metagpt/environment/mincraft_env/mineflayer/package.json b/metagpt/environment/mincraft_env/mineflayer/package.json new file mode 100644 index 000000000..9e389d268 --- /dev/null +++ b/metagpt/environment/mincraft_env/mineflayer/package.json @@ -0,0 +1,38 @@ +{ + "name": "voyager", + "version": "1.0.0", + "description": "", + "main": "index.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "keywords": [], + "author": "", + "license": "ISC", + "dependencies": { + "body-parser": "^1.20.2", + "express": "^4.18.2", + "magic-string": "^0.30.0", + "minecraft-data": "^3.31.0", + "minecrafthawkeye": "^1.3.6", + "mineflayer": "^4.8.1", + "mineflayer-collectblock": "file:mineflayer-collectblock", + "mineflayer-pathfinder": "^2.4.2", + "mineflayer-pvp": "^1.3.2", + "mineflayer-tool": "^1.2.0", + "mocha": "^10.2.0", + "prismarine-biome": "^1.3.0", + "prismarine-block": "=1.16.3", + "prismarine-entity": "^2.2.0", + "prismarine-item": "^1.12.1", + "prismarine-nbt": "^2.2.1", + "prismarine-recipe": "^1.3.1", + "prismarine-viewer": "^1.24.0", + "typescript": "^4.9.5", + "vec3": "^0.1.8", + "graceful-fs": "^4.2.11" + }, + "devDependencies": { + "prettier": "2.8.5" + } +} diff --git a/metagpt/environment/mincraft_env/process_monitor.py b/metagpt/environment/mincraft_env/process_monitor.py new file mode 100644 index 000000000..3183e42ed --- /dev/null +++ b/metagpt/environment/mincraft_env/process_monitor.py @@ -0,0 +1,78 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +import re +import subprocess +import threading +import warnings +from typing import List + +import psutil + +from metagpt.logs import define_log_level + + +class SubprocessMonitor: + def __init__( + self, + commands: List[str], + name: str, + ready_match: str = r".*", + callback_match: str = r"^(?!x)x$", # regex that will never match + callback: callable = None, + finished_callback: callable = None, + ): + self.commands = commands + self.name = name + self.logger = define_log_level(name=name) + self.process = None + self.ready_match = ready_match + self.ready_event = None + self.ready_line = None + self.callback_match = callback_match + self.callback = callback + self.finished_callback = finished_callback + self.thread = None + + def _start(self): + self.logger.info(f"Starting subprocess with commands: {self.commands}") + + self.process = psutil.Popen( + self.commands, + stdout=subprocess.PIPE, + stderr=subprocess.STDOUT, + universal_newlines=True, + ) + self.logger.info(f"Subprocess {self.name} started with PID {self.process.pid}.") + for line in iter(self.process.stdout.readline, ""): + self.logger.info(line.strip()) + if re.search(self.ready_match, line): + self.ready_line = line + self.logger.info("Subprocess is ready.") + self.ready_event.set() + if re.search(self.callback_match, line): + self.callback() + if not self.ready_event.is_set(): + self.ready_event.set() + warnings.warn(f"Subprocess {self.name} failed to start.") + if self.finished_callback: + self.finished_callback() + + def run(self): + self.ready_event = threading.Event() + self.ready_line = None + self.thread = threading.Thread(target=self._start) + self.thread.start() + self.ready_event.wait() + + def stop(self): + self.logger.info("Stopping subprocess.") + if self.process and self.process.is_running(): + self.process.terminate() + self.process.wait() + + @property + def is_running(self): + if self.process is None: + return False + return self.process.is_running() diff --git a/tests/metagpt/environment/mincraft_env/__init__.py b/tests/metagpt/environment/mincraft_env/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/tests/metagpt/environment/mincraft_env/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/tests/metagpt/environment/mincraft_env/test_mincraft_ext_env.py b/tests/metagpt/environment/mincraft_env/test_mincraft_ext_env.py new file mode 100644 index 000000000..b01168d42 --- /dev/null +++ b/tests/metagpt/environment/mincraft_env/test_mincraft_ext_env.py @@ -0,0 +1,14 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : the unittest of MincraftExtEnv + + +from metagpt.const import MC_CKPT_DIR +from metagpt.environment.mincraft_env.mincraft_ext_env import MincraftExtEnv + + +def test_mincraft_ext_env(): + ext_env = MincraftExtEnv() + assert ext_env.server, f"{ext_env.server_host}:{ext_env.server_port}" + assert MC_CKPT_DIR.joinpath("skill/code").exists() + assert ext_env.warm_up.get("optional_inventory_items") == 7 From 51f004141cb8ab5aa5cc6b70d13a5837b9e29753 Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 19:15:32 +0800 Subject: [PATCH 509/637] add software_env --- metagpt/environment/software_env/__init__.py | 3 +++ metagpt/environment/software_env/software_env.py | 12 ++++++++++++ 2 files changed, 15 insertions(+) create mode 100644 metagpt/environment/software_env/__init__.py create mode 100644 metagpt/environment/software_env/software_env.py diff --git a/metagpt/environment/software_env/__init__.py b/metagpt/environment/software_env/__init__.py new file mode 100644 index 000000000..2bcf8efd0 --- /dev/null +++ b/metagpt/environment/software_env/__init__.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : diff --git a/metagpt/environment/software_env/software_env.py b/metagpt/environment/software_env/software_env.py new file mode 100644 index 000000000..94bc11659 --- /dev/null +++ b/metagpt/environment/software_env/software_env.py @@ -0,0 +1,12 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : MG Software Env + + +from metagpt.environment.base_env import Environment + + +class SoftwareEnv(Environment): + """a specific alias name""" + + pass From 6b1f3ee39821e839293a30c762533c5a0ec95896 Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 20:39:56 +0800 Subject: [PATCH 510/637] add dalle/gpt4v example --- examples/dalle_gpt4v_agent.py | 80 +++++++++++++++++++++++++++++++++++ metagpt/utils/common.py | 2 +- 2 files changed, 81 insertions(+), 1 deletion(-) create mode 100644 examples/dalle_gpt4v_agent.py diff --git a/examples/dalle_gpt4v_agent.py b/examples/dalle_gpt4v_agent.py new file mode 100644 index 000000000..3974f5d9b --- /dev/null +++ b/examples/dalle_gpt4v_agent.py @@ -0,0 +1,80 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : use gpt4v to improve prompt and draw image with dall-e-3 + +""" +set the configuration in `config2.yaml` like below +``` +llm: + base_url: "xxx" + api_key: "sk-xxx" + model: "gpt-4-vision-preview" +``` +""" + +import asyncio + +from PIL import Image + +from metagpt.roles.role import Role +from metagpt.actions.action import Action +from metagpt.utils.common import encode_image +from metagpt.schema import Message +from metagpt.logs import logger + + +class GenAndImproveImageAction(Action): + save_image: bool = True + + async def generate_image(self, prompt: str) -> Image: + imgs = await self.llm.gen_image(model="dall-e-3", prompt=prompt) + return imgs[0] + + async def refine_prompt(self, old_prompt: str, image: Image) -> str: + msg = f"You are a creative painter, with the given generated image and old prompt: {old_prompt}, " \ + f"please refine the prompt and generate new one. Just output the new prompt." + b64_img = encode_image(image) + new_prompt = await self.llm.aask(msg=msg, images=[b64_img]) + return new_prompt + + async def evaluate_images(self, old_prompt: str, images: list[Image]) -> str: + msg = "With the prompt and two generated image, to judge if the second one is better than the first one. " \ + "If so, just output True else output False" + b64_imgs = [encode_image(img) for img in images] + res = await self.llm.aask(msg=msg, images=b64_imgs) + return res + + async def run(self, messages: list[Message]) -> str: + prompt = messages[-1].content + + old_img: Image = await self.generate_image(prompt) + new_prompt = await self.refine_prompt(old_prompt=prompt, image=old_img) + logger.info(f"original prompt: {prompt}") + logger.info(f"refined prompt: {new_prompt}") + new_img: Image = await self.generate_image(new_prompt) + if self.save_image: + old_img.save("./img_by-dall-e_old.png") + new_img.save("./img_by-dall-e_new.png") + res = await self.evaluate_images(old_prompt=prompt, images=[old_img, new_img]) + opinion = f"The second generated image is better than the first one: {res}" + logger.info(f"evaluate opinion: {opinion}") + return opinion + + +class Painter(Role): + name: str = "MaLiang" + profile: str = "Painter" + goal: str = "to generate fine painting" + + def __init__(self, **data): + super().__init__(**data) + + self.set_actions([GenAndImproveImageAction]) + + +async def main(): + role = Painter() + await role.run(with_message="a girl with flowers") + +if __name__ == "__main__": + asyncio.run(main()) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 79e7de5b6..2e05afa74 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -619,7 +619,7 @@ def load_mc_skills_code(skill_names: list[str] = None, skills_dir: Path = None) def encode_image(image_path_or_pil: Union[Path, Image], encoding: str = "utf-8") -> str: """encode image from file or PIL.Image into base64""" - if isinstance(image_path_or_pil, Image): + if isinstance(image_path_or_pil, Image.Image): buffer = BytesIO() image_path_or_pil.save(buffer, format="JPEG") bytes_data = buffer.getvalue() From c8df235303a1595ab34bb98750dc6dc3ed69e6e5 Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 20:42:41 +0800 Subject: [PATCH 511/637] update format --- examples/dalle_gpt4v_agent.py | 19 ++++++++++++------- 1 file changed, 12 insertions(+), 7 deletions(-) diff --git a/examples/dalle_gpt4v_agent.py b/examples/dalle_gpt4v_agent.py index 3974f5d9b..2b54b18a0 100644 --- a/examples/dalle_gpt4v_agent.py +++ b/examples/dalle_gpt4v_agent.py @@ -16,11 +16,11 @@ import asyncio from PIL import Image -from metagpt.roles.role import Role from metagpt.actions.action import Action -from metagpt.utils.common import encode_image -from metagpt.schema import Message from metagpt.logs import logger +from metagpt.roles.role import Role +from metagpt.schema import Message +from metagpt.utils.common import encode_image class GenAndImproveImageAction(Action): @@ -31,15 +31,19 @@ class GenAndImproveImageAction(Action): return imgs[0] async def refine_prompt(self, old_prompt: str, image: Image) -> str: - msg = f"You are a creative painter, with the given generated image and old prompt: {old_prompt}, " \ - f"please refine the prompt and generate new one. Just output the new prompt." + msg = ( + f"You are a creative painter, with the given generated image and old prompt: {old_prompt}, " + f"please refine the prompt and generate new one. Just output the new prompt." + ) b64_img = encode_image(image) new_prompt = await self.llm.aask(msg=msg, images=[b64_img]) return new_prompt async def evaluate_images(self, old_prompt: str, images: list[Image]) -> str: - msg = "With the prompt and two generated image, to judge if the second one is better than the first one. " \ - "If so, just output True else output False" + msg = ( + "With the prompt and two generated image, to judge if the second one is better than the first one. " + "If so, just output True else output False" + ) b64_imgs = [encode_image(img) for img in images] res = await self.llm.aask(msg=msg, images=b64_imgs) return res @@ -76,5 +80,6 @@ async def main(): role = Painter() await role.run(with_message="a girl with flowers") + if __name__ == "__main__": asyncio.run(main()) From f9e4a782c74cc24f99eeb20a8e622203a895663a Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 30 Jan 2024 20:46:53 +0800 Subject: [PATCH 512/637] update --- metagpt/environment/mincraft_env/mincraft_env.py | 1 + metagpt/environment/mincraft_env/process_monitor.py | 1 + 2 files changed, 2 insertions(+) diff --git a/metagpt/environment/mincraft_env/mincraft_env.py b/metagpt/environment/mincraft_env/mincraft_env.py index bc093eb61..0248244c5 100644 --- a/metagpt/environment/mincraft_env/mincraft_env.py +++ b/metagpt/environment/mincraft_env/mincraft_env.py @@ -1,6 +1,7 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- # @Desc : MG Mincraft Env +# refs to `voyager voyager.py` import json import re diff --git a/metagpt/environment/mincraft_env/process_monitor.py b/metagpt/environment/mincraft_env/process_monitor.py index 3183e42ed..b62aa6005 100644 --- a/metagpt/environment/mincraft_env/process_monitor.py +++ b/metagpt/environment/mincraft_env/process_monitor.py @@ -1,5 +1,6 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- +# refs to `voyager process_monitor.py` import re import subprocess From 4a7929d880acd921a0ee7db7052041fb1add272b Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 30 Jan 2024 21:04:33 +0800 Subject: [PATCH 513/637] rm immature code, improve naming, add unittest test rsp cache --- .../actions/{ml_da_action.py => ml_action.py} | 72 ++++++--- metagpt/actions/write_analysis_code.py | 141 +--------------- metagpt/actions/write_code_steps.py | 116 ------------- metagpt/actions/write_plan.py | 5 +- .../prompts/{ml_engineer.py => ml_action.py} | 99 +----------- metagpt/prompts/write_analysis_code.py | 95 +++++++++++ metagpt/roles/code_interpreter.py | 6 - metagpt/roles/kaggle_manager.py | 153 ------------------ metagpt/roles/ml_engineer.py | 3 +- metagpt/roles/tool_maker.py | 53 ------ tests/data/rsp_cache.json | 68 +++++++- .../actions/test_write_analysis_code.py | 7 +- tests/metagpt/roles/run_code_interpreter.py | 6 +- tests/metagpt/roles/test_code_interpreter.py | 17 +- tests/metagpt/roles/test_daml.py | 50 ------ tests/metagpt/roles/test_ml_engineer.py | 31 ++++ tests/metagpt/tools/libs/test_udf.py | 49 ------ tests/metagpt/utils/test_save_code.py | 8 +- 18 files changed, 275 insertions(+), 704 deletions(-) rename metagpt/actions/{ml_da_action.py => ml_action.py} (52%) delete mode 100644 metagpt/actions/write_code_steps.py rename metagpt/prompts/{ml_engineer.py => ml_action.py} (64%) create mode 100644 metagpt/prompts/write_analysis_code.py delete mode 100644 metagpt/roles/kaggle_manager.py delete mode 100644 metagpt/roles/tool_maker.py delete mode 100644 tests/metagpt/roles/test_daml.py create mode 100644 tests/metagpt/roles/test_ml_engineer.py delete mode 100644 tests/metagpt/tools/libs/test_udf.py diff --git a/metagpt/actions/ml_da_action.py b/metagpt/actions/ml_action.py similarity index 52% rename from metagpt/actions/ml_da_action.py rename to metagpt/actions/ml_action.py index d4e77773f..a61233e5a 100644 --- a/metagpt/actions/ml_da_action.py +++ b/metagpt/actions/ml_action.py @@ -1,28 +1,64 @@ import json +from typing import List, Tuple from metagpt.actions import Action -from metagpt.prompts.ml_engineer import PRINT_DATA_COLUMNS, UPDATE_DATA_COLUMNS -from metagpt.schema import Plan +from metagpt.actions.write_analysis_code import WriteCodeWithTools +from metagpt.prompts.ml_action import ( + GENERATE_CODE_PROMPT, + ML_TOOL_USAGE_PROMPT, + PRINT_DATA_COLUMNS, + UPDATE_DATA_COLUMNS, +) +from metagpt.prompts.write_analysis_code import CODE_GENERATOR_WITH_TOOLS +from metagpt.schema import Message, Plan from metagpt.utils.common import CodeParser, create_func_config, remove_comments -class SummarizeAnalysis(Action): - PROMPT_TEMPLATE: str = """ - # Context - {context} - # Summary - Output a 30-word summary on analysis tool and modeling algorithms you have used, and the corresponding result. Make sure to announce the complete path to your test prediction file. Your summary: - """ +class WriteCodeWithToolsML(WriteCodeWithTools): + async def run( + self, + context: List[Message], + plan: Plan = None, + column_info: str = "", + **kwargs, + ) -> Tuple[List[Message], str]: + # prepare tool schemas and tool-type-specific instruction + tool_schemas, tool_type_usage_prompt = await self._prepare_tools(plan=plan) - async def run(self, conmpleted_plan: Plan) -> str: - tasks = json.dumps( - [task.dict() for task in conmpleted_plan.tasks], - indent=4, - ensure_ascii=False, - ) # all tasks finished, return all task outputs - prompt = self.PROMPT_TEMPLATE.format(context=tasks) - summary = await self._aask(prompt) - return summary + # ML-specific variables to be used in prompt + code_steps = plan.current_task.code_steps + finished_tasks = plan.get_finished_tasks() + code_context = [remove_comments(task.code) for task in finished_tasks] + code_context = "\n\n".join(code_context) + + # prepare prompt depending on tool availability & LLM call + if tool_schemas: + prompt = ML_TOOL_USAGE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + tool_type_usage_prompt=tool_type_usage_prompt, + code_steps=code_steps, + tool_schemas=tool_schemas, + ) + + else: + prompt = GENERATE_CODE_PROMPT.format( + user_requirement=plan.goal, + history_code=code_context, + current_task=plan.current_task.instruction, + column_info=column_info, + tool_type_usage_prompt=tool_type_usage_prompt, + code_steps=code_steps, + ) + tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + rsp = await self.llm.aask_code(prompt, **tool_config) + + # Extra output to be used for potential debugging + context = [Message(content=prompt, role="user")] + + return context, rsp class Reflect(Action): diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index c6e504b9e..402f56ccc 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -4,19 +4,12 @@ @Author : orange-crow @File : write_code_v2.py """ -import re -from pathlib import Path from typing import Dict, List, Tuple, Union -from tenacity import retry, stop_after_attempt, wait_fixed - from metagpt.actions import Action -from metagpt.llm import LLM from metagpt.logs import logger -from metagpt.prompts.ml_engineer import ( +from metagpt.prompts.write_analysis_code import ( CODE_GENERATOR_WITH_TOOLS, - GENERATE_CODE_PROMPT, - ML_TOOL_USAGE_PROMPT, SELECT_FUNCTION_TOOLS, TOOL_RECOMMENDATION_PROMPT, TOOL_USAGE_PROMPT, @@ -24,7 +17,7 @@ from metagpt.prompts.ml_engineer import ( from metagpt.schema import Message, Plan from metagpt.tools import TOOL_REGISTRY from metagpt.tools.tool_registry import validate_tool_names -from metagpt.utils.common import create_func_config, remove_comments +from metagpt.utils.common import create_func_config class BaseWriteAnalysisCode(Action): @@ -195,133 +188,3 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): rsp = await self.llm.aask_code(prompt, **tool_config) return rsp - - -class WriteCodeWithToolsML(WriteCodeWithTools): - async def run( - self, - context: List[Message], - plan: Plan = None, - column_info: str = "", - **kwargs, - ) -> Tuple[List[Message], str]: - # prepare tool schemas and tool-type-specific instruction - tool_schemas, tool_type_usage_prompt = await self._prepare_tools(plan=plan) - - # ML-specific variables to be used in prompt - code_steps = plan.current_task.code_steps - finished_tasks = plan.get_finished_tasks() - code_context = [remove_comments(task.code) for task in finished_tasks] - code_context = "\n\n".join(code_context) - - # prepare prompt depending on tool availability & LLM call - if tool_schemas: - prompt = ML_TOOL_USAGE_PROMPT.format( - user_requirement=plan.goal, - history_code=code_context, - current_task=plan.current_task.instruction, - column_info=column_info, - tool_type_usage_prompt=tool_type_usage_prompt, - code_steps=code_steps, - tool_schemas=tool_schemas, - ) - - else: - prompt = GENERATE_CODE_PROMPT.format( - user_requirement=plan.goal, - history_code=code_context, - current_task=plan.current_task.instruction, - column_info=column_info, - tool_type_usage_prompt=tool_type_usage_prompt, - code_steps=code_steps, - ) - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) - rsp = await self.llm.aask_code(prompt, **tool_config) - - # Extra output to be used for potential debugging - context = [Message(content=prompt, role="user")] - - return context, rsp - - -class MakeTools(WriteCodeByGenerate): - DEFAULT_SYSTEM_MSG: str = """Convert any codes provied for you to a very General Function Code startswith `def`.\n - **Notice: - 1. Your code must contain a general function start with `def`. - 2. Refactor your code to get the most efficient implementation for large input data in the shortest amount of time. - 3. Must use Google style for function docstring, and your docstring must be consistent with the code,without missing anything. - 4. Write example code after `if __name__ == '__main__':`by using old varibales in old code, - and make sure it could be execute in the user's machine. - 5. Only use the imported packages** - """ - - def __init__(self, name: str = "", context: list[Message] = None, llm: LLM = None, workspace: str = None): - """ - :param str name: name, defaults to '' - :param list[Message] context: context, defaults to None - :param LLM llm: llm, defaults to None - :param str workspace: tools code saved file path dir, defaults to None - """ - super().__init__(name, context, llm) - self.workspace = workspace or str(Path(__file__).parents[1].joinpath("./tools/functions/libs/udf")) - self.file_suffix: str = ".py" - self.context = [] - - def parse_function_name(self, function_code: str) -> str: - # 定义正则表达式模式 - pattern = r"\bdef\s+([a-zA-Z_]\w*)\s*\(" - # 在代码中搜索匹配的模式 - match = re.search(pattern, function_code) - # 如果找到匹配项,则返回匹配的函数名;否则返回None - if match: - return match.group(1) - else: - return None - - def save(self, tool_code: str) -> None: - func_name = self.parse_function_name(tool_code) - if func_name is None: - raise ValueError(f"No function name found in {tool_code}") - saved_path = Path(self.workspace).joinpath(func_name + self.file_suffix) - logger.info(f"Saved tool_code {func_name} in {str(saved_path)}.") - saved_path.write_text(tool_code, encoding="utf-8") - - @retry(stop=stop_after_attempt(3), wait=wait_fixed(1)) - async def run(self, code: Union[str, List[dict]], code_desc: str = None, **kwargs) -> str: - # 拼接code prompt - code_prompt = f"The following code is about {code_desc}, convert it to be a General Function, {code}" - if not self.context: - self.context = self.process_msg(code_prompt) - else: - self.context.append(self.process_msg(code_prompt)[-1]) - logger.info(f"\n\nAsk to Make tools:\n{'-'*60}\n {self.context[-1]}") - - # 更新kwargs - if "code" in kwargs: - kwargs.pop("code") - if "code_desc" in kwargs: - kwargs.pop("code_desc") - - max_tries, current_try = 3, 0 - while True: - tool_code = await self.llm.aask_code(self.context, **kwargs) - func_name = self.parse_function_name(tool_code["code"]) - current_try += 1 - # make tools failed, add error message to context. - if not func_name: - logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") - logger.error(f"No function name found in code, we will retry make tools.\n{tool_code['code']}\n") - self.context.append( - {"role": "user", "content": "We need a general function in above code,but not found function."} - ) - # end make tools - if func_name is not None or current_try >= max_tries: - if current_try >= max_tries: - logger.error( - f"We have tried the maximum number of attempts {max_tries}\ - and still have not created tools successfully, we will skip it." - ) - break - logger.info(f"\n\nTools Respond\n{'-'*60}\n: {tool_code}") - self.save(tool_code["code"]) - return tool_code["code"] diff --git a/metagpt/actions/write_code_steps.py b/metagpt/actions/write_code_steps.py deleted file mode 100644 index 7ba22fde4..000000000 --- a/metagpt/actions/write_code_steps.py +++ /dev/null @@ -1,116 +0,0 @@ -import json - -from metagpt.actions import Action -from metagpt.schema import Plan -from metagpt.utils.common import CodeParser - -# CODE_STEPS_PROMPT_TEMPLATE = """ -# # Context -# {context} -# -# ----- -# Tasks are all code development tasks. -# You are a professional engineer, the main goal is to plan out concise solution steps for Current Task before coding. -# A planning process can reduce the difficulty and improve the quality of coding. -# You may be given some code plans for the tasks ahead, but you don't have to follow the existing plan when planning the current task. -# The output plan should following the subsequent principles: -# 1.The plan is a rough checklist of steps outlining the entire program's structure.Try to keep the number of steps fewer than 5. -# 2.The steps should be written concisely and at a high level, avoiding overly detailed implementation specifics. -# 3.The execution of the plan happens sequentially, but the plan can incorporate conditional (if) and looping(loop) keywords for more complex structures. -# -# Output the code steps in a JSON format, as shown in this example: -# ```json -# { -# "Step 1": "", -# "Step 2": "", -# "Step 3": "", -# ... -# } -# ``` -# """ - -CODE_STEPS_PROMPT_TEMPLATE = """ -# Context -{context} - ------ -Tasks are all code development tasks. -You are a professional engineer, the main goal is to plan out concise solution steps for Current Task before coding. -A planning process can reduce the difficulty and improve the quality of coding. -You may be given some code plans for the tasks ahead, but you don't have to follow the existing plan when planning the current task. -The output plan should following the subsequent principles: -1.The plan is a rough checklist of steps outlining the entire program's structure.Try to keep the number of steps fewer than 5. -2.The steps should be written concisely and at a high level, avoiding overly detailed implementation specifics. -3.The execution of the plan happens sequentially, but the plan can incorporate conditional (if) and looping(loop) keywords for more complex structures. -4.Design and provide code steps by following the code logic. Analyze the provided code step by step and reuse the imported library. - -Output the code steps in a JSON format, as shown in this example: -```json -{ - "Step 1": "", - "Step 2": "", - "Step 3": "", - ... -} -``` -""" - -# STRUCTURAL_CONTEXT = """ -# ## User Requirement -# {user_requirement} -# ## Current Plan -# {tasks} -# ## Current Task -# {current_task} -# """ - -STRUCTURAL_CONTEXT = """ -## User Requirement -{user_requirement} -## Plan -{tasks} -## Codes -{codes} -## Current Task -{current_task} -""" - - -class WriteCodeSteps(Action): - async def run(self, plan: Plan) -> str: - """Run of a task guide writing action, used in ml engineer - - Args: - plan (plan): task plan - useful_memories (list): useful_memories - Returns: - str: The dataset_descriptions string. - """ - - context = self.get_context(plan) - code_steps_prompt = CODE_STEPS_PROMPT_TEMPLATE.replace("{context}", context) - code_steps = await self._aask(code_steps_prompt) - code_steps = CodeParser.parse_code(block=None, text=code_steps) - return code_steps - - def get_context(self, plan: Plan): - user_requirement = plan.goal - # select_task_keys = ['task_id', 'instruction', 'is_finished', 'code'] - # select_task_keys = ['task_id','instruction'] - - def process_task(task): - task_dict = task.dict() - # ptask = {k: task_dict[k] for k in task_dict if k in select_task_keys } - ptask = f"task_id_{task_dict['task_id']}:{task_dict['instruction']}" - return ptask - - tasks = json.dumps([process_task(task) for task in plan.tasks], indent=4, ensure_ascii=False) - - code_lists = [task.code for task in plan.tasks if task.is_finished == True] - codes = "\n\n".join(code_lists) - current_task = json.dumps(process_task(plan.current_task)) if plan.current_task else {} - context = STRUCTURAL_CONTEXT.format( - user_requirement=user_requirement, tasks=tasks, codes=codes, current_task=current_task - ) - # print(context) - return context diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index 60dcef43b..335a09841 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -10,7 +10,10 @@ from typing import Dict, List, Tuple from metagpt.actions import Action from metagpt.logs import logger -from metagpt.prompts.ml_engineer import ASSIGN_TASK_TYPE_CONFIG, ASSIGN_TASK_TYPE_PROMPT +from metagpt.prompts.write_analysis_code import ( + ASSIGN_TASK_TYPE_CONFIG, + ASSIGN_TASK_TYPE_PROMPT, +) from metagpt.schema import Message, Plan, Task from metagpt.tools import TOOL_REGISTRY from metagpt.utils.common import CodeParser, create_func_config diff --git a/metagpt/prompts/ml_engineer.py b/metagpt/prompts/ml_action.py similarity index 64% rename from metagpt/prompts/ml_engineer.py rename to metagpt/prompts/ml_action.py index ac95e14bd..582b01146 100644 --- a/metagpt/prompts/ml_engineer.py +++ b/metagpt/prompts/ml_action.py @@ -2,7 +2,7 @@ # -*- coding: utf-8 -*- # @Time : 2023/11/24 15:43 # @Author : lidanyang -# @File : ml_engineer +# @File : ml_action # @Desc : UPDATE_DATA_COLUMNS = """ # Background @@ -49,85 +49,6 @@ Output the information in a JSON format, as shown in this example: - Don't contain specific values or examples found in the data column. """ -ASSIGN_TASK_TYPE_PROMPT = """ -Please assign a task type to each task in the list below from the given categories: -{task_list} - -## All Task Type: -{task_type_desc} -""" - -ASSIGN_TASK_TYPE_CONFIG = { - "name": "assign_task_type", - "description": "Assign task type to each task by order.", - "parameters": { - "type": "object", - "properties": { - "task_type": { - "type": "array", - "description": "List of task type. The length should as long as task list", - "items": { - "type": "string", - }, - }, - }, - "required": ["task_type"], - }, -} - -TOOL_RECOMMENDATION_PROMPT = """ -## User Requirement: -{current_task} - -## Task -Recommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. -This is a detailed code steps for current task. You can refer to it when recommending tools. -{code_steps} - -## Available Tools: -{available_tools} - -## Tool Selection and Instructions: -- Select tools most relevant to completing the 'User Requirement'. -- If you believe that no tools are suitable, indicate with an empty list. -- Only list the names of the tools, not the full schema of each tool. -- Ensure selected tools are listed in 'Available Tools'. -""" - -SELECT_FUNCTION_TOOLS = { - "name": "select_function_tools", - "description": "For current task, select suitable tools for it.", - "parameters": { - "type": "object", - "properties": { - "recommend_tools": { - "type": "array", - "description": "List of tool names. Empty list if no tool is suitable.", - "items": { - "type": "string", - }, - }, - }, - "required": ["recommend_tools"], - }, -} - -CODE_GENERATOR_WITH_TOOLS = { - "name": "add_subtask_code", - "description": "Add new code cell of current task to the end of an active Jupyter notebook.", - "parameters": { - "type": "object", - "properties": { - "code": { - "type": "string", - "description": "The code to be added to a new cell in jupyter.", - }, - }, - "required": ["code"], - }, -} - - PRINT_DATA_COLUMNS = { "name": "print_column_info", "description": "Print the latest column information after 'Done Tasks' code if first read or data changed.", @@ -189,24 +110,6 @@ model.fit(train, y_train) - The output code should contain all steps implemented in 'Code Steps'. """ -TOOL_USAGE_PROMPT = """ -# Instruction -Write complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc. -Specifically, {tool_type_usage_prompt} - -# Capabilities -- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class. -- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. - -# Available Tools (can be empty): -Each Class tool is described in JSON format. When you call a tool, import the tool first. -{tool_schemas} - -# Constraints: -- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. -- Always prioritize using pre-defined tools for the same functionality. -""" - ML_TOOL_USAGE_PROMPT = """ # Background As a data scientist, you need to help user to achieve their goal [{user_requirement}] step-by-step in an continuous Jupyter notebook. diff --git a/metagpt/prompts/write_analysis_code.py b/metagpt/prompts/write_analysis_code.py new file mode 100644 index 000000000..4c8a5081e --- /dev/null +++ b/metagpt/prompts/write_analysis_code.py @@ -0,0 +1,95 @@ +ASSIGN_TASK_TYPE_PROMPT = """ +Please assign a task type to each task in the list below from the given categories: +{task_list} + +## All Task Type: +{task_type_desc} +""" + +ASSIGN_TASK_TYPE_CONFIG = { + "name": "assign_task_type", + "description": "Assign task type to each task by order.", + "parameters": { + "type": "object", + "properties": { + "task_type": { + "type": "array", + "description": "List of task type. The length should as long as task list", + "items": { + "type": "string", + }, + }, + }, + "required": ["task_type"], + }, +} + +TOOL_RECOMMENDATION_PROMPT = """ +## User Requirement: +{current_task} + +## Task +Recommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. +This is a detailed code steps for current task. You can refer to it when recommending tools. +{code_steps} + +## Available Tools: +{available_tools} + +## Tool Selection and Instructions: +- Select tools most relevant to completing the 'User Requirement'. +- If you believe that no tools are suitable, indicate with an empty list. +- Only list the names of the tools, not the full schema of each tool. +- Ensure selected tools are listed in 'Available Tools'. +""" + +SELECT_FUNCTION_TOOLS = { + "name": "select_function_tools", + "description": "For current task, select suitable tools for it.", + "parameters": { + "type": "object", + "properties": { + "recommend_tools": { + "type": "array", + "description": "List of tool names. Empty list if no tool is suitable.", + "items": { + "type": "string", + }, + }, + }, + "required": ["recommend_tools"], + }, +} + +CODE_GENERATOR_WITH_TOOLS = { + "name": "add_subtask_code", + "description": "Add new code cell of current task to the end of an active Jupyter notebook.", + "parameters": { + "type": "object", + "properties": { + "code": { + "type": "string", + "description": "The code to be added to a new cell in jupyter.", + }, + }, + "required": ["code"], + }, +} + +TOOL_USAGE_PROMPT = """ +# Instruction +Write complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc. +Specifically, {tool_type_usage_prompt} + +# Capabilities +- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class. +- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. + +# Available Tools (can be empty): +Each Class tool is described in JSON format. When you call a tool, import the tool first. +{tool_schemas} + +# Constraints: +- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. +- Always prioritize using pre-defined tools for the same functionality. +""" diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index d1136a1d4..b4f9622d3 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -3,7 +3,6 @@ from pydantic import Field from metagpt.actions.ask_review import ReviewConst from metagpt.actions.execute_code import ExecutePyCode from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools -from metagpt.actions.write_code_steps import WriteCodeSteps from metagpt.logs import logger from metagpt.roles import Role from metagpt.schema import Message, Task, TaskResult @@ -12,7 +11,6 @@ from metagpt.schema import Message, Task, TaskResult class CodeInterpreter(Role): auto_run: bool = True use_tools: bool = False - use_code_steps: bool = False execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) tools: list[str] = [] @@ -48,10 +46,6 @@ class CodeInterpreter(Role): return task_result async def _write_and_exec_code(self, max_retry: int = 3): - self.planner.current_task.code_steps = ( - await WriteCodeSteps().run(self.planner.plan) if self.use_code_steps else "" - ) - counter = 0 success = False diff --git a/metagpt/roles/kaggle_manager.py b/metagpt/roles/kaggle_manager.py deleted file mode 100644 index 3ef573a8c..000000000 --- a/metagpt/roles/kaggle_manager.py +++ /dev/null @@ -1,153 +0,0 @@ -import json -import os -import subprocess - -import fire -import pandas as pd - -from metagpt.actions import Action, UserRequirement -from metagpt.actions.ml_da_action import SummarizeAnalysis -from metagpt.config import CONFIG -from metagpt.logs import logger -from metagpt.roles import Role -from metagpt.schema import Message -from metagpt.utils.common import CodeParser - -os.environ["KAGGLE_USERNAME"] = CONFIG.kaggle_username -os.environ["KAGGLE_KEY"] = CONFIG.kaggle_key - - -def run_command(cmd): - print(cmd) - output = subprocess.run(cmd, shell=True, capture_output=True, text=True) - if output.returncode != 0: - print("Error output:", output.stderr) - exit() - else: - print(output.stdout) - return output.stdout - - -class DownloadData(Action): - async def run(self, competition, data_desc="") -> str: - data_path = CONFIG.workspace_path / competition - - output = run_command(f"kaggle competitions list --search {competition}") - assert output != "No competitions found", "You must provide the correct competition name" - - run_command(f"kaggle competitions download {competition} --path {WORKSPACE_ROOT}") - - if not os.path.exists(data_path): - # if True: - # run_command(f"rm -r {data_path / '*'}") - run_command(f"unzip -o {CONFIG.workspace_path / '*.zip'} -d {data_path}") # FIXME: not safe - - file_list = run_command(f"ls {data_path}") - - rsp = f""" - Location: - Data downloaded at {data_path} folder, including {file_list} - Data Description: - {data_desc} - """ - return rsp - - -class SubmitResult(Action): - PROMPT_TEMPLATE: str = """ - # Summary - __summary__ - # Your task - Extract the file path for test set prediction from the summary above, output a json following the format: - ```json - {"file_path": str = "the file path, for example, /path/to/the/prediction/file/xxx.csv, /path/to/the/prediction/file/xxx.xlsx"} - ``` - """ - - def __init__(self, name: str = "", context=None, llm=None) -> str: - super().__init__(name, context, llm) - - async def _parse_submit_file_path(self, context) -> str: - prompt = self.PROMPT_TEMPLATE.replace("__summary__", context) - rsp = await self._aask(prompt) - rsp = CodeParser.parse_code(block=None, text=rsp) - file_path = json.loads(rsp)["file_path"] - return file_path - - async def run(self, competition, submit_message="") -> str: - submit_file_path = await self._parse_submit_file_path(submit_message) - - data_path = CONFIG.workspace_path / competition - submit_message = submit_message.replace("'", "") - - run_command(f"kaggle competitions submit {competition} -f {submit_file_path} -m '{submit_message}'") - run_command(f"kaggle competitions leaderboard --show --csv {competition} > {data_path / 'leaderboard.csv'}") - run_command(f"kaggle competitions submissions --csv {competition} > {data_path / 'submission.csv'}") - - leaderboard = pd.read_csv(data_path / "leaderboard.csv") - submission = pd.read_csv(data_path / "submission.csv") - print(submission) # submission.to_json(orient="records") - - submission_score = submission.loc[0, "publicScore"] - best_score = max(submission["publicScore"]) # might be min - rank = leaderboard.loc[leaderboard["score"] == best_score].index[0] - rank_pct = round(rank / len(leaderboard), 4) * 100 - - submission_summary = f""" - # All histories: - {submission.head(5).to_string()} - # Current - Current submission score: {submission_score}, best score: {best_score}, best rank: {rank} (top {rank_pct}%) - """ - logger.info(submission_summary) - return submission_summary - - -class KaggleManager(Role): - def __init__(self, name="ABC", profile="KaggleManager", goal="", competition="titanic", data_desc=""): - super().__init__(name=name, profile=profile, goal=goal) - self._init_actions([DownloadData, SubmitResult]) - self._watch([UserRequirement, SummarizeAnalysis]) - self.competition = competition - self.data_desc = data_desc # currently passed in, later can be scrapped down from web by another Role - - async def _think(self): - observed = self.get_memories()[-1].cause_by - if observed == UserRequirement: - self._set_state(0) # DownloadData, get competition of interest from human, download datasets - elif observed == SummarizeAnalysis: - self._set_state(1) # SubmitResult, get prediction from MLEngineer and submit it to Kaggle - - async def _act(self): - todo = self.rc.todo - logger.info(f"{self._setting}: ready to {self.rc.todo}") - - if isinstance(todo, DownloadData): - rsp = await todo.run(self.competition, self.data_desc) - - elif isinstance(todo, SubmitResult): - submit_message = self.get_memories()[ - -1 - ].content # use analysis summary from MLEngineer as submission message - rsp = await todo.run(competition=self.competition, submit_message=submit_message) - - msg = Message(content=rsp, role="user", cause_by=type(todo)) - - return msg - - -if __name__ == "__main__": - competition, data_desc, requirement = ( - "titanic", - "Training set is train.csv.\nTest set is test.csv. We also include gender_submission.csv, a set of predictions that assume all and only female passengers survive, as an example of what a submission file should look like.", - "Run EDA on the train dataset, train a model to predict survival (20% as validation) and save it, predict the test set using saved model, save the test result according to format", - ) - - summary = "I used Python with pandas for data preprocessing, sklearn's RandomForestClassifier for modeling, and achieved 82.12% accuracy on validation. Predictions saved at '/Users/gary/Desktop/data_agents_opt/workspace/titanic/gender_submission.csv'." - - async def main(requirement: str = requirement): - role = KaggleManager(competition=competition, data_desc=data_desc) - # await role.run(Message(content="", cause_by=UserRequirement)) - await role.run(Message(content=summary, cause_by=SummarizeAnalysis)) - - fire.Fire(main) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index d1a22b9d3..e7abee560 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,7 +1,6 @@ from metagpt.actions.debug_code import DebugCode from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.ml_da_action import UpdateDataColumns -from metagpt.actions.write_analysis_code import WriteCodeWithToolsML +from metagpt.actions.ml_action import UpdateDataColumns, WriteCodeWithToolsML from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter from metagpt.tools.tool_data_type import ToolTypeEnum diff --git a/metagpt/roles/tool_maker.py b/metagpt/roles/tool_maker.py deleted file mode 100644 index 68d84b1e6..000000000 --- a/metagpt/roles/tool_maker.py +++ /dev/null @@ -1,53 +0,0 @@ -from pydantic import Field - -from metagpt.actions.ask_review import AskReview -from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_analysis_code import MakeTools -from metagpt.logs import logger -from metagpt.roles import Role -from metagpt.utils.common import remove_comments - - -class ToolMaker(Role): - execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) - - async def make_tool(self, code: str, instruction: str, task_id: str = "", auto_run=True): - if len(remove_comments(code).split("\n")) < 5: # no need to consider trivial codes with fewer than 5 lines - return - - logger.warning( - f"Making tools for task_id {task_id}: \ - `{instruction}` \n code: \n {code}" - ) - make_tools = MakeTools() - make_tool_retries, make_tool_current_retry = 3, 0 - while True: - # start make tools - tool_code = await make_tools.run(code, instruction) - make_tool_current_retry += 1 - - # check tool_code by execute_code - logger.info(f"Checking task_id {task_id} tool code by executor...") - execute_result, execute_success = await self.execute_code.run(tool_code) - if not execute_success: - logger.error(f"Tool code faild to execute, \n{execute_result}\n.We will try to fix it ...") - # end make tools - if execute_success or make_tool_current_retry >= make_tool_retries: - if make_tool_current_retry >= make_tool_retries: - logger.error( - f"We have tried the maximum number of attempts {make_tool_retries}\ - and still have not created tools for task_id {task_id} successfully,\ - we will skip it." - ) - break - # save successful tool code in udf - if execute_success: - _, confirmed = await self.ask_review(auto_run=auto_run) - if confirmed: - make_tools.save(tool_code) - - async def ask_review(self, auto_run: bool = True): - if not auto_run: - review, confirmed = await AskReview().run() - return review, confirmed - return "", True diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index db452f676..cb3c1642c 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -141,5 +141,71 @@ "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.txt\n## Development Code\n```python\nprint('Hello, World')\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\n\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\ninstruction: There are no errors in the provided code.\n\nFile To Rewrite: N/A\n\nStatus: PASS\n\nSend To: NoOne\n---", "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.sh\n## Development Code\n```python\necho 'Hello World'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\necho Hello World\n## Running Output\nstandard output: \n```text\nHello World\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "The running result indicates no error. The code works fine.\n\n## File To Rewrite:\nNo file needs to be rewritten.\n\n## Status:\nPASS\n\n## Send To:\nNoOne\n\n---", "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.py\n## Development Code\n```python\npython -c \"print(1/0)\"\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\npython -c print(1/0)\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\nTraceback (most recent call last):\n File \"\", line 1, in \nZeroDivisionError: division by zero\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\nThe error is caused by the development code in file a.py, which attempts to divide by zero. To fix the error, the development code in a.py should be modified to handle the ZeroDivisionError, for example by using a try-except block.\n\nFile To Rewrite:\na.py\n\nStatus:\nFAIL\n\nSend To:\nEngineer\n---", - "\nNOTICE\n1. Role: You are a Development Engineer or QA engineer;\n2. Task: You received this message from another Development Engineer or QA engineer who ran or tested your code. \nBased on the message, first, figure out your own role, i.e. Engineer or QaEngineer,\nthen rewrite the development code or the test code based on your role, the error, and the summary, such that all bugs are fixed and the code performs well.\nAttention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script and triple quotes.\nThe message is as follows:\n# Legacy Code\n```python\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n if self.score > 21 and any(card.rank == 'A' for card in self.hand):\n self.score -= 10\n return self.score\n\n```\n---\n# Unit Test Code\n```python\n\nimport unittest\nfrom blackjack_game.player import Player\nfrom blackjack_game.deck import Deck\nfrom blackjack_game.card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n\n\n```\n---\n# Console logs\n```text\n\n```\n---\nNow you should start rewriting the code:\n## file name of the code to rewrite: Write code with triple quote. Do your best to implement THIS IN ONLY ONE FILE.\n": "```python\n## Development Engineer\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n ace_count = sum(1 for card in self.hand if card.rank == 'A')\n while self.score > 21 and ace_count > 0:\n self.score -= 10\n ace_count -= 1\n return self.score\n```\n\n```python\n## QaEngineer\n\nimport unittest\nfrom player import Player\nfrom deck import Deck\nfrom card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n```" + "\nNOTICE\n1. Role: You are a Development Engineer or QA engineer;\n2. Task: You received this message from another Development Engineer or QA engineer who ran or tested your code. \nBased on the message, first, figure out your own role, i.e. Engineer or QaEngineer,\nthen rewrite the development code or the test code based on your role, the error, and the summary, such that all bugs are fixed and the code performs well.\nAttention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script and triple quotes.\nThe message is as follows:\n# Legacy Code\n```python\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n if self.score > 21 and any(card.rank == 'A' for card in self.hand):\n self.score -= 10\n return self.score\n\n```\n---\n# Unit Test Code\n```python\n\nimport unittest\nfrom blackjack_game.player import Player\nfrom blackjack_game.deck import Deck\nfrom blackjack_game.card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n\n\n```\n---\n# Console logs\n```text\n\n```\n---\nNow you should start rewriting the code:\n## file name of the code to rewrite: Write code with triple quote. Do your best to implement THIS IN ONLY ONE FILE.\n": "```python\n## Development Engineer\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n ace_count = sum(1 for card in self.hand if card.rank == 'A')\n while self.score > 21 and ace_count > 0:\n self.score -= 10\n ace_count -= 1\n return self.score\n```\n\n```python\n## QaEngineer\n\nimport unittest\nfrom player import Player\nfrom deck import Deck\nfrom card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n```", + "\n # Context:\n user: \n## User Requirement\nThis is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Perform exploratory data analysis on the train dataset to understand the features and target variable.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Preprocess the data by handling missing values, encoding categorical variables, and scaling features.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Conduct feature engineering to create new features that may help improve model performance.\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Select and train a machine learning model using the processed train dataset.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Evaluate the model's accuracy using the eval dataset and report the results.\"\n }\n]\n```", + "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Perform exploratory data analysis on the train dataset to understand the features and target variable.\\nTask 2: Preprocess the data by handling missing values, encoding categorical variables, and scaling features.\\nTask 3: Conduct feature engineering to create new features that may help improve model performance.\\nTask 4: Select and train a machine learning model using the processed train dataset.\\nTask 5: Evaluate the model's accuracy using the eval dataset and report the results.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { + "task_type": [ + "eda", + "data_preprocess", + "feature_engineering", + "model_train", + "model_evaluate" + ] + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n\\n```end\\n\\n## Current Task\\nPerform exploratory data analysis on the train dataset to understand the features and target variable.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { + "code": "# Step 1: Load the training dataset\nimport pandas as pd\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\n\n# Step 2: Display the first few rows of the dataset\ndisplay(train_data.head())\n\n# Step 3: Get a summary of the dataset\ntrain_data.info()\n\n# Step 4: Describe the numerical features\ntrain_data.describe()\n\n# Step 5: Describe the categorical features\ntrain_data.describe(include=['O'])\n\n# Step 6: Check for missing values\nmissing_values = train_data.isnull().sum()\nmissing_values[missing_values > 0]\n\n# Step 7: Visualize the distribution of the target variable 'Survived'\nimport seaborn as sns\nimport matplotlib.pyplot as plt\nsns.countplot(x='Survived', data=train_data)\nplt.title('Distribution of Survival')\nplt.show()\n\n# Step 8: Explore the relationship between features and the target variable\n# Visualize the survival rate by sex\nsns.barplot(x='Sex', y='Survived', data=train_data)\nplt.title('Survival Rate by Sex')\nplt.show()\n\n# Visualize the survival rate by class\nsns.barplot(x='Pclass', y='Survived', data=train_data)\nplt.title('Survival Rate by Class')\nplt.show()\n\n# Visualize the survival rate by embarkation port\nsns.barplot(x='Embarked', y='Survived', data=train_data)\nplt.title('Survival Rate by Embarkation Port')\nplt.show()" + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import get_column_info\n\ncolumn_info = get_column_info(train_data)\nprint(\"column_info\")\nprint(column_info)" + }, + "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nPreprocess the data by handling missing values, encoding categorical variables, and scaling features.\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Completing missing values with simple strategies'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { + "recommend_tools": [ + "FillMissingValue" + ] + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n```end\\n\\n## Current Task\\nPreprocess the data by handling missing values, encoding categorical variables, and scaling features.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Name', 'Sex', 'Ticket', 'Cabin', 'Embarked'], 'Numeric': ['PassengerId', 'Survived', 'Pclass', 'Age', 'SibSp', 'Parch', 'Fare'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}, 'strategy': {'type': 'str', 'description': 'the imputation strategy, notice mean/median can only be used for numeric features', 'default': 'mean', 'enum': ['mean', 'median', 'most_frequent', 'constant']}, 'fill_value': {'type': 'int', 'description': 'fill_value is used to replace all occurrences of missing_values', 'default': None}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { + "code": "# Step 1: Handle missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create copies of the datasets to avoid changing the original data\ntrain_processed = train_data.copy()\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\n\n# Fill missing values for 'Age' with median and 'Embarked' with most frequent value\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\ntrain_processed = fill_missing_age.fit_transform(train_processed)\neval_processed = fill_missing_age.transform(eval_data)\n\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\neval_processed = fill_missing_embarked.transform(eval_data)\n\n# Since 'Cabin' has a lot of missing values, we'll create a binary feature indicating whether cabin info is missing\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\n\n# Step 2: Encode categorical variables\nfrom sklearn.preprocessing import LabelEncoder\n\nlabel_encoder = LabelEncoder()\n\n# Encode 'Sex'\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\n\n# Encode 'Embarked'\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\n\n# Step 3: Scale features\nfrom sklearn.preprocessing import StandardScaler\n\nscaler = StandardScaler()\n\n# Scale 'Age' and 'Fare'\nfor feature in ['Age', 'Fare']:\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\n\n# Display the first few rows of the processed train dataset\ndisplay(train_processed.head())" + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import get_column_info\ncolumn_info = get_column_info(train_processed)\nprint(\"column_info\")\nprint(column_info)" + }, + "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nConduct feature engineering to create new features that may help improve model performance.\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'CatCross': 'Add pairwise crossed features and convert them to numerical features.'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { + "recommend_tools": [ + "CatCross" + ] + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n```end\\n\\n## Current Task\\nConduct feature engineering to create new features that may help improve model performance.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Name', 'Ticket', 'Cabin'], 'Numeric': ['PassengerId', 'Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about feature engineering. when performing it, please adhere to the following principles:\\n- Generate as diverse features as possible to improve the model's performance step-by-step. \\n- If potential impactful features are not included in 'Code Steps', add new steps to generate them.\\n- Avoid creating redundant or excessively numerous features in one step.\\n- Exclude ID columns from feature generation and remove them.\\n- Each step do feature engineering to train, must do same for test separately at the same time.\\n- Avoid using the label column to create features, except for cat encoding.\\n- Use the data from previous task result if exist, do not mock or reload data yourself.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'CatCross': {'type': 'class', 'description': 'Add pairwise crossed features and convert them to numerical features.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'cols': {'type': 'list', 'description': 'Columns to be pairwise crossed, at least 2 columns.'}, 'max_cat_num': {'type': 'int', 'description': 'Maximum unique categories per crossed feature.', 'default': 100}}}, 'required': ['cols']}, 'fit': {'description': 'Fit the CatCross model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/feature_engineering.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { + "code": "# Step 1: Create a feature for the number of family members on board\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\n\n# Step 2: Create a feature to indicate whether a passenger was alone or not\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\n\n# Step 3: Create a feature for the title extracted from the passenger's name\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\.', expand=False)\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\.', expand=False)\n\n# Step 4: Create a feature for the deck extracted from the cabin number\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\n\n# Step 5: Create a feature for the ticket class and sex cross\n# Tools used: ['CatCross']\nfrom metagpt.tools.libs.feature_engineering import CatCross\n\n# Initialize CatCross with the desired columns\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\n\n# Fit and transform the train data\ntrain_processed = cat_cross.fit_transform(train_processed)\n\n# Transform the eval data\neval_processed = cat_cross.transform(eval_processed)\n\n# Remove unnecessary columns\nfor df in [train_processed, eval_processed]:\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)" + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import get_column_info\ncolumn_info = get_column_info(train_processed)\nprint(\"column_info\")\nprint(column_info)" + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n## Current Task\\nSelect and train a machine learning model using the processed train dataset.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Title', 'Deck'], 'Numeric': ['Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind', 'Family_Size', 'Is_Alone', 'Pclass_Sex'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about training a model, please ensure high performance:\\n- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as XGBoost, CatBoost, etc.\\n- If non-numeric columns exist, perform label encode together with all steps.\\n- Use the data from previous task result directly, do not mock or reload data yourself.\\n- Set suitable hyperparameters for the model, make metrics as high as possible.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { + "code": "from sklearn.model_selection import train_test_split\nfrom sklearn.metrics import accuracy_score\nfrom xgboost import XGBClassifier\n\n# Step 1: Prepare the data\n# Separate features and target variable\nX = train_processed.drop('Survived', axis=1)\ny = train_processed['Survived']\n\n# Step 2: Split the data into training and validation sets\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\n\n# Step 3: Initialize and train the XGBoost model\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\nmodel.fit(X_train, y_train)\n\n# Step 4: Evaluate the model\ny_pred = model.predict(X_val)\naccuracy = accuracy_score(y_val, y_pred)\nprint(f'Validation Accuracy: {accuracy:.4f}')" + }, + "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\n[user: \\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n## Current Task\\nSelect and train a machine learning model using the processed train dataset.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Title', 'Deck'], 'Numeric': ['Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind', 'Family_Size', 'Is_Alone', 'Pclass_Sex'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about training a model, please ensure high performance:\\n- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as XGBoost, CatBoost, etc.\\n- If non-numeric columns exist, perform label encode together with all steps.\\n- Use the data from previous task result directly, do not mock or reload data yourself.\\n- Set suitable hyperparameters for the model, make metrics as high as possible.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n]\\n\\n[previous impl]\\nfrom sklearn.model_selection import train_test_split\\nfrom sklearn.metrics import accuracy_score\\nfrom xgboost import XGBClassifier\\n\\n# Step 1: Prepare the data\\n# Separate features and target variable\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\n\\n# Step 2: Split the data into training and validation sets\\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\\n\\n# Step 3: Initialize and train the XGBoost model\\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X_train, y_train)\\n\\n# Step 4: Evaluate the model\\ny_pred = model.predict(X_val)\\naccuracy = accuracy_score(y_val, y_pred)\\nprint(f'Validation Accuracy: {accuracy:.4f}')\\n[runtime Error]\\n[assistant: from sklearn.model_selection import train_test_split\\nfrom sklearn.metrics import accuracy_score\\nfrom xgboost import XGBClassifier\\n\\n# Step 1: Prepare the data\\n# Separate features and target variable\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\n\\n# Step 2: Split the data into training and validation sets\\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\\n\\n# Step 3: Initialize and train the XGBoost model\\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X_train, y_train)\\n\\n# Step 4: Evaluate the model\\ny_pred = model.predict(X_val)\\naccuracy = accuracy_score(y_val, y_pred)\\nprint(f'Validation Accuracy: {accuracy:.4f}'), user: Executed code failed, please reflect the cause of bug and then debug. Truncated to show only last 2000 characters\\n= self._temporary_data\\n 622 else:\\n--> 623 new, cat_codes, feature_names, feature_types = _proxy_transform(\\n 624 data,\\n 625 feature_names,\\n 626 feature_types,\\n 627 self._enable_categorical,\\n 628 )\\n 629 # Stage the data, meta info are copied inside C++ MetaInfo.\\n 630 self._temporary_data = (new, cat_codes, feature_names, feature_types)\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:1315, in _proxy_transform(data, feature_names, feature_types, enable_categorical)\\n 1313 data = pd.DataFrame(data)\\n 1314 if _is_pandas_df(data):\\n-> 1315 arr, feature_names, feature_types = _transform_pandas_df(\\n 1316 data, enable_categorical, feature_names, feature_types\\n 1317 )\\n 1318 arr, _ = _ensure_np_dtype(arr, arr.dtype)\\n 1319 return arr, None, feature_names, feature_types\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:490, in _transform_pandas_df(data, enable_categorical, feature_names, feature_types, meta, meta_type)\\n 483 for dtype in data.dtypes:\\n 484 if not (\\n 485 (dtype.name in _pandas_dtype_mapper)\\n 486 or is_pd_sparse_dtype(dtype)\\n 487 or (is_pd_cat_dtype(dtype) and enable_categorical)\\n 488 or is_pa_ext_dtype(dtype)\\n 489 ):\\n--> 490 _invalid_dataframe_dtype(data)\\n 491 if is_pa_ext_dtype(dtype):\\n 492 pyarrow_extension = True\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:308, in _invalid_dataframe_dtype(data)\\n 306 type_err = \\\"DataFrame.dtypes for data must be int, float, bool or category.\\\"\\n 307 msg = f\\\"\\\"\\\"{type_err} {_ENABLE_CAT_ERR} {err}\\\"\\\"\\\"\\n--> 308 raise ValueError(msg)\\n\\nValueError: DataFrame.dtypes for data must be int, float, bool or category. When categorical type is supplied, The experimental DMatrix parameter`enable_categorical` must be set to `True`. Invalid columns:Title: object, Deck: object\\n\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { + "reflection": "The error message indicates that the XGBoost model cannot handle columns with 'object' data types directly. The columns 'Title' and 'Deck' are of 'object' type and need to be encoded as numeric values before they can be used for training the XGBoost model. The error also suggests that if we want to use categorical data, we must set the `enable_categorical` parameter to `True` when creating the DMatrix. However, a more common practice is to convert these object types to numeric types using techniques such as label encoding or one-hot encoding. To resolve this issue, we should encode the 'Title' and 'Deck' columns using label encoding before training the model. Additionally, we should ensure that any other non-numeric columns are also converted to numeric types if they exist.", + "improved_impl": "from sklearn.preprocessing import LabelEncoder\n\n# Step 1: Encode non-numeric columns\nencoder = LabelEncoder()\ntrain_processed['Title'] = encoder.fit_transform(train_processed['Title'])\neval_processed['Title'] = encoder.transform(eval_processed['Title'])\ntrain_processed['Deck'] = encoder.fit_transform(train_processed['Deck'].astype(str))\neval_processed['Deck'] = encoder.transform(eval_processed['Deck'].astype(str))\n\n# Step 2: Prepare the data\n# Separate features and target variable\nX = train_processed.drop('Survived', axis=1)\ny = train_processed['Survived']\n\n# Step 3: Train the XGBoost model\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\nmodel.fit(X, y)\n\n# Step 4: Evaluate the model on evaluation data\nX_eval = eval_processed.drop('Survived', axis=1)\ny_eval = eval_processed['Survived']\ny_pred_eval = model.predict(X_eval)\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')" + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n\\nfrom sklearn.preprocessing import LabelEncoder\\nencoder = LabelEncoder()\\ntrain_processed['Title'] = encoder.fit_transform(train_processed['Title'])\\neval_processed['Title'] = encoder.transform(eval_processed['Title'])\\ntrain_processed['Deck'] = encoder.fit_transform(train_processed['Deck'].astype(str))\\neval_processed['Deck'] = encoder.transform(eval_processed['Deck'].astype(str))\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X, y)\\nX_eval = eval_processed.drop('Survived', axis=1)\\ny_eval = eval_processed['Survived']\\ny_pred_eval = model.predict(X_eval)\\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')\\n```end\\n\\n## Current Task\\nEvaluate the model's accuracy using the eval dataset and report the results.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about evaluating a model, please note the following:\\n- Ensure that the evaluated data is same processed as the training data. If not, remember use object in 'Done Tasks' to transform the data.\\n- Use trained model from previous task result directly, do not mock or reload model yourself.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { + "code": "# Evaluate the model's accuracy using the eval dataset\nfrom sklearn.metrics import accuracy_score\n\n# Predict the target for the evaluation set\ny_pred_eval = model.predict(X_eval)\n\n# Calculate the accuracy for the evaluation set\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\n\n# Report the accuracy\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')" + }, + "\n # Context:\n user: \n## User Requirement\nRun data analysis on sklearn Iris dataset, include a plot\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Load the sklearn Iris dataset.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis on the Iris dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Create a plot visualizing the Iris dataset features.\"\n }\n]\n```", + "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Load the sklearn Iris dataset.\\nTask 2: Perform exploratory data analysis on the Iris dataset.\\nTask 3: Create a plot visualizing the Iris dataset features.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { + "task_type": [ + "other", + "eda", + "other" + ] + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"1\\\",\\\"dependent_task_ids\\\":[],\\\"instruction\\\":\\\"Load the sklearn Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from sklearn.datasets import load_iris\niris_data = load_iris()" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "import pandas as pd\n\n# Convert the loaded Iris dataset into a DataFrame for easier manipulation\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\niris_df['target'] = iris_data.target\n\n# Display basic statistical details like percentile, mean, std etc. of a data frame\niris_stats = iris_df.describe()\n\n# Display the first few rows of the DataFrame\niris_head = iris_df.head()\n\n# Display the class distribution\niris_target_counts = iris_df['target'].value_counts()\n\n# Output the results\nprint('Basic Statistical Details:\\n', iris_stats)\nprint('\\nFirst Five Rows:\\n', iris_head)\nprint('\\nClass Distribution:\\n', iris_target_counts)" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\n\\\\n# Convert the loaded Iris dataset into a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\\\\niris_df['target'] = iris_data.target\\\\n\\\\n# Display basic statistical details like percentile, mean, std etc. of a data frame\\\\niris_stats = iris_df.describe()\\\\n\\\\n# Display the first few rows of the DataFrame\\\\niris_head = iris_df.head()\\\\n\\\\n# Display the class distribution\\\\niris_target_counts = iris_df['target'].value_counts()\\\\n\\\\n# Output the results\\\\nprint('Basic Statistical Details:\\\\\\\\n', iris_stats)\\\\nprint('\\\\\\\\nFirst Five Rows:\\\\\\\\n', iris_head)\\\\nprint('\\\\\\\\nClass Distribution:\\\\\\\\n', iris_target_counts)\\\",\\n \\\"result\\\": \\\"Basic Statistical Details:\\\\n sepal length (cm) sepal width (cm) petal length (cm) \\\\\\\\\\\\ncount 150.000000 150.000000 150.000000 \\\\nmean 5.843333 3.057333 3.758000 \\\\nstd 0.828066 0.435866 1.765298 \\\\nmin 4.300000 2.000000 1.000000 \\\\n25% 5.100000 2.800000 1.600000 \\\\n50% 5.800000 3.000000 4.350000 \\\\n75% 6.400000 3.300000 5.100000 \\\\nmax 7.900000 4.400000 6.900000 \\\\n\\\\n petal width (cm) target \\\\ncount 150.000000 150.000000 \\\\nmean 1.199333 1.000000 \\\\nstd 0.762238 0.819232 \\\\nmin 0.100000 0.000000 \\\\n25% 0.300000 0.000000 \\\\n50% 1.300000 1.000000 \\\\n75% 1.800000 2.000000 \\\\nmax 2.500000 2.000000 \\\\n\\\\nFirst Five Rows:\\\\n sepal length (cm) sepal width (cm) petal length (cm) petal width (cm) \\\\\\\\\\\\n0 5.1 3.5 1.4 0.2 \\\\n1 4.9 3.0 1.4 0.2 \\\\n2 4.7 3.2 1.3 0.2 \\\\n3 4.6 3.1 1.5 0.2 \\\\n4 5.0 3.6 1.4 0.2 \\\\n\\\\n target \\\\n0 0 \\\\n1 0 \\\\n2 0 \\\\n3 0 \\\\n4 0 \\\\n\\\\nClass Distribution:\\\\n target\\\\n0 50\\\\n1 50\\\\n2 50\\\\nName: count, dtype: int64\\\\n\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset features.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "import matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Set the style of seaborn\nsns.set(style=\"whitegrid\")\n\n# Create a pairplot to visualize the relationships between features\npair_plot = sns.pairplot(iris_df, hue='target', markers=['o', 's', 'D'], palette='husl')\n\n# Show the plot\nplt.show()" + }, + "\n # Context:\n user: run analysis on sklearn iris dataset\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 5 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Import the Iris dataset from scikit-learn.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis to understand the dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Preprocess the data if necessary (e.g., scaling, encoding).\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Split the dataset into training and testing sets.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Choose a suitable model and train it on the dataset.\"\n },\n {\n \"task_id\": \"6\",\n \"dependent_task_ids\": [\"5\"],\n \"instruction\": \"Evaluate the model's performance on the test set.\"\n },\n {\n \"task_id\": \"7\",\n \"dependent_task_ids\": [\"6\"],\n \"instruction\": \"Report the results of the analysis.\"\n }\n]\n```" } \ No newline at end of file diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index e64b4a551..3e20a8bfb 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -3,11 +3,8 @@ import asyncio import pytest from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_analysis_code import ( - WriteCodeByGenerate, - WriteCodeWithTools, - WriteCodeWithToolsML, -) +from metagpt.actions.ml_action import WriteCodeWithToolsML +from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.logs import logger from metagpt.plan.planner import STRUCTURAL_CONTEXT from metagpt.schema import Message, Plan, Task diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py index 379194534..1c5b2873f 100644 --- a/tests/metagpt/roles/run_code_interpreter.py +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -9,7 +9,7 @@ from metagpt.schema import Plan from metagpt.utils.recovery_util import load_history, save_history -async def run_code_interpreter(role_class, requirement, auto_run, use_tools, use_code_steps, save_dir, tools): +async def run_code_interpreter(role_class, requirement, auto_run, use_tools, save_dir, tools): """ The main function to run the MLEngineer with optional history loading. @@ -28,7 +28,6 @@ async def run_code_interpreter(role_class, requirement, auto_run, use_tools, use role = MLEngineer( auto_run=auto_run, use_tools=use_tools, - use_code_steps=use_code_steps, tools=tools, ) @@ -75,10 +74,9 @@ if __name__ == "__main__": requirement: str = requirement, auto_run: bool = auto_run, use_tools: bool = use_tools, - use_code_steps: bool = False, save_dir: str = save_dir, tools=tools, ): - await run_code_interpreter(role_class, requirement, auto_run, use_tools, use_code_steps, save_dir, tools) + await run_code_interpreter(role_class, requirement, auto_run, use_tools, save_dir, tools) fire.Fire(main) diff --git a/tests/metagpt/roles/test_code_interpreter.py b/tests/metagpt/roles/test_code_interpreter.py index 8595b9b15..aeb7070fd 100644 --- a/tests/metagpt/roles/test_code_interpreter.py +++ b/tests/metagpt/roles/test_code_interpreter.py @@ -3,11 +3,24 @@ import pytest from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter +# from metagpt.const import DATA_PATH + @pytest.mark.asyncio -async def test_code_interpreter(): +@pytest.mark.parametrize("use_tools", [(True)]) +async def test_code_interpreter(use_tools): requirement = "Run data analysis on sklearn Iris dataset, include a plot" - ci = CodeInterpreter(goal=requirement, auto_run=True, use_tools=False) + # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" + # data_path = f"{DATA_PATH}/titanic" + # requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" + # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." + # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" + # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + tools = [] + # tools = ["FillMissingValue", "CatCross", "a"] + + ci = CodeInterpreter(auto_run=True, use_tools=use_tools, tools=tools) rsp = await ci.run(requirement) logger.info(rsp) assert len(rsp.content) > 0 diff --git a/tests/metagpt/roles/test_daml.py b/tests/metagpt/roles/test_daml.py deleted file mode 100644 index 2e2c003d9..000000000 --- a/tests/metagpt/roles/test_daml.py +++ /dev/null @@ -1,50 +0,0 @@ -import pytest -from tqdm import tqdm - -from metagpt.logs import logger -from metagpt.roles.ml_engineer import ExecutePyCode, MLEngineer -from metagpt.schema import Plan - - -def reset(role): - """Restart role with the same goal.""" - role.working_memory.clear() - role.planner.plan = Plan(goal=role.planner.plan.goal) - role.execute_code = ExecutePyCode() - - -async def make_use_tools(requirement: str, auto_run: bool = True): - """make and use tools for requirement.""" - role = MLEngineer(goal=requirement, auto_run=auto_run) - # make udfs - role.use_tools = False - role.use_code_steps = False - role.make_udfs = True - role.use_udfs = False - await role.run(requirement) - # use udfs - reset(role) - role.make_udfs = False - role.use_udfs = True - role.use_code_steps = False - role.use_tools = False - await role.run(requirement) - - -@pytest.mark.asyncio -async def test_make_use_tools(): - requirements = [ - "Run data analysis on sklearn Iris dataset, include a plot", - "Run data analysis on sklearn Diabetes dataset, include a plot", - "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy", - "Run data analysis on sklearn Wisconsin Breast Cancer dataset, include a plot, train a model to predict targets (20% as validation), and show validation accuracy", - "Run EDA and visualization on this dataset, train a model to predict survival, report metrics on validation set (20%), dataset: tests/data/titanic.csv", - ] - success = 0 - for requirement in tqdm(requirements, total=len(requirements)): - try: - await make_use_tools(requirement) - success += 1 - except Exception as e: - logger.error(f"Found Error in {requirement}, {e}") - logger.info(f"success: {round(success/len(requirements), 1)*100}%") diff --git a/tests/metagpt/roles/test_ml_engineer.py b/tests/metagpt/roles/test_ml_engineer.py new file mode 100644 index 000000000..23570b0f1 --- /dev/null +++ b/tests/metagpt/roles/test_ml_engineer.py @@ -0,0 +1,31 @@ +import pytest + +from metagpt.const import DATA_PATH +from metagpt.logs import logger +from metagpt.roles.ml_engineer import MLEngineer + + +def test_mle_init(): + ci = MLEngineer(goal="test", auto_run=True, use_tools=True, tools=["tool1", "tool2"]) + assert ci.tools == [] + + +@pytest.mark.asyncio +@pytest.mark.parametrize("use_tools", [(True)]) +async def test_code_interpreter(use_tools): + # requirement = "Run data analysis on sklearn Iris dataset, include a plot" + # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" + data_path = f"{DATA_PATH}/titanic" + requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" + # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." + # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" + # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report AUC Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." + # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" + # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." + tools = ["FillMissingValue", "CatCross", "dummy_tool"] + + mle = MLEngineer(goal=requirement, auto_run=True, use_tools=use_tools, tools=tools) + rsp = await mle.run(requirement) + logger.info(rsp) + assert len(rsp.content) > 0 diff --git a/tests/metagpt/tools/libs/test_udf.py b/tests/metagpt/tools/libs/test_udf.py deleted file mode 100644 index 19e523448..000000000 --- a/tests/metagpt/tools/libs/test_udf.py +++ /dev/null @@ -1,49 +0,0 @@ -import json - -import yaml - -from metagpt.logs import logger -from metagpt.tools.libs.udf import UDFS, UDFS_YAML, docstring_to_yaml - - -def test_udfs(): - assert len(UDFS) > 0 - assert "udf_name" in UDFS[0] - assert "udf_doc" in UDFS[0] - logger.info(UDFS) - - -def test_docstring2yaml(): - docstring = """Calculate the duration in hours between two datetime columns. - - Args: - dataframe (pd.DataFrame): The dataframe containing the datetime columns. - - Returns: - pd.DataFrame: The dataframe with an additional column 'duration_hour' added. - """ - - yaml_result = docstring_to_yaml(docstring, return_vars="dataframe") - assert "parameters" in yaml_result - assert "properties" in yaml_result["parameters"] - assert "dataframe" in yaml_result["parameters"]["properties"] - - -def test_UDFS_YAML(): - assert len(UDFS_YAML) > 0 - logger.info(f"\n\n{json.dumps(UDFS_YAML, indent=2, ensure_ascii=False)}") - function_schema = UDFS_YAML - assert "description" in function_schema[list(function_schema.keys())[0]] - assert "type" in function_schema[list(function_schema.keys())[0]] - assert "parameters" in function_schema[list(function_schema.keys())[0]] - assert "properties" in function_schema[list(function_schema.keys())[0]]["parameters"] - assert "required" in function_schema[list(function_schema.keys())[0]]["parameters"] - assert "returns" in function_schema[list(function_schema.keys())[0]] - # 指定要保存的文件路径 - file_path = "./tests/data/function_schema.yaml" - - # 使用 PyYAML 将字典保存为 YAML 文件 - with open(file_path, "w") as file: - yaml.dump(function_schema, file, default_flow_style=False) - - print(f"Data has been saved to {file_path}") diff --git a/tests/metagpt/utils/test_save_code.py b/tests/metagpt/utils/test_save_code.py index 278d9a539..0674315d0 100644 --- a/tests/metagpt/utils/test_save_code.py +++ b/tests/metagpt/utils/test_save_code.py @@ -9,7 +9,6 @@ import nbformat import pytest from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_analysis_code import WriteCodeByGenerate from metagpt.utils.save_code import DATA_PATH, save_code_file @@ -17,11 +16,6 @@ def test_save_code_file_python(): save_code_file("example", "print('Hello, World!')") file_path = DATA_PATH / "output" / "example" / "code.py" assert os.path.exists(file_path), f"File does not exist: {file_path}" - - -def test_save_code_file_python(): - save_code_file("example", "print('Hello, World!')") - file_path = DATA_PATH / "output" / "example" / "code.py" with open(file_path, "r", encoding="utf-8") as fp: content = fp.read() assert "print('Hello, World!')" in content, "File content does not match" @@ -38,7 +32,7 @@ def test_save_code_file_json(): @pytest.mark.asyncio async def test_save_code_file_notebook(): - code = await WriteCodeByGenerate().run(context="basic python, hello world", plan="", code_steps="", temperature=0.0) + code = "print('Hello, World!')" executor = ExecutePyCode() await executor.run(code) # Save as a Notebook file From ede04f20f6a7392e073b1c0c6bed80ddc47988d1 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 30 Jan 2024 22:04:00 +0800 Subject: [PATCH 514/637] fix test_write_analysis_code --- metagpt/actions/write_analysis_code.py | 4 +- tests/metagpt/actions/test_ml_action.py | 46 +++++++++++++++++++ .../actions/test_write_analysis_code.py | 36 +++++++-------- 3 files changed, 64 insertions(+), 22 deletions(-) create mode 100644 tests/metagpt/actions/test_ml_action.py diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 402f56ccc..5cea9fe51 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -77,8 +77,8 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): ) -> dict: # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) prompt = self.process_msg(context, system_msg) - code_content = await self.llm.aask_code(prompt, **kwargs) - return code_content + rsp = await self.llm.aask_code(prompt, **kwargs) + return rsp class WriteCodeWithTools(BaseWriteAnalysisCode): diff --git a/tests/metagpt/actions/test_ml_action.py b/tests/metagpt/actions/test_ml_action.py new file mode 100644 index 000000000..2c8d34da8 --- /dev/null +++ b/tests/metagpt/actions/test_ml_action.py @@ -0,0 +1,46 @@ +import pytest + +from metagpt.actions.ml_action import WriteCodeWithToolsML +from metagpt.schema import Plan, Task + + +@pytest.mark.asyncio +async def test_write_code_with_tools(): + write_code_ml = WriteCodeWithToolsML() + + task_map = { + "1": Task( + task_id="1", + instruction="随机生成一个pandas DataFrame数据集", + task_type="other", + dependent_task_ids=[], + code=""" + import pandas as pd + df = pd.DataFrame({ + 'a': [1, 2, 3, 4, 5], + 'b': [1.1, 2.2, 3.3, 4.4, np.nan], + 'c': ['aa', 'bb', 'cc', 'dd', 'ee'], + 'd': [1, 2, 3, 4, 5] + }) + """, + is_finished=True, + ), + "2": Task( + task_id="2", + instruction="对数据集进行数据清洗", + task_type="data_preprocess", + dependent_task_ids=["1"], + ), + } + plan = Plan( + goal="构造数据集并进行数据清洗", + tasks=list(task_map.values()), + task_map=task_map, + current_task_id="2", + ) + column_info = "" + + _, code_with_ml = await write_code_ml.run([], plan, column_info) + code_with_ml = code_with_ml["code"] + assert len(code_with_ml) > 0 + print(code_with_ml) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index 3e20a8bfb..43f23848d 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -3,13 +3,13 @@ import asyncio import pytest from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.ml_action import WriteCodeWithToolsML from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.logs import logger from metagpt.plan.planner import STRUCTURAL_CONTEXT from metagpt.schema import Message, Plan, Task +@pytest.mark.skip @pytest.mark.asyncio async def test_write_code_by_list_plan(): write_code = WriteCodeByGenerate() @@ -20,35 +20,31 @@ async def test_write_code_by_list_plan(): print(f"\n任务: {task}\n\n") messages.append(Message(task, role="assistant")) code = await write_code.run(messages) - messages.append(Message(code, role="assistant")) + messages.append(Message(code["code"], role="assistant")) assert len(code) > 0 - output = await execute_code.run(code) + output = await execute_code.run(code["code"]) print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") messages.append(output[0]) @pytest.mark.asyncio async def test_tool_recommendation(): - task = "对已经读取的数据集进行数据清洗" - code_steps = """ - step 1: 对数据集进行去重 - step 2: 对数据集进行缺失值处理 - """ + task = "clean and preprocess the data" + code_steps = "" available_tools = { - "fill_missing_value": "Completing missing values with simple strategies", - "split_bins": "Bin continuous data into intervals and return the bin identifier encoded as an integer value", + "FillMissingValue": "Filling missing values", + "SplitBins": "Bin continuous data into intervals and return the bin identifier encoded as an integer value", } write_code = WriteCodeWithTools() - tools = await write_code._tool_recommendation(task, code_steps, available_tools) + tools = await write_code._recommend_tool(task, code_steps, available_tools) assert len(tools) == 1 - assert tools[0] == "fill_missing_value" + assert "FillMissingValue" in tools @pytest.mark.asyncio async def test_write_code_with_tools(): write_code = WriteCodeWithTools() - write_code_ml = WriteCodeWithToolsML() requirement = "构造数据集并进行数据清洗" task_map = { @@ -81,7 +77,6 @@ async def test_write_code_with_tools(): task_map=task_map, current_task_id="2", ) - column_info = "" context = STRUCTURAL_CONTEXT.format( user_requirement=requirement, @@ -92,13 +87,10 @@ async def test_write_code_with_tools(): context_msg = [Message(content=context, role="user")] code = await write_code.run(context_msg, plan) + code = code["code"] assert len(code) > 0 print(code) - code_with_ml = await write_code_ml.run([], plan, column_info) - assert len(code_with_ml) > 0 - print(code_with_ml) - @pytest.mark.asyncio async def test_write_code_to_correct_error(): @@ -147,6 +139,7 @@ async def test_write_code_to_correct_error(): Message(content=error, role="user"), ] new_code = await WriteCodeByGenerate().run(context=context) + new_code = new_code["code"] print(new_code) assert "read_csv" in new_code # should correct read_excel to read_csv @@ -186,10 +179,12 @@ async def test_write_code_reuse_code_simple(): Message(content=structural_context, role="user"), ] code = await WriteCodeByGenerate().run(context=context) + code = code["code"] print(code) assert "pandas" not in code and "read_csv" not in code # should reuse import and read statement from previous one +@pytest.mark.skip @pytest.mark.asyncio async def test_write_code_reuse_code_long(): """test code reuse for long context""" @@ -242,13 +237,14 @@ async def test_write_code_reuse_code_long(): trial_results = await asyncio.gather(*trials) print(*trial_results, sep="\n\n***\n\n") success = [ - "load_iris" not in result and "iris_data" in result for result in trial_results + "load_iris" not in result["code"] and "iris_data" in result["code"] for result in trial_results ] # should reuse iris_data from previous tasks success_rate = sum(success) / trials_num logger.info(f"success rate: {success_rate :.2f}") assert success_rate >= 0.8 +@pytest.mark.skip @pytest.mark.asyncio async def test_write_code_reuse_code_long_for_wine(): """test code reuse for long context""" @@ -315,7 +311,7 @@ async def test_write_code_reuse_code_long_for_wine(): trial_results = await asyncio.gather(*trials) print(*trial_results, sep="\n\n***\n\n") success = [ - "load_wine" not in result and "wine_data" in result for result in trial_results + "load_wine" not in result["code"] and "wine_data" in result["code"] for result in trial_results ] # should reuse iris_data from previous tasks success_rate = sum(success) / trials_num logger.info(f"success rate: {success_rate :.2f}") From 274747e72fb3587ad5a20e7823a2c205a54af3b4 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 30 Jan 2024 22:20:34 +0800 Subject: [PATCH 515/637] fix test_debug_code --- metagpt/tools/tool_registry.py | 9 ++++--- tests/data/rsp_cache.json | 34 +++++++++++++++++++++++- tests/metagpt/actions/test_debug_code.py | 2 +- 3 files changed, 40 insertions(+), 5 deletions(-) diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index d16defa0a..7e4ee5ead 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -24,9 +24,10 @@ class ToolRegistry(BaseModel): tool_types: dict = {} tools_by_types: dict = defaultdict(dict) # two-layer k-v, {tool_type: {tool_name: {...}, ...}, ...} - def register_tool_type(self, tool_type: ToolType): + def register_tool_type(self, tool_type: ToolType, verbose: bool = False): self.tool_types[tool_type.name] = tool_type - logger.info(f"tool type {tool_type.name} registered") + if verbose: + logger.info(f"tool type {tool_type.name} registered") def register_tool( self, @@ -38,6 +39,7 @@ class ToolRegistry(BaseModel): tool_source_object=None, include_functions=[], make_schema_if_not_exists=True, + verbose=False, ): if self.has_tool(tool_name): return @@ -68,7 +70,8 @@ class ToolRegistry(BaseModel): tool = Tool(name=tool_name, path=tool_path, schemas=schemas, code=tool_code) self.tools[tool_name] = tool self.tools_by_types[tool_type][tool_name] = tool - logger.info(f"{tool_name} registered") + if verbose: + logger.info(f"{tool_name} registered") def has_tool(self, key: str) -> Tool: return key in self.tools diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index cb3c1642c..31eb7ebc0 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -207,5 +207,37 @@ "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\n\\\\n# Convert the loaded Iris dataset into a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\\\\niris_df['target'] = iris_data.target\\\\n\\\\n# Display basic statistical details like percentile, mean, std etc. of a data frame\\\\niris_stats = iris_df.describe()\\\\n\\\\n# Display the first few rows of the DataFrame\\\\niris_head = iris_df.head()\\\\n\\\\n# Display the class distribution\\\\niris_target_counts = iris_df['target'].value_counts()\\\\n\\\\n# Output the results\\\\nprint('Basic Statistical Details:\\\\\\\\n', iris_stats)\\\\nprint('\\\\\\\\nFirst Five Rows:\\\\\\\\n', iris_head)\\\\nprint('\\\\\\\\nClass Distribution:\\\\\\\\n', iris_target_counts)\\\",\\n \\\"result\\\": \\\"Basic Statistical Details:\\\\n sepal length (cm) sepal width (cm) petal length (cm) \\\\\\\\\\\\ncount 150.000000 150.000000 150.000000 \\\\nmean 5.843333 3.057333 3.758000 \\\\nstd 0.828066 0.435866 1.765298 \\\\nmin 4.300000 2.000000 1.000000 \\\\n25% 5.100000 2.800000 1.600000 \\\\n50% 5.800000 3.000000 4.350000 \\\\n75% 6.400000 3.300000 5.100000 \\\\nmax 7.900000 4.400000 6.900000 \\\\n\\\\n petal width (cm) target \\\\ncount 150.000000 150.000000 \\\\nmean 1.199333 1.000000 \\\\nstd 0.762238 0.819232 \\\\nmin 0.100000 0.000000 \\\\n25% 0.300000 0.000000 \\\\n50% 1.300000 1.000000 \\\\n75% 1.800000 2.000000 \\\\nmax 2.500000 2.000000 \\\\n\\\\nFirst Five Rows:\\\\n sepal length (cm) sepal width (cm) petal length (cm) petal width (cm) \\\\\\\\\\\\n0 5.1 3.5 1.4 0.2 \\\\n1 4.9 3.0 1.4 0.2 \\\\n2 4.7 3.2 1.3 0.2 \\\\n3 4.6 3.1 1.5 0.2 \\\\n4 5.0 3.6 1.4 0.2 \\\\n\\\\n target \\\\n0 0 \\\\n1 0 \\\\n2 0 \\\\n3 0 \\\\n4 0 \\\\n\\\\nClass Distribution:\\\\n target\\\\n0 50\\\\n1 50\\\\n2 50\\\\nName: count, dtype: int64\\\\n\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset features.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { "code": "import matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Set the style of seaborn\nsns.set(style=\"whitegrid\")\n\n# Create a pairplot to visualize the relationships between features\npair_plot = sns.pairplot(iris_df, hue='target', markers=['o', 's', 'D'], palette='husl')\n\n# Show the plot\nplt.show()" }, - "\n # Context:\n user: run analysis on sklearn iris dataset\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 5 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Import the Iris dataset from scikit-learn.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis to understand the dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Preprocess the data if necessary (e.g., scaling, encoding).\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Split the dataset into training and testing sets.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Choose a suitable model and train it on the dataset.\"\n },\n {\n \"task_id\": \"6\",\n \"dependent_task_ids\": [\"5\"],\n \"instruction\": \"Evaluate the model's performance on the test set.\"\n },\n {\n \"task_id\": \"7\",\n \"dependent_task_ids\": [\"6\"],\n \"instruction\": \"Report the results of the analysis.\"\n }\n]\n```" + "\n # Context:\n user: run analysis on sklearn iris dataset\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 5 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Import the Iris dataset from scikit-learn.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis to understand the dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Preprocess the data if necessary (e.g., scaling, encoding).\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Split the dataset into training and testing sets.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Choose a suitable model and train it on the dataset.\"\n },\n {\n \"task_id\": \"6\",\n \"dependent_task_ids\": [\"5\"],\n \"instruction\": \"Evaluate the model's performance on the test set.\"\n },\n {\n \"task_id\": \"7\",\n \"dependent_task_ids\": [\"6\"],\n \"instruction\": \"Report the results of the analysis.\"\n }\n]\n```", + "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\n对数据集进行数据清洗\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Completing missing values with simple strategies', 'MinMaxScale': 'Transform features by scaling each feature to a range, witch is (0, 1)', 'StandardScale': 'Standardize features by removing the mean and scaling to unit variance', 'MaxAbsScale': 'cale each feature by its maximum absolute value', 'RobustScale': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'OrdinalEncode': 'Encode categorical features as ordinal integers.', 'OneHotEncode': 'Apply one-hot encoding to specified categorical columns, the original columns will be dropped.', 'LabelEncode': 'Apply label encoding to specified categorical columns in-place.'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { + "recommend_tools": [ + "FillMissingValue", + "MinMaxScale", + "StandardScale", + "RobustScale", + "OneHotEncode" + ] + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}, 'strategy': {'type': 'str', 'description': 'the imputation strategy, notice mean/median can only be used for numeric features', 'default': 'mean', 'enum': ['mean', 'median', 'most_frequent', 'constant']}, 'fill_value': {'type': 'int', 'description': 'fill_value is used to replace all occurrences of missing_values', 'default': None}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, witch is (0, 1)', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MinMaxScale model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the StandardScale model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize the RobustScale instance with feature names. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'List of feature names to be scaled.'}}, 'required': ['features']}}, 'fit': {'description': 'Compute the median and IQR for scaling. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Scale features using the previously computed median and IQR. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features to be scaled.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'A new dataframe with scaled features.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'OneHotEncode': {'type': 'class', 'description': 'Apply one-hot encoding to specified categorical columns, the original columns will be dropped.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Categorical columns to be one-hot encoded and dropped'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the OneHotEncoding model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { + "code": "# Step 1: Fill missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\ndf_processed = df.copy()\n\n# Identify numeric columns with missing values\nnumeric_cols_with_na = df_processed.select_dtypes(include=['number']).columns[df_processed.isna().any()].tolist()\n\n# Fill missing values with the mean for numeric columns\nfill_missing_value = FillMissingValue(features=numeric_cols_with_na, strategy='mean')\ndf_processed = fill_missing_value.fit_transform(df_processed)\n\n# Step 2: Scale numeric columns\nfrom metagpt.tools.libs.data_preprocess import MinMaxScale\n\n# Identify numeric columns\nnumeric_cols = df_processed.select_dtypes(include=['number']).columns.tolist()\n\n# Scale numeric columns\nmin_max_scaler = MinMaxScale(features=numeric_cols)\ndf_processed = min_max_scaler.fit_transform(df_processed)\n\n# Note: Since there is no separate test set provided, the scaling is only applied to the existing dataset." + }, + "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nclean and preprocess the data\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Filling missing values', 'SplitBins': 'Bin continuous data into intervals and return the bin identifier encoded as an integer value'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { + "recommend_tools": [ + "FillMissingValue" + ] + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code_steps='', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code_steps='', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}, 'strategy': {'type': 'str', 'description': 'the imputation strategy, notice mean/median can only be used for numeric features', 'default': 'mean', 'enum': ['mean', 'median', 'most_frequent', 'constant']}, 'fill_value': {'type': 'int', 'description': 'fill_value is used to replace all occurrences of missing_values', 'default': None}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, witch is (0, 1)', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MinMaxScale model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the StandardScale model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize the RobustScale instance with feature names. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'List of feature names to be scaled.'}}, 'required': ['features']}}, 'fit': {'description': 'Compute the median and IQR for scaling. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Scale features using the previously computed median and IQR. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features to be scaled.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'A new dataframe with scaled features.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'OneHotEncode': {'type': 'class', 'description': 'Apply one-hot encoding to specified categorical columns, the original columns will be dropped.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Categorical columns to be one-hot encoded and dropped'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the OneHotEncoding model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Instantiate the FillMissingValue class for numeric columns with missing values\ncolumns_with_missing_values = ['b']\nfill_missing_value = FillMissingValue(features=columns_with_missing_values, strategy='mean')\n\n# Apply the fit_transform method to the DataFrame to fill missing values\ndf = fill_missing_value.fit_transform(df)\n" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n ## User Requirement\\n read a dataset test.csv and print its head\\n ## Current Plan\\n [\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n }\\n ]\\n ## Current Task\\n {\\\"task_id\\\": \\\"1\\\", \\\"dependent_task_ids\\\": [], \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\", \\\"task_type\\\": \\\"\\\", \\\"code\\\": \\\"\\\", \\\"result\\\": \\\"\\\", \\\"is_finished\\\": false}\\n \"}, {\"role\": \"assistant\", \"content\": \"import pandas as pd\\ndata = pd.read_excel('test.csv')\\ndata\"}, {\"role\": \"user\", \"content\": \"\\n Traceback (most recent call last):\\n File \\\"\\\", line 2, in \\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 478, in read_excel\\n io = ExcelFile(io, storage_options=storage_options, engine=engine)\\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 1500, in __init__\\n raise ValueError(\\n ValueError: Excel file format cannot be determined, you must specify an engine manually.\\n \"}]": { + "language": "python", + "code": "import pandas as pd\ndata = pd.read_csv('test.csv')" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n ## User Requirement\\n read a dataset test.csv and print its head\\n ## Current Plan\\n [\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"import pandas as pd\\ndata = pd.read_csv('test.csv')\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n }\\n ]\\n ## Current Task\\n {\\\"task_id\\\": \\\"2\\\", \\\"dependent_task_ids\\\": [\\\"1\\\"], \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\", \\\"task_type\\\": \\\"\\\", \\\"code\\\": \\\"\\\", \\\"result\\\": \\\"\\\", \\\"is_finished\\\": false}\\n \"}]": { + "language": "python", + "code": "print(data.head())" + }, + "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nuser: Solve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\nTested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { + "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not handle negative numbers correctly, as the binary representation of negative numbers in Python includes a minus sign, which is not considered in the count_ones function. To fix this issue, we should modify the count_ones function to handle negative numbers appropriately by ensuring that it only counts the '1's in the binary representation of the absolute value of the number. This will ensure that the function returns the correct output for both positive and negative numbers.", + "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(abs(n)).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\n \n return sorted_arr" + } } \ No newline at end of file diff --git a/tests/metagpt/actions/test_debug_code.py b/tests/metagpt/actions/test_debug_code.py index 262f2e60d..83ce75761 100644 --- a/tests/metagpt/actions/test_debug_code.py +++ b/tests/metagpt/actions/test_debug_code.py @@ -48,7 +48,7 @@ def sort_array(arr): async def test_debug_code(): debug_context = Message(content=DebugContext) new_code = await DebugCode().run(context=debug_context, code=CODE, runtime_result=ErrorStr) - assert "def sort_array(arr)" in new_code + assert "def sort_array(arr)" in new_code["code"] def test_messages_to_str(): From 5dde5a8875bf906c0347db3c9870b1770a3e4e77 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 30 Jan 2024 22:41:30 +0800 Subject: [PATCH 516/637] rm unused & format --- examples/imitate_webpage.py | 4 +- tests/metagpt/actions/test_make_tools.py | 52 ------------------------ 2 files changed, 2 insertions(+), 54 deletions(-) delete mode 100644 tests/metagpt/actions/test_make_tools.py diff --git a/examples/imitate_webpage.py b/examples/imitate_webpage.py index 6c12c7eda..b69101861 100644 --- a/examples/imitate_webpage.py +++ b/examples/imitate_webpage.py @@ -9,7 +9,7 @@ from metagpt.roles.code_interpreter import CodeInterpreter async def main(): - web_url = 'https://pytorch.org/' + web_url = "https://pytorch.org/" prompt = f"""This is a URL of webpage: '{web_url}' . Firstly, utilize Selenium and WebDriver for rendering. Secondly, convert image to a webpage including HTML, CSS and JS in one go. @@ -20,7 +20,7 @@ Note: All required dependencies and environments have been fully installed and c await ci.run(prompt) -if __name__ == '__main__': +if __name__ == "__main__": import asyncio asyncio.run(main()) diff --git a/tests/metagpt/actions/test_make_tools.py b/tests/metagpt/actions/test_make_tools.py deleted file mode 100644 index 8e94c6eee..000000000 --- a/tests/metagpt/actions/test_make_tools.py +++ /dev/null @@ -1,52 +0,0 @@ -import pytest - -from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_analysis_code import MakeTools -from metagpt.logs import logger - - -@pytest.mark.asyncio -async def test_make_tools(): - code = "import yfinance as yf\n\n# Collect Alibaba stock data\nalibaba = yf.Ticker('BABA')\ndata = alibaba.history(period='1d', start='2022-01-01', end='2022-12-31')\nprint(data.head())" - msgs = [{"role": "assistant", "content": code}] - mt = MakeTools() - tool_code = await mt.run(msgs) - logger.debug(tool_code) - ep = ExecutePyCode() - tool_code = "!pip install yfinance\n" + tool_code - result, res_type = await ep.run(tool_code) - assert res_type is True - logger.debug(result) - - -@pytest.mark.asyncio -async def test_make_tools2(): - code = """import pandas as pd\npath = "./tests/data/test.csv"\ndf = pd.read_csv(path)\ndata = df.copy()\n - data['started_at'] = data['started_at'].apply(lambda r: pd.to_datetime(r))\n - data['ended_at'] = data['ended_at'].apply(lambda r: pd.to_datetime(r))\ndata.head()""" - msgs = [{"role": "assistant", "content": code}] - mt = MakeTools() - tool_code = await mt.run(msgs) - logger.debug(tool_code) - ep = ExecutePyCode() - tool_code = tool_code - result, res_type = await ep.run(tool_code) - assert res_type is True - logger.debug(result) - - -@pytest.mark.asyncio -async def test_make_tools3(): - code = """import pandas as pd\npath = "./tests/data/test.csv"\ndf = pd.read_csv(path)\ndata = df.copy()\n - data['started_at'] = data['started_at'].apply(lambda r: pd.to_datetime(r))\n - data['ended_at'] = data['ended_at'].apply(lambda r: pd.to_datetime(r))\n - data['duration_hour'] = (data['ended_at'] - data['started_at']).dt.seconds/3600\ndata.head()""" - msgs = [{"role": "assistant", "content": code}] - mt = MakeTools() - tool_code = await mt.run(msgs) - logger.debug(tool_code) - ep = ExecutePyCode() - tool_code = tool_code - result, res_type = await ep.run(tool_code) - assert res_type is True - logger.debug(result) From f9519ca417f9ae72c8814c70d44de35bcc1be587 Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 31 Jan 2024 00:39:57 +0800 Subject: [PATCH 517/637] change ways of using config --- metagpt/tools/libs/gpt_v_generator.py | 10 +++++----- metagpt/tools/libs/sd_engine.py | 8 ++++---- metagpt/tools/web_browser_engine_playwright.py | 5 ++++- tests/mock/mock_llm.py | 3 ++- 4 files changed, 15 insertions(+), 11 deletions(-) diff --git a/metagpt/tools/libs/gpt_v_generator.py b/metagpt/tools/libs/gpt_v_generator.py index adc3b1051..e079a8eef 100644 --- a/metagpt/tools/libs/gpt_v_generator.py +++ b/metagpt/tools/libs/gpt_v_generator.py @@ -33,12 +33,12 @@ Now, please generate the corresponding webpage code including HTML, CSS and Java @register_tool(tool_type=ToolTypeEnum.IMAGE2WEBPAGE.value) class GPTvGenerator: def __init__(self): - from metagpt.config import CONFIG + from metagpt.config2 import config - OPENAI_API_BASE = CONFIG.OPENAI_BASE_URL - API_KEY = CONFIG.OPENAI_API_KEY - MODEL = CONFIG.OPENAI_VISION_MODEL - MAX_TOKENS = CONFIG.VISION_MAX_TOKENS + OPENAI_API_BASE = config.llm.base_url + API_KEY = config.llm.api_key + MODEL = config.OPENAI_VISION_MODEL + MAX_TOKENS = config.VISION_MAX_TOKENS self.api_key = API_KEY self.api_base = OPENAI_API_BASE self.model = MODEL diff --git a/metagpt/tools/libs/sd_engine.py b/metagpt/tools/libs/sd_engine.py index 794758f77..47b0da7e9 100644 --- a/metagpt/tools/libs/sd_engine.py +++ b/metagpt/tools/libs/sd_engine.py @@ -55,11 +55,11 @@ default_negative_prompt = "(easynegative:0.8),black, dark,Low resolution" @register_tool(tool_type=ToolTypeEnum.STABLE_DIFFUSION.value) class SDEngine: def __init__(self, sd_url=""): - from metagpt.config import CONFIG + from metagpt.config2 import config # Initialize the SDEngine with configuration - self.sd_url = sd_url if sd_url else CONFIG.get("SD_URL") - self.sd_t2i_url = f"{self.sd_url}{CONFIG.get('SD_T2I_API')}" + self.sd_url = sd_url if sd_url else config.get("SD_URL") + self.sd_t2i_url = f"{self.sd_url}{config.get('SD_T2I_API')}" # Define default payload settings for SD API self.payload = payload logger.info(self.sd_t2i_url) @@ -82,7 +82,7 @@ class SDEngine: return self.payload def save(self, imgs, save_name=""): - save_dir = CONFIG.workspace_path / SD_OUTPUT_FILE_REPO + save_dir = config.workspace_path / SD_OUTPUT_FILE_REPO if not save_dir.exists(): save_dir.mkdir(parents=True, exist_ok=True) batch_decode_base64_to_image(imgs, str(save_dir), save_name=save_name) diff --git a/metagpt/tools/web_browser_engine_playwright.py b/metagpt/tools/web_browser_engine_playwright.py index f8dabd5ac..7c33da923 100644 --- a/metagpt/tools/web_browser_engine_playwright.py +++ b/metagpt/tools/web_browser_engine_playwright.py @@ -10,7 +10,6 @@ from typing import Literal from playwright.async_api import async_playwright -from metagpt.config2 import config from metagpt.logs import logger from metagpt.utils.parse_html import WebPage @@ -30,6 +29,10 @@ class PlaywrightWrapper: launch_kwargs: dict | None = None, **kwargs, ) -> None: + from metagpt.config2 import ( + config, # avoid circular import error when importing tools" + ) + self.browser_type = browser_type launch_kwargs = launch_kwargs or {} if config.proxy and "proxy" not in launch_kwargs: diff --git a/tests/mock/mock_llm.py b/tests/mock/mock_llm.py index 3671e8fb7..e2fff214f 100644 --- a/tests/mock/mock_llm.py +++ b/tests/mock/mock_llm.py @@ -2,12 +2,13 @@ import json from typing import Optional, Union from metagpt.config2 import config +from metagpt.configs.llm_config import LLMType from metagpt.logs import log_llm_stream, logger from metagpt.provider.azure_openai_api import AzureOpenAILLM from metagpt.provider.openai_api import OpenAILLM from metagpt.schema import Message -OriginalLLM = OpenAILLM if not config.openai_api_type else AzureOpenAILLM +OriginalLLM = OpenAILLM if config.llm.api_type == LLMType.OPENAI else AzureOpenAILLM class MockLLM(OriginalLLM): From 56f5dc9f2e9dc174b9e7e9d2ecb5e68aff2e29bf Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 31 Jan 2024 10:49:49 +0800 Subject: [PATCH 518/637] fix planner serialization bug, add test data --- metagpt/plan/planner.py | 4 +- metagpt/roles/role.py | 2 +- tests/data/ml_datasets/titanic/split_eval.csv | 180 +++++ .../data/ml_datasets/titanic/split_train.csv | 713 ++++++++++++++++++ tests/data/rsp_cache.json | 24 +- tests/metagpt/roles/test_ml_engineer.py | 16 +- 6 files changed, 911 insertions(+), 28 deletions(-) create mode 100644 tests/data/ml_datasets/titanic/split_eval.csv create mode 100644 tests/data/ml_datasets/titanic/split_train.csv diff --git a/metagpt/plan/planner.py b/metagpt/plan/planner.py index fea5f0f8d..0d8870fd3 100644 --- a/metagpt/plan/planner.py +++ b/metagpt/plan/planner.py @@ -32,8 +32,8 @@ class Planner(BaseModel): auto_run: bool = False use_tools: bool = False - def __init__(self, goal: str, **kwargs): - plan = Plan(goal=goal) + def __init__(self, goal: str = "", plan: Plan = None, **kwargs): + plan = plan or Plan(goal=goal) super().__init__(plan=plan, **kwargs) @property diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 641d037ff..9efcf470e 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -144,7 +144,7 @@ class Role(SerializationMixin, ContextMixin, BaseModel): actions: list[SerializeAsAny[Action]] = Field(default=[], validate_default=True) rc: RoleContext = Field(default_factory=RoleContext) addresses: set[str] = set() - planner: Planner = None + planner: Planner = Field(default_factory=Planner) # builtin variables recovered: bool = False # to tag if a recovered role diff --git a/tests/data/ml_datasets/titanic/split_eval.csv b/tests/data/ml_datasets/titanic/split_eval.csv new file mode 100644 index 000000000..6da6ff6b3 --- /dev/null +++ b/tests/data/ml_datasets/titanic/split_eval.csv @@ -0,0 +1,180 @@ +PassengerId,Survived,Pclass,Name,Sex,Age,SibSp,Parch,Ticket,Fare,Cabin,Embarked +206,0,3,"Strom, Miss. Telma Matilda",female,2.0,0,1,347054,10.4625,G6,S +45,1,3,"Devaney, Miss. Margaret Delia",female,19.0,0,0,330958,7.8792,,Q +822,1,3,"Lulic, Mr. Nikola",male,27.0,0,0,315098,8.6625,,S +459,1,2,"Toomey, Miss. Ellen",female,50.0,0,0,F.C.C. 13531,10.5,,S +796,0,2,"Otter, Mr. Richard",male,39.0,0,0,28213,13.0,,S +119,0,1,"Baxter, Mr. Quigg Edmond",male,24.0,0,1,PC 17558,247.5208,B58 B60,C +425,0,3,"Rosblom, Mr. Viktor Richard",male,18.0,1,1,370129,20.2125,,S +679,0,3,"Goodwin, Mrs. Frederick (Augusta Tyler)",female,43.0,1,6,CA 2144,46.9,,S +270,1,1,"Bissette, Miss. Amelia",female,35.0,0,0,PC 17760,135.6333,C99,S +230,0,3,"Lefebre, Miss. Mathilde",female,,3,1,4133,25.4667,,S +690,1,1,"Madill, Miss. Georgette Alexandra",female,15.0,0,1,24160,211.3375,B5,S +321,0,3,"Dennis, Mr. Samuel",male,22.0,0,0,A/5 21172,7.25,,S +406,0,2,"Gale, Mr. Shadrach",male,34.0,1,0,28664,21.0,,S +41,0,3,"Ahlin, Mrs. Johan (Johanna Persdotter Larsson)",female,40.0,1,0,7546,9.475,,S +25,0,3,"Palsson, Miss. Torborg Danira",female,8.0,3,1,349909,21.075,,S +554,1,3,"Leeni, Mr. Fahim (""Philip Zenni"")",male,22.0,0,0,2620,7.225,,C +413,1,1,"Minahan, Miss. Daisy E",female,33.0,1,0,19928,90.0,C78,Q +513,1,1,"McGough, Mr. James Robert",male,36.0,0,0,PC 17473,26.2875,E25,S +756,1,2,"Hamalainen, Master. Viljo",male,0.67,1,1,250649,14.5,,S +392,1,3,"Jansson, Mr. Carl Olof",male,21.0,0,0,350034,7.7958,,S +602,0,3,"Slabenoff, Mr. Petco",male,,0,0,349214,7.8958,,S +326,1,1,"Young, Miss. Marie Grice",female,36.0,0,0,PC 17760,135.6333,C32,C +373,0,3,"Beavan, Mr. William Thomas",male,19.0,0,0,323951,8.05,,S +377,1,3,"Landergren, Miss. Aurora Adelia",female,22.0,0,0,C 7077,7.25,,S +201,0,3,"Vande Walle, Mr. Nestor Cyriel",male,28.0,0,0,345770,9.5,,S +512,0,3,"Webber, Mr. James",male,,0,0,SOTON/OQ 3101316,8.05,,S +601,1,2,"Jacobsohn, Mrs. Sidney Samuel (Amy Frances Christy)",female,24.0,2,1,243847,27.0,,S +631,1,1,"Barkworth, Mr. Algernon Henry Wilson",male,80.0,0,0,27042,30.0,A23,S +364,0,3,"Asim, Mr. Adola",male,35.0,0,0,SOTON/O.Q. 3101310,7.05,,S +144,0,3,"Burke, Mr. Jeremiah",male,19.0,0,0,365222,6.75,,Q +202,0,3,"Sage, Mr. Frederick",male,,8,2,CA. 2343,69.55,,S +134,1,2,"Weisz, Mrs. Leopold (Mathilde Francoise Pede)",female,29.0,1,0,228414,26.0,,S +431,1,1,"Bjornstrom-Steffansson, Mr. Mauritz Hakan",male,28.0,0,0,110564,26.55,C52,S +419,0,2,"Matthews, Mr. William John",male,30.0,0,0,28228,13.0,,S +782,1,1,"Dick, Mrs. Albert Adrian (Vera Gillespie)",female,17.0,1,0,17474,57.0,B20,S +705,0,3,"Hansen, Mr. Henrik Juul",male,26.0,1,0,350025,7.8542,,S +536,1,2,"Hart, Miss. Eva Miriam",female,7.0,0,2,F.C.C. 13529,26.25,,S +335,1,1,"Frauenthal, Mrs. Henry William (Clara Heinsheimer)",female,,1,0,PC 17611,133.65,,S +273,1,2,"Mellinger, Mrs. (Elizabeth Anne Maidment)",female,41.0,0,1,250644,19.5,,S +108,1,3,"Moss, Mr. Albert Johan",male,,0,0,312991,7.775,,S +403,0,3,"Jussila, Miss. Mari Aina",female,21.0,1,0,4137,9.825,,S +307,1,1,"Fleming, Miss. Margaret",female,,0,0,17421,110.8833,,C +218,0,2,"Jacobsohn, Mr. Sidney Samuel",male,42.0,1,0,243847,27.0,,S +789,1,3,"Dean, Master. Bertram Vere",male,1.0,1,2,C.A. 2315,20.575,,S +160,0,3,"Sage, Master. Thomas Henry",male,,8,2,CA. 2343,69.55,,S +20,1,3,"Masselmani, Mrs. Fatima",female,,0,0,2649,7.225,,C +174,0,3,"Sivola, Mr. Antti Wilhelm",male,21.0,0,0,STON/O 2. 3101280,7.925,,S +311,1,1,"Hays, Miss. Margaret Bechstein",female,24.0,0,0,11767,83.1583,C54,C +595,0,2,"Chapman, Mr. John Henry",male,37.0,1,0,SC/AH 29037,26.0,,S +592,1,1,"Stephenson, Mrs. Walter Bertram (Martha Eustis)",female,52.0,1,0,36947,78.2667,D20,C +164,0,3,"Calic, Mr. Jovo",male,17.0,0,0,315093,8.6625,,S +563,0,2,"Norman, Mr. Robert Douglas",male,28.0,0,0,218629,13.5,,S +172,0,3,"Rice, Master. Arthur",male,4.0,4,1,382652,29.125,,Q +871,0,3,"Balkic, Mr. Cerin",male,26.0,0,0,349248,7.8958,,S +176,0,3,"Klasen, Mr. Klas Albin",male,18.0,1,1,350404,7.8542,,S +434,0,3,"Kallio, Mr. Nikolai Erland",male,17.0,0,0,STON/O 2. 3101274,7.125,,S +462,0,3,"Morley, Mr. William",male,34.0,0,0,364506,8.05,,S +49,0,3,"Samaan, Mr. Youssef",male,,2,0,2662,21.6792,,C +126,1,3,"Nicola-Yarred, Master. Elias",male,12.0,1,0,2651,11.2417,,C +125,0,1,"White, Mr. Percival Wayland",male,54.0,0,1,35281,77.2875,D26,S +266,0,2,"Reeves, Mr. David",male,36.0,0,0,C.A. 17248,10.5,,S +550,1,2,"Davies, Master. John Morgan Jr",male,8.0,1,1,C.A. 33112,36.75,,S +589,0,3,"Gilinski, Mr. Eliezer",male,22.0,0,0,14973,8.05,,S +779,0,3,"Kilgannon, Mr. Thomas J",male,,0,0,36865,7.7375,,Q +179,0,2,"Hale, Mr. Reginald",male,30.0,0,0,250653,13.0,,S +107,1,3,"Salkjelsvik, Miss. Anna Kristine",female,21.0,0,0,343120,7.65,,S +624,0,3,"Hansen, Mr. Henry Damsgaard",male,21.0,0,0,350029,7.8542,,S +115,0,3,"Attalah, Miss. Malake",female,17.0,0,0,2627,14.4583,,C +42,0,2,"Turpin, Mrs. William John Robert (Dorothy Ann Wonnacott)",female,27.0,1,0,11668,21.0,,S +664,0,3,"Coleff, Mr. Peju",male,36.0,0,0,349210,7.4958,,S +661,1,1,"Frauenthal, Dr. Henry William",male,50.0,2,0,PC 17611,133.65,,S +762,0,3,"Nirva, Mr. Iisakki Antino Aijo",male,41.0,0,0,SOTON/O2 3101272,7.125,,S +580,1,3,"Jussila, Mr. Eiriik",male,32.0,0,0,STON/O 2. 3101286,7.925,,S +265,0,3,"Henry, Miss. Delia",female,,0,0,382649,7.75,,Q +757,0,3,"Carlsson, Mr. August Sigfrid",male,28.0,0,0,350042,7.7958,,S +666,0,2,"Hickman, Mr. Lewis",male,32.0,2,0,S.O.C. 14879,73.5,,S +634,0,1,"Parr, Mr. William Henry Marsh",male,,0,0,112052,0.0,,S +532,0,3,"Toufik, Mr. Nakli",male,,0,0,2641,7.2292,,C +640,0,3,"Thorneycroft, Mr. Percival",male,,1,0,376564,16.1,,S +599,0,3,"Boulos, Mr. Hanna",male,,0,0,2664,7.225,,C +220,0,2,"Harris, Mr. Walter",male,30.0,0,0,W/C 14208,10.5,,S +150,0,2,"Byles, Rev. Thomas Roussel Davids",male,42.0,0,0,244310,13.0,,S +269,1,1,"Graham, Mrs. William Thompson (Edith Junkins)",female,58.0,0,1,PC 17582,153.4625,C125,S +670,1,1,"Taylor, Mrs. Elmer Zebley (Juliet Cummins Wright)",female,,1,0,19996,52.0,C126,S +578,1,1,"Silvey, Mrs. William Baird (Alice Munger)",female,39.0,1,0,13507,55.9,E44,S +786,0,3,"Harmer, Mr. Abraham (David Lishin)",male,25.0,0,0,374887,7.25,,S +82,1,3,"Sheerlinck, Mr. Jan Baptist",male,29.0,0,0,345779,9.5,,S +400,1,2,"Trout, Mrs. William H (Jessie L)",female,28.0,0,0,240929,12.65,,S +135,0,2,"Sobey, Mr. Samuel James Hayden",male,25.0,0,0,C.A. 29178,13.0,,S +223,0,3,"Green, Mr. George Henry",male,51.0,0,0,21440,8.05,,S +693,1,3,"Lam, Mr. Ali",male,,0,0,1601,56.4958,,S +280,1,3,"Abbott, Mrs. Stanton (Rosa Hunt)",female,35.0,1,1,C.A. 2673,20.25,,S +102,0,3,"Petroff, Mr. Pastcho (""Pentcho"")",male,,0,0,349215,7.8958,,S +288,0,3,"Naidenoff, Mr. Penko",male,22.0,0,0,349206,7.8958,,S +711,1,1,"Mayne, Mlle. Berthe Antonine (""Mrs de Villiers"")",female,24.0,0,0,PC 17482,49.5042,C90,C +256,1,3,"Touma, Mrs. Darwis (Hanne Youssef Razi)",female,29.0,0,2,2650,15.2458,,C +23,1,3,"McGowan, Miss. Anna ""Annie""",female,15.0,0,0,330923,8.0292,,Q +582,1,1,"Thayer, Mrs. John Borland (Marian Longstreth Morris)",female,39.0,1,1,17421,110.8833,C68,C +564,0,3,"Simmons, Mr. John",male,,0,0,SOTON/OQ 392082,8.05,,S +405,0,3,"Oreskovic, Miss. Marija",female,20.0,0,0,315096,8.6625,,S +429,0,3,"Flynn, Mr. James",male,,0,0,364851,7.75,,Q +848,0,3,"Markoff, Mr. Marin",male,35.0,0,0,349213,7.8958,,C +726,0,3,"Oreskovic, Mr. Luka",male,20.0,0,0,315094,8.6625,,S +721,1,2,"Harper, Miss. Annie Jessie ""Nina""",female,6.0,0,1,248727,33.0,,S +637,0,3,"Leinonen, Mr. Antti Gustaf",male,32.0,0,0,STON/O 2. 3101292,7.925,,S +863,1,1,"Swift, Mrs. Frederick Joel (Margaret Welles Barron)",female,48.0,0,0,17466,25.9292,D17,S +615,0,3,"Brocklebank, Mr. William Alfred",male,35.0,0,0,364512,8.05,,S +199,1,3,"Madigan, Miss. Margaret ""Maggie""",female,,0,0,370370,7.75,,Q +787,1,3,"Sjoblom, Miss. Anna Sofia",female,18.0,0,0,3101265,7.4958,,S +156,0,1,"Williams, Mr. Charles Duane",male,51.0,0,1,PC 17597,61.3792,,C +190,0,3,"Turcin, Mr. Stjepan",male,36.0,0,0,349247,7.8958,,S +556,0,1,"Wright, Mr. George",male,62.0,0,0,113807,26.55,,S +890,1,1,"Behr, Mr. Karl Howell",male,26.0,0,0,111369,30.0,C148,C +827,0,3,"Lam, Mr. Len",male,,0,0,1601,56.4958,,S +534,1,3,"Peter, Mrs. Catherine (Catherine Rizk)",female,,0,2,2668,22.3583,,C +834,0,3,"Augustsson, Mr. Albert",male,23.0,0,0,347468,7.8542,,S +279,0,3,"Rice, Master. Eric",male,7.0,4,1,382652,29.125,,Q +189,0,3,"Bourke, Mr. John",male,40.0,1,1,364849,15.5,,Q +561,0,3,"Morrow, Mr. Thomas Rowan",male,,0,0,372622,7.75,,Q +375,0,3,"Palsson, Miss. Stina Viola",female,3.0,3,1,349909,21.075,,S +322,0,3,"Danoff, Mr. Yoto",male,27.0,0,0,349219,7.8958,,S +158,0,3,"Corn, Mr. Harry",male,30.0,0,0,SOTON/OQ 392090,8.05,,S +524,1,1,"Hippach, Mrs. Louis Albert (Ida Sophia Fischer)",female,44.0,0,1,111361,57.9792,B18,C +175,0,1,"Smith, Mr. James Clinch",male,56.0,0,0,17764,30.6958,A7,C +117,0,3,"Connors, Mr. Patrick",male,70.5,0,0,370369,7.75,,Q +810,1,1,"Chambers, Mrs. Norman Campbell (Bertha Griggs)",female,33.0,1,0,113806,53.1,E8,S +472,0,3,"Cacic, Mr. Luka",male,38.0,0,0,315089,8.6625,,S +228,0,3,"Lovell, Mr. John Hall (""Henry"")",male,20.5,0,0,A/5 21173,7.25,,S +330,1,1,"Hippach, Miss. Jean Gertrude",female,16.0,0,1,111361,57.9792,B18,C +147,1,3,"Andersson, Mr. August Edvard (""Wennerstrom"")",male,27.0,0,0,350043,7.7958,,S +98,1,1,"Greenfield, Mr. William Bertram",male,23.0,0,1,PC 17759,63.3583,D10 D12,C +493,0,1,"Molson, Mr. Harry Markland",male,55.0,0,0,113787,30.5,C30,S +73,0,2,"Hood, Mr. Ambrose Jr",male,21.0,0,0,S.O.C. 14879,73.5,,S +645,1,3,"Baclini, Miss. Eugenie",female,0.75,2,1,2666,19.2583,,C +303,0,3,"Johnson, Mr. William Cahoone Jr",male,19.0,0,0,LINE,0.0,,S +699,0,1,"Thayer, Mr. John Borland",male,49.0,1,1,17421,110.8833,C68,C +704,0,3,"Gallagher, Mr. Martin",male,25.0,0,0,36864,7.7417,,Q +639,0,3,"Panula, Mrs. Juha (Maria Emilia Ojala)",female,41.0,0,5,3101295,39.6875,,S +99,1,2,"Doling, Mrs. John T (Ada Julia Bone)",female,34.0,0,1,231919,23.0,,S +74,0,3,"Chronopoulos, Mr. Apostolos",male,26.0,1,0,2680,14.4542,,C +157,1,3,"Gilnagh, Miss. Katherine ""Katie""",female,16.0,0,0,35851,7.7333,,Q +475,0,3,"Strandberg, Miss. Ida Sofia",female,22.0,0,0,7553,9.8375,,S +240,0,2,"Hunt, Mr. George Henry",male,33.0,0,0,SCO/W 1585,12.275,,S +801,0,2,"Ponesell, Mr. Martin",male,34.0,0,0,250647,13.0,,S +829,1,3,"McCormack, Mr. Thomas Joseph",male,,0,0,367228,7.75,,Q +208,1,3,"Albimona, Mr. Nassef Cassem",male,26.0,0,0,2699,18.7875,,C +29,1,3,"O'Dwyer, Miss. Ellen ""Nellie""",female,,0,0,330959,7.8792,,Q +616,1,2,"Herman, Miss. Alice",female,24.0,1,2,220845,65.0,,S +309,0,2,"Abelson, Mr. Samuel",male,30.0,1,0,P/PP 3381,24.0,,C +382,1,3,"Nakid, Miss. Maria (""Mary"")",female,1.0,0,2,2653,15.7417,,C +703,0,3,"Barbara, Miss. Saiide",female,18.0,0,1,2691,14.4542,,C +623,1,3,"Nakid, Mr. Sahid",male,20.0,1,1,2653,15.7417,,C +26,1,3,"Asplund, Mrs. Carl Oscar (Selma Augusta Emilia Johansson)",female,38.0,1,5,347077,31.3875,,S +519,1,2,"Angle, Mrs. William A (Florence ""Mary"" Agnes Hughes)",female,36.0,1,0,226875,26.0,,S +638,0,2,"Collyer, Mr. Harvey",male,31.0,1,1,C.A. 31921,26.25,,S +360,1,3,"Mockler, Miss. Helen Mary ""Ellie""",female,,0,0,330980,7.8792,,Q +736,0,3,"Williams, Mr. Leslie",male,28.5,0,0,54636,16.1,,S +101,0,3,"Petranec, Miss. Matilda",female,28.0,0,0,349245,7.8958,,S +165,0,3,"Panula, Master. Eino Viljami",male,1.0,4,1,3101295,39.6875,,S +591,0,3,"Rintamaki, Mr. Matti",male,35.0,0,0,STON/O 2. 3101273,7.125,,S +11,1,3,"Sandstrom, Miss. Marguerite Rut",female,4.0,1,1,PP 9549,16.7,G6,S +217,1,3,"Honkanen, Miss. Eliina",female,27.0,0,0,STON/O2. 3101283,7.925,,S +734,0,2,"Berriman, Mr. William John",male,23.0,0,0,28425,13.0,,S +385,0,3,"Plotcharsky, Mr. Vasil",male,,0,0,349227,7.8958,,S +854,1,1,"Lines, Miss. Mary Conover",female,16.0,0,1,PC 17592,39.4,D28,S +860,0,3,"Razi, Mr. Raihed",male,,0,0,2629,7.2292,,C +359,1,3,"McGovern, Miss. Mary",female,,0,0,330931,7.8792,,Q +448,1,1,"Seward, Mr. Frederic Kimber",male,34.0,0,0,113794,26.55,,S +214,0,2,"Givard, Mr. Hans Kristensen",male,30.0,0,0,250646,13.0,,S +652,1,2,"Doling, Miss. Elsie",female,18.0,0,1,231919,23.0,,S +192,0,2,"Carbines, Mr. William",male,19.0,0,0,28424,13.0,,S +57,1,2,"Rugg, Miss. Emily",female,21.0,0,0,C.A. 31026,10.5,,S +868,0,1,"Roebling, Mr. Washington Augustus II",male,31.0,0,0,PC 17590,50.4958,A24,S +531,1,2,"Quick, Miss. Phyllis May",female,2.0,1,1,26360,26.0,,S +248,1,2,"Hamalainen, Mrs. William (Anna)",female,24.0,0,2,250649,14.5,,S +260,1,2,"Parrish, Mrs. (Lutie Davis)",female,50.0,0,1,230433,26.0,,S +354,0,3,"Arnold-Franchi, Mr. Josef",male,25.0,1,0,349237,17.8,,S +784,0,3,"Johnston, Mr. Andrew G",male,,1,2,W./C. 6607,23.45,,S +853,0,3,"Boulos, Miss. Nourelain",female,9.0,1,1,2678,15.2458,,C diff --git a/tests/data/ml_datasets/titanic/split_train.csv b/tests/data/ml_datasets/titanic/split_train.csv new file mode 100644 index 000000000..a48680208 --- /dev/null +++ b/tests/data/ml_datasets/titanic/split_train.csv @@ -0,0 +1,713 @@ +PassengerId,Survived,Pclass,Name,Sex,Age,SibSp,Parch,Ticket,Fare,Cabin,Embarked +409,0,3,"Birkeland, Mr. Hans Martin Monsen",male,21.0,0,0,312992,7.775,,S +481,0,3,"Goodwin, Master. Harold Victor",male,9.0,5,2,CA 2144,46.9,,S +511,1,3,"Daly, Mr. Eugene Patrick",male,29.0,0,0,382651,7.75,,Q +610,1,1,"Shutes, Miss. Elizabeth W",female,40.0,0,0,PC 17582,153.4625,C125,S +548,1,2,"Padro y Manent, Mr. Julian",male,,0,0,SC/PARIS 2146,13.8625,,C +710,1,3,"Moubarek, Master. Halim Gonios (""William George"")",male,,1,1,2661,15.2458,,C +153,0,3,"Meo, Mr. Alfonzo",male,55.5,0,0,A.5. 11206,8.05,,S +494,0,1,"Artagaveytia, Mr. Ramon",male,71.0,0,0,PC 17609,49.5042,,C +393,0,3,"Gustafsson, Mr. Johan Birger",male,28.0,2,0,3101277,7.925,,S +824,1,3,"Moor, Mrs. (Beila)",female,27.0,0,1,392096,12.475,E121,S +577,1,2,"Garside, Miss. Ethel",female,34.0,0,0,243880,13.0,,S +773,0,2,"Mack, Mrs. (Mary)",female,57.0,0,0,S.O./P.P. 3,10.5,E77,S +745,1,3,"Stranden, Mr. Juho",male,31.0,0,0,STON/O 2. 3101288,7.925,,S +328,1,2,"Ball, Mrs. (Ada E Hall)",female,36.0,0,0,28551,13.0,D,S +460,0,3,"O'Connor, Mr. Maurice",male,,0,0,371060,7.75,,Q +222,0,2,"Bracken, Mr. James H",male,27.0,0,0,220367,13.0,,S +851,0,3,"Andersson, Master. Sigvard Harald Elias",male,4.0,4,2,347082,31.275,,S +558,0,1,"Robbins, Mr. Victor",male,,0,0,PC 17757,227.525,,C +47,0,3,"Lennon, Mr. Denis",male,,1,0,370371,15.5,,Q +449,1,3,"Baclini, Miss. Marie Catherine",female,5.0,2,1,2666,19.2583,,C +371,1,1,"Harder, Mr. George Achilles",male,25.0,1,0,11765,55.4417,E50,C +196,1,1,"Lurette, Miss. Elise",female,58.0,0,0,PC 17569,146.5208,B80,C +761,0,3,"Garfirth, Mr. John",male,,0,0,358585,14.5,,S +55,0,1,"Ostby, Mr. Engelhart Cornelius",male,65.0,0,1,113509,61.9792,B30,C +573,1,1,"Flynn, Mr. John Irwin (""Irving"")",male,36.0,0,0,PC 17474,26.3875,E25,S +379,0,3,"Betros, Mr. Tannous",male,20.0,0,0,2648,4.0125,,C +198,0,3,"Olsen, Mr. Karl Siegwart Andreas",male,42.0,0,1,4579,8.4042,,S +396,0,3,"Johansson, Mr. Erik",male,22.0,0,0,350052,7.7958,,S +111,0,1,"Porter, Mr. Walter Chamberlain",male,47.0,0,0,110465,52.0,C110,S +138,0,1,"Futrelle, Mr. Jacques Heath",male,37.0,1,0,113803,53.1,C123,S +312,1,1,"Ryerson, Miss. Emily Borie",female,18.0,2,2,PC 17608,262.375,B57 B59 B63 B66,C +391,1,1,"Carter, Mr. William Ernest",male,36.0,1,2,113760,120.0,B96 B98,S +24,1,1,"Sloper, Mr. William Thompson",male,28.0,0,0,113788,35.5,A6,S +818,0,2,"Mallet, Mr. Albert",male,31.0,1,1,S.C./PARIS 2079,37.0042,,C +110,1,3,"Moran, Miss. Bertha",female,,1,0,371110,24.15,,Q +302,1,3,"McCoy, Mr. Bernard",male,,2,0,367226,23.25,,Q +104,0,3,"Johansson, Mr. Gustaf Joel",male,33.0,0,0,7540,8.6542,,S +875,1,2,"Abelson, Mrs. Samuel (Hannah Wizosky)",female,28.0,1,0,P/PP 3381,24.0,,C +62,1,1,"Icard, Miss. Amelie",female,38.0,0,0,113572,80.0,B28, +154,0,3,"van Billiard, Mr. Austin Blyler",male,40.5,0,2,A/5. 851,14.5,,S +289,1,2,"Hosono, Mr. Masabumi",male,42.0,0,0,237798,13.0,,S +245,0,3,"Attalah, Mr. Sleiman",male,30.0,0,0,2694,7.225,,C +681,0,3,"Peters, Miss. Katie",female,,0,0,330935,8.1375,,Q +797,1,1,"Leader, Dr. Alice (Farnham)",female,49.0,0,0,17465,25.9292,D17,S +226,0,3,"Berglund, Mr. Karl Ivar Sven",male,22.0,0,0,PP 4348,9.35,,S +857,1,1,"Wick, Mrs. George Dennick (Mary Hitchcock)",female,45.0,1,1,36928,164.8667,,S +621,0,3,"Yasbeck, Mr. Antoni",male,27.0,1,0,2659,14.4542,,C +451,0,2,"West, Mr. Edwy Arthur",male,36.0,1,2,C.A. 34651,27.75,,S +424,0,3,"Danbom, Mrs. Ernst Gilbert (Anna Sigrid Maria Brogren)",female,28.0,1,1,347080,14.4,,S +450,1,1,"Peuchen, Major. Arthur Godfrey",male,52.0,0,0,113786,30.5,C104,S +161,0,3,"Cribb, Mr. John Hatfield",male,44.0,0,1,371362,16.1,,S +743,1,1,"Ryerson, Miss. Susan Parker ""Suzette""",female,21.0,2,2,PC 17608,262.375,B57 B59 B63 B66,C +651,0,3,"Mitkoff, Mr. Mito",male,,0,0,349221,7.8958,,S +250,0,2,"Carter, Rev. Ernest Courtenay",male,54.0,1,0,244252,26.0,,S +540,1,1,"Frolicher, Miss. Hedwig Margaritha",female,22.0,0,2,13568,49.5,B39,C +414,0,2,"Cunningham, Mr. Alfred Fleming",male,,0,0,239853,0.0,,S +207,0,3,"Backstrom, Mr. Karl Alfred",male,32.0,1,0,3101278,15.85,,S +828,1,2,"Mallet, Master. Andre",male,1.0,0,2,S.C./PARIS 2079,37.0042,,C +484,1,3,"Turkula, Mrs. (Hedwig)",female,63.0,0,0,4134,9.5875,,S +607,0,3,"Karaic, Mr. Milan",male,30.0,0,0,349246,7.8958,,S +185,1,3,"Kink-Heilmann, Miss. Luise Gretchen",female,4.0,0,2,315153,22.025,,S +683,0,3,"Olsvigen, Mr. Thor Anderson",male,20.0,0,0,6563,9.225,,S +794,0,1,"Hoyt, Mr. William Fisher",male,,0,0,PC 17600,30.6958,,C +13,0,3,"Saundercock, Mr. William Henry",male,20.0,0,0,A/5. 2151,8.05,,S +118,0,2,"Turpin, Mr. William John Robert",male,29.0,1,0,11668,21.0,,S +483,0,3,"Rouse, Mr. Richard Henry",male,50.0,0,0,A/5 3594,8.05,,S +421,0,3,"Gheorgheff, Mr. Stanio",male,,0,0,349254,7.8958,,C +543,0,3,"Andersson, Miss. Sigrid Elisabeth",female,11.0,4,2,347082,31.275,,S +884,0,2,"Banfield, Mr. Frederick James",male,28.0,0,0,C.A./SOTON 34068,10.5,,S +877,0,3,"Gustafsson, Mr. Alfred Ossian",male,20.0,0,0,7534,9.8458,,S +109,0,3,"Rekic, Mr. Tido",male,38.0,0,0,349249,7.8958,,S +603,0,1,"Harrington, Mr. Charles H",male,,0,0,113796,42.4,,S +575,0,3,"Rush, Mr. Alfred George John",male,16.0,0,0,A/4. 20589,8.05,,S +253,0,1,"Stead, Mr. William Thomas",male,62.0,0,0,113514,26.55,C87,S +712,0,1,"Klaber, Mr. Herman",male,,0,0,113028,26.55,C124,S +397,0,3,"Olsson, Miss. Elina",female,31.0,0,0,350407,7.8542,,S +194,1,2,"Navratil, Master. Michel M",male,3.0,1,1,230080,26.0,F2,S +567,0,3,"Stoytcheff, Mr. Ilia",male,19.0,0,0,349205,7.8958,,S +204,0,3,"Youseff, Mr. Gerious",male,45.5,0,0,2628,7.225,,C +491,0,3,"Hagland, Mr. Konrad Mathias Reiersen",male,,1,0,65304,19.9667,,S +815,0,3,"Tomlin, Mr. Ernest Portage",male,30.5,0,0,364499,8.05,,S +219,1,1,"Bazzani, Miss. Albina",female,32.0,0,0,11813,76.2917,D15,C +446,1,1,"Dodge, Master. Washington",male,4.0,0,2,33638,81.8583,A34,S +490,1,3,"Coutts, Master. Eden Leslie ""Neville""",male,9.0,1,1,C.A. 37671,15.9,,S +112,0,3,"Zabour, Miss. Hileni",female,14.5,1,0,2665,14.4542,,C +731,1,1,"Allen, Miss. Elisabeth Walton",female,29.0,0,0,24160,211.3375,B5,S +106,0,3,"Mionoff, Mr. Stoytcho",male,28.0,0,0,349207,7.8958,,S +480,1,3,"Hirvonen, Miss. Hildur E",female,2.0,0,1,3101298,12.2875,,S +278,0,2,"Parkes, Mr. Francis ""Frank""",male,,0,0,239853,0.0,,S +70,0,3,"Kink, Mr. Vincenz",male,26.0,2,0,315151,8.6625,,S +86,1,3,"Backstrom, Mrs. Karl Alfred (Maria Mathilda Gustafsson)",female,33.0,3,0,3101278,15.85,,S +795,0,3,"Dantcheff, Mr. Ristiu",male,25.0,0,0,349203,7.8958,,S +162,1,2,"Watt, Mrs. James (Elizabeth ""Bessie"" Inglis Milne)",female,40.0,0,0,C.A. 33595,15.75,,S +816,0,1,"Fry, Mr. Richard",male,,0,0,112058,0.0,B102,S +517,1,2,"Lemore, Mrs. (Amelia Milley)",female,34.0,0,0,C.A. 34260,10.5,F33,S +300,1,1,"Baxter, Mrs. James (Helene DeLaudeniere Chaput)",female,50.0,0,1,PC 17558,247.5208,B58 B60,C +455,0,3,"Peduzzi, Mr. Joseph",male,,0,0,A/5 2817,8.05,,S +60,0,3,"Goodwin, Master. William Frederick",male,11.0,5,2,CA 2144,46.9,,S +880,1,1,"Potter, Mrs. Thomas Jr (Lily Alexenia Wilson)",female,56.0,0,1,11767,83.1583,C50,C +43,0,3,"Kraeff, Mr. Theodor",male,,0,0,349253,7.8958,,C +500,0,3,"Svensson, Mr. Olof",male,24.0,0,0,350035,7.7958,,S +236,0,3,"Harknett, Miss. Alice Phoebe",female,,0,0,W./C. 6609,7.55,,S +255,0,3,"Rosblom, Mrs. Viktor (Helena Wilhelmina)",female,41.0,0,2,370129,20.2125,,S +346,1,2,"Brown, Miss. Amelia ""Mildred""",female,24.0,0,0,248733,13.0,F33,S +105,0,3,"Gustafsson, Mr. Anders Vilhelm",male,37.0,2,0,3101276,7.925,,S +316,1,3,"Nilsson, Miss. Helmina Josefina",female,26.0,0,0,347470,7.8542,,S +873,0,1,"Carlsson, Mr. Frans Olof",male,33.0,0,0,695,5.0,B51 B53 B55,S +4,1,1,"Futrelle, Mrs. Jacques Heath (Lily May Peel)",female,35.0,1,0,113803,53.1,C123,S +805,1,3,"Hedman, Mr. Oskar Arvid",male,27.0,0,0,347089,6.975,,S +225,1,1,"Hoyt, Mr. Frederick Maxfield",male,38.0,1,0,19943,90.0,C93,S +772,0,3,"Jensen, Mr. Niels Peder",male,48.0,0,0,350047,7.8542,,S +539,0,3,"Risien, Mr. Samuel Beard",male,,0,0,364498,14.5,,S +249,1,1,"Beckwith, Mr. Richard Leonard",male,37.0,1,1,11751,52.5542,D35,S +32,1,1,"Spencer, Mrs. William Augustus (Marie Eugenie)",female,,1,0,PC 17569,146.5208,B78,C +268,1,3,"Persson, Mr. Ernst Ulrik",male,25.0,1,0,347083,7.775,,S +544,1,2,"Beane, Mr. Edward",male,32.0,1,0,2908,26.0,,S +685,0,2,"Brown, Mr. Thomas William Solomon",male,60.0,1,1,29750,39.0,,S +608,1,1,"Daniel, Mr. Robert Williams",male,27.0,0,0,113804,30.5,,S +749,0,1,"Marvin, Mr. Daniel Warner",male,19.0,1,0,113773,53.1,D30,S +234,1,3,"Asplund, Miss. Lillian Gertrud",female,5.0,4,2,347077,31.3875,,S +641,0,3,"Jensen, Mr. Hans Peder",male,20.0,0,0,350050,7.8542,,S +707,1,2,"Kelly, Mrs. Florence ""Fannie""",female,45.0,0,0,223596,13.5,,S +611,0,3,"Andersson, Mrs. Anders Johan (Alfrida Konstantia Brogren)",female,39.0,1,5,347082,31.275,,S +647,0,3,"Cor, Mr. Liudevit",male,19.0,0,0,349231,7.8958,,S +148,0,3,"Ford, Miss. Robina Maggie ""Ruby""",female,9.0,2,2,W./C. 6608,34.375,,S +574,1,3,"Kelly, Miss. Mary",female,,0,0,14312,7.75,,Q +809,0,2,"Meyer, Mr. August",male,39.0,0,0,248723,13.0,,S +535,0,3,"Cacic, Miss. Marija",female,30.0,0,0,315084,8.6625,,S +588,1,1,"Frolicher-Stehli, Mr. Maxmillian",male,60.0,1,1,13567,79.2,B41,C +331,1,3,"McCoy, Miss. Agnes",female,,2,0,367226,23.25,,Q +569,0,3,"Doharr, Mr. Tannous",male,,0,0,2686,7.2292,,C +725,1,1,"Chambers, Mr. Norman Campbell",male,27.0,1,0,113806,53.1,E8,S +100,0,2,"Kantor, Mr. Sinai",male,34.0,1,0,244367,26.0,,S +708,1,1,"Calderhead, Mr. Edward Pennington",male,42.0,0,0,PC 17476,26.2875,E24,S +277,0,3,"Lindblom, Miss. Augusta Charlotta",female,45.0,0,0,347073,7.75,,S +418,1,2,"Silven, Miss. Lyyli Karoliina",female,18.0,0,2,250652,13.0,,S +463,0,1,"Gee, Mr. Arthur H",male,47.0,0,0,111320,38.5,E63,S +665,1,3,"Lindqvist, Mr. Eino William",male,20.0,1,0,STON/O 2. 3101285,7.925,,S +718,1,2,"Troutt, Miss. Edwina Celia ""Winnie""",female,27.0,0,0,34218,10.5,E101,S +850,1,1,"Goldenberg, Mrs. Samuel L (Edwiga Grabowska)",female,,1,0,17453,89.1042,C92,C +516,0,1,"Walker, Mr. William Anderson",male,47.0,0,0,36967,34.0208,D46,S +633,1,1,"Stahelin-Maeglin, Dr. Max",male,32.0,0,0,13214,30.5,B50,C +538,1,1,"LeRoy, Miss. Bertha",female,30.0,0,0,PC 17761,106.425,,C +151,0,2,"Bateman, Rev. Robert James",male,51.0,0,0,S.O.P. 1166,12.525,,S +79,1,2,"Caldwell, Master. Alden Gates",male,0.83,0,2,248738,29.0,,S +10,1,2,"Nasser, Mrs. Nicholas (Adele Achem)",female,14.0,1,0,237736,30.0708,,C +143,1,3,"Hakkarainen, Mrs. Pekka Pietari (Elin Matilda Dolck)",female,24.0,1,0,STON/O2. 3101279,15.85,,S +76,0,3,"Moen, Mr. Sigurd Hansen",male,25.0,0,0,348123,7.65,F G73,S +254,0,3,"Lobb, Mr. William Arthur",male,30.0,1,0,A/5. 3336,16.1,,S +30,0,3,"Todoroff, Mr. Lalio",male,,0,0,349216,7.8958,,S +170,0,3,"Ling, Mr. Lee",male,28.0,0,0,1601,56.4958,,S +747,0,3,"Abbott, Mr. Rossmore Edward",male,16.0,1,1,C.A. 2673,20.25,,S +212,1,2,"Cameron, Miss. Clear Annie",female,35.0,0,0,F.C.C. 13528,21.0,,S +636,1,2,"Davis, Miss. Mary",female,28.0,0,0,237668,13.0,,S +689,0,3,"Fischer, Mr. Eberhard Thelander",male,18.0,0,0,350036,7.7958,,S +600,1,1,"Duff Gordon, Sir. Cosmo Edmund (""Mr Morgan"")",male,49.0,1,0,PC 17485,56.9292,A20,C +423,0,3,"Zimmerman, Mr. Leo",male,29.0,0,0,315082,7.875,,S +59,1,2,"West, Miss. Constance Mirium",female,5.0,1,2,C.A. 34651,27.75,,S +504,0,3,"Laitinen, Miss. Kristina Sofia",female,37.0,0,0,4135,9.5875,,S +352,0,1,"Williams-Lambert, Mr. Fletcher Fellows",male,,0,0,113510,35.0,C128,S +542,0,3,"Andersson, Miss. Ingeborg Constanzia",female,9.0,4,2,347082,31.275,,S +89,1,1,"Fortune, Miss. Mabel Helen",female,23.0,3,2,19950,263.0,C23 C25 C27,S +433,1,2,"Louch, Mrs. Charles Alexander (Alice Adelaide Slow)",female,42.0,1,0,SC/AH 3085,26.0,,S +566,0,3,"Davies, Mr. Alfred J",male,24.0,2,0,A/4 48871,24.15,,S +502,0,3,"Canavan, Miss. Mary",female,21.0,0,0,364846,7.75,,Q +128,1,3,"Madsen, Mr. Fridtjof Arne",male,24.0,0,0,C 17369,7.1417,,S +688,0,3,"Dakic, Mr. Branko",male,19.0,0,0,349228,10.1708,,S +329,1,3,"Goldsmith, Mrs. Frank John (Emily Alice Brown)",female,31.0,1,1,363291,20.525,,S +845,0,3,"Culumovic, Mr. Jeso",male,17.0,0,0,315090,8.6625,,S +886,0,3,"Rice, Mrs. William (Margaret Norton)",female,39.0,0,5,382652,29.125,,Q +581,1,2,"Christy, Miss. Julie Rachel",female,25.0,1,1,237789,30.0,,S +568,0,3,"Palsson, Mrs. Nils (Alma Cornelia Berglund)",female,29.0,0,4,349909,21.075,,S +152,1,1,"Pears, Mrs. Thomas (Edith Wearne)",female,22.0,1,0,113776,66.6,C2,S +342,1,1,"Fortune, Miss. Alice Elizabeth",female,24.0,3,2,19950,263.0,C23 C25 C27,S +272,1,3,"Tornquist, Mr. William Henry",male,25.0,0,0,LINE,0.0,,S +737,0,3,"Ford, Mrs. Edward (Margaret Ann Watson)",female,48.0,1,3,W./C. 6608,34.375,,S +700,0,3,"Humblen, Mr. Adolf Mathias Nicolai Olsen",male,42.0,0,0,348121,7.65,F G63,S +291,1,1,"Barber, Miss. Ellen ""Nellie""",female,26.0,0,0,19877,78.85,,S +141,0,3,"Boulos, Mrs. Joseph (Sultana)",female,,0,2,2678,15.2458,,C +261,0,3,"Smith, Mr. Thomas",male,,0,0,384461,7.75,,Q +163,0,3,"Bengtsson, Mr. John Viktor",male,26.0,0,0,347068,7.775,,S +232,0,3,"Larsson, Mr. Bengt Edvin",male,29.0,0,0,347067,7.775,,S +802,1,2,"Collyer, Mrs. Harvey (Charlotte Annie Tate)",female,31.0,1,1,C.A. 31921,26.25,,S +844,0,3,"Lemberopolous, Mr. Peter L",male,34.5,0,0,2683,6.4375,,C +691,1,1,"Dick, Mr. Albert Adrian",male,31.0,1,0,17474,57.0,B20,S +649,0,3,"Willey, Mr. Edward",male,,0,0,S.O./P.P. 751,7.55,,S +137,1,1,"Newsom, Miss. Helen Monypeny",female,19.0,0,2,11752,26.2833,D47,S +570,1,3,"Jonsson, Mr. Carl",male,32.0,0,0,350417,7.8542,,S +862,0,2,"Giles, Mr. Frederick Edward",male,21.0,1,0,28134,11.5,,S +445,1,3,"Johannesen-Bratthammer, Mr. Bernt",male,,0,0,65306,8.1125,,S +697,0,3,"Kelly, Mr. James",male,44.0,0,0,363592,8.05,,S +674,1,2,"Wilhelms, Mr. Charles",male,31.0,0,0,244270,13.0,,S +748,1,2,"Sinkkonen, Miss. Anna",female,30.0,0,0,250648,13.0,,S +367,1,1,"Warren, Mrs. Frank Manley (Anna Sophia Atkinson)",female,60.0,1,0,110813,75.25,D37,C +626,0,1,"Sutton, Mr. Frederick",male,61.0,0,0,36963,32.3208,D50,S +741,1,1,"Hawksford, Mr. Walter James",male,,0,0,16988,30.0,D45,S +821,1,1,"Hays, Mrs. Charles Melville (Clara Jennings Gregg)",female,52.0,1,1,12749,93.5,B69,S +282,0,3,"Olsson, Mr. Nils Johan Goransson",male,28.0,0,0,347464,7.8542,,S +546,0,1,"Nicholson, Mr. Arthur Ernest",male,64.0,0,0,693,26.0,,S +237,0,2,"Hold, Mr. Stephen",male,44.0,1,0,26707,26.0,,S +16,1,2,"Hewlett, Mrs. (Mary D Kingcome) ",female,55.0,0,0,248706,16.0,,S +565,0,3,"Meanwell, Miss. (Marion Ogden)",female,,0,0,SOTON/O.Q. 392087,8.05,,S +798,1,3,"Osman, Mrs. Mara",female,31.0,0,0,349244,8.6833,,S +740,0,3,"Nankoff, Mr. Minko",male,,0,0,349218,7.8958,,S +549,0,3,"Goldsmith, Mr. Frank John",male,33.0,1,1,363291,20.525,,S +663,0,1,"Colley, Mr. Edward Pomeroy",male,47.0,0,0,5727,25.5875,E58,S +482,0,2,"Frost, Mr. Anthony Wood ""Archie""",male,,0,0,239854,0.0,,S +113,0,3,"Barton, Mr. David John",male,22.0,0,0,324669,8.05,,S +458,1,1,"Kenyon, Mrs. Frederick R (Marion)",female,,1,0,17464,51.8625,D21,S +842,0,2,"Mudd, Mr. Thomas Charles",male,16.0,0,0,S.O./P.P. 3,10.5,,S +518,0,3,"Ryan, Mr. Patrick",male,,0,0,371110,24.15,,Q +553,0,3,"O'Brien, Mr. Timothy",male,,0,0,330979,7.8292,,Q +388,1,2,"Buss, Miss. Kate",female,36.0,0,0,27849,13.0,,S +514,1,1,"Rothschild, Mrs. Martin (Elizabeth L. Barrett)",female,54.0,1,0,PC 17603,59.4,,C +560,1,3,"de Messemaeker, Mrs. Guillaume Joseph (Emma)",female,36.0,1,0,345572,17.4,,S +701,1,1,"Astor, Mrs. John Jacob (Madeleine Talmadge Force)",female,18.0,1,0,PC 17757,227.525,C62 C64,C +241,0,3,"Zabour, Miss. Thamine",female,,1,0,2665,14.4542,,C +428,1,2,"Phillips, Miss. Kate Florence (""Mrs Kate Louise Phillips Marshall"")",female,19.0,0,0,250655,26.0,,S +593,0,3,"Elsbury, Mr. William James",male,47.0,0,0,A/5 3902,7.25,,S +116,0,3,"Pekoniemi, Mr. Edvard",male,21.0,0,0,STON/O 2. 3101294,7.925,,S +686,0,2,"Laroche, Mr. Joseph Philippe Lemercier",male,25.0,1,2,SC/Paris 2123,41.5792,,C +155,0,3,"Olsen, Mr. Ole Martin",male,,0,0,Fa 265302,7.3125,,S +308,1,1,"Penasco y Castellana, Mrs. Victor de Satode (Maria Josefa Perez de Soto y Vallejo)",female,17.0,1,0,PC 17758,108.9,C65,C +765,0,3,"Eklund, Mr. Hans Linus",male,16.0,0,0,347074,7.775,,S +597,1,2,"Leitch, Miss. Jessie Wills",female,,0,0,248727,33.0,,S +242,1,3,"Murphy, Miss. Katherine ""Kate""",female,,1,0,367230,15.5,,Q +823,0,1,"Reuchlin, Jonkheer. John George",male,38.0,0,0,19972,0.0,,S +380,0,3,"Gustafsson, Mr. Karl Gideon",male,19.0,0,0,347069,7.775,,S +336,0,3,"Denkoff, Mr. Mitto",male,,0,0,349225,7.8958,,S +488,0,1,"Kent, Mr. Edward Austin",male,58.0,0,0,11771,29.7,B37,C +672,0,1,"Davidson, Mr. Thornton",male,31.0,1,0,F.C. 12750,52.0,B71,S +791,0,3,"Keane, Mr. Andrew ""Andy""",male,,0,0,12460,7.75,,Q +340,0,1,"Blackwell, Mr. Stephen Weart",male,45.0,0,0,113784,35.5,T,S +879,0,3,"Laleff, Mr. Kristo",male,,0,0,349217,7.8958,,S +464,0,2,"Milling, Mr. Jacob Christian",male,48.0,0,0,234360,13.0,,S +717,1,1,"Endres, Miss. Caroline Louise",female,38.0,0,0,PC 17757,227.525,C45,C +343,0,2,"Collander, Mr. Erik Gustaf",male,28.0,0,0,248740,13.0,,S +276,1,1,"Andrews, Miss. Kornelia Theodosia",female,63.0,1,0,13502,77.9583,D7,S +530,0,2,"Hocking, Mr. Richard George",male,23.0,2,1,29104,11.5,,S +861,0,3,"Hansen, Mr. Claus Peter",male,41.0,2,0,350026,14.1083,,S +8,0,3,"Palsson, Master. Gosta Leonard",male,2.0,3,1,349909,21.075,,S +841,0,3,"Alhomaki, Mr. Ilmari Rudolf",male,20.0,0,0,SOTON/O2 3101287,7.925,,S +231,1,1,"Harris, Mrs. Henry Birkhardt (Irene Wallach)",female,35.0,1,0,36973,83.475,C83,S +338,1,1,"Burns, Miss. Elizabeth Margaret",female,41.0,0,0,16966,134.5,E40,C +286,0,3,"Stankovic, Mr. Ivan",male,33.0,0,0,349239,8.6625,,C +381,1,1,"Bidois, Miss. Rosalie",female,42.0,0,0,PC 17757,227.525,,C +468,0,1,"Smart, Mr. John Montgomery",male,56.0,0,0,113792,26.55,,S +838,0,3,"Sirota, Mr. Maurice",male,,0,0,392092,8.05,,S +742,0,1,"Cavendish, Mr. Tyrell William",male,36.0,1,0,19877,78.85,C46,S +617,0,3,"Danbom, Mr. Ernst Gilbert",male,34.0,1,1,347080,14.4,,S +485,1,1,"Bishop, Mr. Dickinson H",male,25.0,1,0,11967,91.0792,B49,C +437,0,3,"Ford, Miss. Doolina Margaret ""Daisy""",female,21.0,2,2,W./C. 6608,34.375,,S +885,0,3,"Sutehall, Mr. Henry Jr",male,25.0,0,0,SOTON/OQ 392076,7.05,,S +28,0,1,"Fortune, Mr. Charles Alexander",male,19.0,3,2,19950,263.0,C23 C25 C27,S +751,1,2,"Wells, Miss. Joan",female,4.0,1,1,29103,23.0,,S +97,0,1,"Goldschmidt, Mr. George B",male,71.0,0,0,PC 17754,34.6542,A5,C +6,0,3,"Moran, Mr. James",male,,0,0,330877,8.4583,,Q +271,0,1,"Cairns, Mr. Alexander",male,,0,0,113798,31.0,,S +301,1,3,"Kelly, Miss. Anna Katherine ""Annie Kate""",female,,0,0,9234,7.75,,Q +366,0,3,"Adahl, Mr. Mauritz Nils Martin",male,30.0,0,0,C 7076,7.25,,S +200,0,2,"Yrois, Miss. Henriette (""Mrs Harbeck"")",female,24.0,0,0,248747,13.0,,S +776,0,3,"Myhrman, Mr. Pehr Fabian Oliver Malkolm",male,18.0,0,0,347078,7.75,,S +178,0,1,"Isham, Miss. Ann Elizabeth",female,50.0,0,0,PC 17595,28.7125,C49,C +728,1,3,"Mannion, Miss. Margareth",female,,0,0,36866,7.7375,,Q +167,1,1,"Chibnall, Mrs. (Edith Martha Bowerman)",female,,0,1,113505,55.0,E33,S +869,0,3,"van Melkebeke, Mr. Philemon",male,,0,0,345777,9.5,,S +313,0,2,"Lahtinen, Mrs. William (Anna Sylfven)",female,26.0,1,1,250651,26.0,,S +285,0,1,"Smith, Mr. Richard William",male,,0,0,113056,26.0,A19,S +495,0,3,"Stanley, Mr. Edward Roland",male,21.0,0,0,A/4 45380,8.05,,S +33,1,3,"Glynn, Miss. Mary Agatha",female,,0,0,335677,7.75,,Q +417,1,2,"Drew, Mrs. James Vivian (Lulu Thorne Christian)",female,34.0,1,1,28220,32.5,,S +887,0,2,"Montvila, Rev. Juozas",male,27.0,0,0,211536,13.0,,S +559,1,1,"Taussig, Mrs. Emil (Tillie Mandelbaum)",female,39.0,1,1,110413,79.65,E67,S +806,0,3,"Johansson, Mr. Karl Johan",male,31.0,0,0,347063,7.775,,S +294,0,3,"Haas, Miss. Aloisia",female,24.0,0,0,349236,8.85,,S +209,1,3,"Carr, Miss. Helen ""Ellen""",female,16.0,0,0,367231,7.75,,Q +85,1,2,"Ilett, Miss. Bertha",female,17.0,0,0,SO/C 14885,10.5,,S +38,0,3,"Cann, Mr. Ernest Charles",male,21.0,0,0,A./5. 2152,8.05,,S +7,0,1,"McCarthy, Mr. Timothy J",male,54.0,0,0,17463,51.8625,E46,S +426,0,3,"Wiseman, Mr. Phillippe",male,,0,0,A/4. 34244,7.25,,S +790,0,1,"Guggenheim, Mr. Benjamin",male,46.0,0,0,PC 17593,79.2,B82 B84,C +389,0,3,"Sadlier, Mr. Matthew",male,,0,0,367655,7.7292,,Q +258,1,1,"Cherry, Miss. Gladys",female,30.0,0,0,110152,86.5,B77,S +643,0,3,"Skoog, Miss. Margit Elizabeth",female,2.0,3,2,347088,27.9,,S +355,0,3,"Yousif, Mr. Wazli",male,,0,0,2647,7.225,,C +830,1,1,"Stone, Mrs. George Nelson (Martha Evelyn)",female,62.0,0,0,113572,80.0,B28, +781,1,3,"Ayoub, Miss. Banoura",female,13.0,0,0,2687,7.2292,,C +267,0,3,"Panula, Mr. Ernesti Arvid",male,16.0,4,1,3101295,39.6875,,S +506,0,1,"Penasco y Castellana, Mr. Victor de Satode",male,18.0,1,0,PC 17758,108.9,C65,C +52,0,3,"Nosworthy, Mr. Richard Cater",male,21.0,0,0,A/4. 39886,7.8,,S +401,1,3,"Niskanen, Mr. Juha",male,39.0,0,0,STON/O 2. 3101289,7.925,,S +533,0,3,"Elias, Mr. Joseph Jr",male,17.0,1,1,2690,7.2292,,C +283,0,3,"de Pelsmaeker, Mr. Alfons",male,16.0,0,0,345778,9.5,,S +442,0,3,"Hampe, Mr. Leon",male,20.0,0,0,345769,9.5,,S +361,0,3,"Skoog, Mr. Wilhelm",male,40.0,1,4,347088,27.9,,S +840,1,1,"Marechal, Mr. Pierre",male,,0,0,11774,29.7,C47,C +509,0,3,"Olsen, Mr. Henry Margido",male,28.0,0,0,C 4001,22.525,,S +121,0,2,"Hickman, Mr. Stanley George",male,21.0,2,0,S.O.C. 14879,73.5,,S +320,1,1,"Spedden, Mrs. Frederic Oakley (Margaretta Corning Stone)",female,40.0,1,1,16966,134.5,E34,C +858,1,1,"Daly, Mr. Peter Denis ",male,51.0,0,0,113055,26.55,E17,S +501,0,3,"Calic, Mr. Petar",male,17.0,0,0,315086,8.6625,,S +91,0,3,"Christmann, Mr. Emil",male,29.0,0,0,343276,8.05,,S +727,1,2,"Renouf, Mrs. Peter Henry (Lillian Jefferys)",female,30.0,3,0,31027,21.0,,S +671,1,2,"Brown, Mrs. Thomas William Solomon (Elizabeth Catherine Ford)",female,40.0,1,1,29750,39.0,,S +456,1,3,"Jalsevac, Mr. Ivan",male,29.0,0,0,349240,7.8958,,C +427,1,2,"Clarke, Mrs. Charles V (Ada Maria Winfield)",female,28.0,1,0,2003,26.0,,S +63,0,1,"Harris, Mr. Henry Birkhardt",male,45.0,1,0,36973,83.475,C83,S +51,0,3,"Panula, Master. Juha Niilo",male,7.0,4,1,3101295,39.6875,,S +454,1,1,"Goldenberg, Mr. Samuel L",male,49.0,1,0,17453,89.1042,C92,C +394,1,1,"Newell, Miss. Marjorie",female,23.0,1,0,35273,113.275,D36,C +188,1,1,"Romaine, Mr. Charles Hallace (""Mr C Rolmane"")",male,45.0,0,0,111428,26.55,,S +368,1,3,"Moussa, Mrs. (Mantoura Boulos)",female,,0,0,2626,7.2292,,C +759,0,3,"Theobald, Mr. Thomas Leonard",male,34.0,0,0,363294,8.05,,S +804,1,3,"Thomas, Master. Assad Alexander",male,0.42,0,1,2625,8.5167,,C +510,1,3,"Lang, Mr. Fang",male,26.0,0,0,1601,56.4958,,S +788,0,3,"Rice, Master. George Hugh",male,8.0,4,1,382652,29.125,,Q +298,0,1,"Allison, Miss. Helen Loraine",female,2.0,1,2,113781,151.55,C22 C26,S +92,0,3,"Andreasson, Mr. Paul Edvin",male,20.0,0,0,347466,7.8542,,S +754,0,3,"Jonkoff, Mr. Lalio",male,23.0,0,0,349204,7.8958,,S +547,1,2,"Beane, Mrs. Edward (Ethel Clarke)",female,19.0,1,0,2908,26.0,,S +492,0,3,"Windelov, Mr. Einar",male,21.0,0,0,SOTON/OQ 3101317,7.25,,S +2,1,1,"Cumings, Mrs. John Bradley (Florence Briggs Thayer)",female,38.0,1,0,PC 17599,71.2833,C85,C +777,0,3,"Tobin, Mr. Roger",male,,0,0,383121,7.75,F38,Q +473,1,2,"West, Mrs. Edwy Arthur (Ada Mary Worth)",female,33.0,1,2,C.A. 34651,27.75,,S +252,0,3,"Strom, Mrs. Wilhelm (Elna Matilda Persson)",female,29.0,1,1,347054,10.4625,G6,S +93,0,1,"Chaffee, Mr. Herbert Fuller",male,46.0,1,0,W.E.P. 5734,61.175,E31,S +635,0,3,"Skoog, Miss. Mabel",female,9.0,3,2,347088,27.9,,S +44,1,2,"Laroche, Miss. Simonne Marie Anne Andree",female,3.0,1,2,SC/Paris 2123,41.5792,,C +835,0,3,"Allum, Mr. Owen George",male,18.0,0,0,2223,8.3,,S +48,1,3,"O'Driscoll, Miss. Bridget",female,,0,0,14311,7.75,,Q +891,0,3,"Dooley, Mr. Patrick",male,32.0,0,0,370376,7.75,,Q +264,0,1,"Harrison, Mr. William",male,40.0,0,0,112059,0.0,B94,S +356,0,3,"Vanden Steen, Mr. Leo Peter",male,28.0,0,0,345783,9.5,,S +528,0,1,"Farthing, Mr. John",male,,0,0,PC 17483,221.7792,C95,S +339,1,3,"Dahl, Mr. Karl Edwart",male,45.0,0,0,7598,8.05,,S +780,1,1,"Robert, Mrs. Edward Scott (Elisabeth Walton McMillan)",female,43.0,0,1,24160,211.3375,B3,S +21,0,2,"Fynney, Mr. Joseph J",male,35.0,0,0,239865,26.0,,S +723,0,2,"Gillespie, Mr. William Henry",male,34.0,0,0,12233,13.0,,S +677,0,3,"Sawyer, Mr. Frederick Charles",male,24.5,0,0,342826,8.05,,S +349,1,3,"Coutts, Master. William Loch ""William""",male,3.0,1,1,C.A. 37671,15.9,,S +817,0,3,"Heininen, Miss. Wendla Maria",female,23.0,0,0,STON/O2. 3101290,7.925,,S +334,0,3,"Vander Planke, Mr. Leo Edmondus",male,16.0,2,0,345764,18.0,,S +470,1,3,"Baclini, Miss. Helene Barbara",female,0.75,2,1,2666,19.2583,,C +130,0,3,"Ekstrom, Mr. Johan",male,45.0,0,0,347061,6.975,,S +191,1,2,"Pinsky, Mrs. (Rosa)",female,32.0,0,0,234604,13.0,,S +760,1,1,"Rothes, the Countess. of (Lucy Noel Martha Dyer-Edwards)",female,33.0,0,0,110152,86.5,B77,S +520,0,3,"Pavlovic, Mr. Stefo",male,32.0,0,0,349242,7.8958,,S +67,1,2,"Nye, Mrs. (Elizabeth Ramell)",female,29.0,0,0,C.A. 29395,10.5,F33,S +487,1,1,"Hoyt, Mrs. Frederick Maxfield (Jane Anne Forby)",female,35.0,1,0,19943,90.0,C93,S +19,0,3,"Vander Planke, Mrs. Julius (Emelia Maria Vandemoortele)",female,31.0,1,0,345763,18.0,,S +702,1,1,"Silverthorne, Mr. Spencer Victor",male,35.0,0,0,PC 17475,26.2875,E24,S +826,0,3,"Flynn, Mr. John",male,,0,0,368323,6.95,,Q +333,0,1,"Graham, Mr. George Edward",male,38.0,0,1,PC 17582,153.4625,C91,S +855,0,2,"Carter, Mrs. Ernest Courtenay (Lilian Hughes)",female,44.0,1,0,244252,26.0,,S +441,1,2,"Hart, Mrs. Benjamin (Esther Ada Bloomfield)",female,45.0,1,1,F.C.C. 13529,26.25,,S +775,1,2,"Hocking, Mrs. Elizabeth (Eliza Needs)",female,54.0,1,3,29105,23.0,,S +675,0,2,"Watson, Mr. Ennis Hastings",male,,0,0,239856,0.0,,S +552,0,2,"Sharp, Mr. Percival James R",male,27.0,0,0,244358,26.0,,S +56,1,1,"Woolner, Mr. Hugh",male,,0,0,19947,35.5,C52,S +653,0,3,"Kalvik, Mr. Johannes Halvorsen",male,21.0,0,0,8475,8.4333,,S +849,0,2,"Harper, Rev. John",male,28.0,0,1,248727,33.0,,S +730,0,3,"Ilmakangas, Miss. Pieta Sofia",female,25.0,1,0,STON/O2. 3101271,7.925,,S +233,0,2,"Sjostedt, Mr. Ernst Adolf",male,59.0,0,0,237442,13.5,,S +660,0,1,"Newell, Mr. Arthur Webster",male,58.0,0,2,35273,113.275,D48,C +243,0,2,"Coleridge, Mr. Reginald Charles",male,29.0,0,0,W./C. 14263,10.5,,S +36,0,1,"Holverson, Mr. Alexander Oskar",male,42.0,1,0,113789,52.0,,S +541,1,1,"Crosby, Miss. Harriet R",female,36.0,0,2,WE/P 5735,71.0,B22,S +719,0,3,"McEvoy, Mr. Michael",male,,0,0,36568,15.5,,Q +752,1,3,"Moor, Master. Meier",male,6.0,0,1,392096,12.475,E121,S +888,1,1,"Graham, Miss. Margaret Edith",female,19.0,0,0,112053,30.0,B42,S +122,0,3,"Moore, Mr. Leonard Charles",male,,0,0,A4. 54510,8.05,,S +411,0,3,"Sdycoff, Mr. Todor",male,,0,0,349222,7.8958,,S +353,0,3,"Elias, Mr. Tannous",male,15.0,1,1,2695,7.2292,,C +34,0,2,"Wheadon, Mr. Edward H",male,66.0,0,0,C.A. 24579,10.5,,S +180,0,3,"Leonard, Mr. Lionel",male,36.0,0,0,LINE,0.0,,S +646,1,1,"Harper, Mr. Henry Sleeper",male,48.0,1,0,PC 17572,76.7292,D33,C +819,0,3,"Holm, Mr. John Fredrik Alexander",male,43.0,0,0,C 7075,6.45,,S +22,1,2,"Beesley, Mr. Lawrence",male,34.0,0,0,248698,13.0,D56,S +412,0,3,"Hart, Mr. Henry",male,,0,0,394140,6.8583,,Q +422,0,3,"Charters, Mr. David",male,21.0,0,0,A/5. 13032,7.7333,,Q +584,0,1,"Ross, Mr. John Hugo",male,36.0,0,0,13049,40.125,A10,C +729,0,2,"Bryhl, Mr. Kurt Arnold Gottfrid",male,25.0,1,0,236853,26.0,,S +813,0,2,"Slemen, Mr. Richard James",male,35.0,0,0,28206,10.5,,S +562,0,3,"Sivic, Mr. Husein",male,40.0,0,0,349251,7.8958,,S +332,0,1,"Partner, Mr. Austen",male,45.5,0,0,113043,28.5,C124,S +341,1,2,"Navratil, Master. Edmond Roger",male,2.0,1,1,230080,26.0,F2,S +247,0,3,"Lindahl, Miss. Agda Thorilda Viktoria",female,25.0,0,0,347071,7.775,,S +127,0,3,"McMahon, Mr. Martin",male,,0,0,370372,7.75,,Q +324,1,2,"Caldwell, Mrs. Albert Francis (Sylvia Mae Harbaugh)",female,22.0,1,1,248738,29.0,,S +398,0,2,"McKane, Mr. Peter David",male,46.0,0,0,28403,26.0,,S +46,0,3,"Rogers, Mr. William John",male,,0,0,S.C./A.4. 23567,8.05,,S +65,0,1,"Stewart, Mr. Albert A",male,,0,0,PC 17605,27.7208,,C +262,1,3,"Asplund, Master. Edvin Rojj Felix",male,3.0,4,2,347077,31.3875,,S +372,0,3,"Wiklund, Mr. Jakob Alfred",male,18.0,1,0,3101267,6.4958,,S +376,1,1,"Meyer, Mrs. Edgar Joseph (Leila Saks)",female,,1,0,PC 17604,82.1708,,C +676,0,3,"Edvardsson, Mr. Gustaf Hjalmar",male,18.0,0,0,349912,7.775,,S +471,0,3,"Keefe, Mr. Arthur",male,,0,0,323592,7.25,,S +210,1,1,"Blank, Mr. Henry",male,40.0,0,0,112277,31.0,A31,C +733,0,2,"Knight, Mr. Robert J",male,,0,0,239855,0.0,,S +81,0,3,"Waelens, Mr. Achille",male,22.0,0,0,345767,9.0,,S +609,1,2,"Laroche, Mrs. Joseph (Juliette Marie Louise Lafargue)",female,22.0,1,2,SC/Paris 2123,41.5792,,C +874,0,3,"Vander Cruyssen, Mr. Victor",male,47.0,0,0,345765,9.0,,S +435,0,1,"Silvey, Mr. William Baird",male,50.0,1,0,13507,55.9,E44,S +767,0,1,"Brewe, Dr. Arthur Jackson",male,,0,0,112379,39.6,,C +768,0,3,"Mangan, Miss. Mary",female,30.5,0,0,364850,7.75,,Q +168,0,3,"Skoog, Mrs. William (Anna Bernhardina Karlsson)",female,45.0,1,4,347088,27.9,,S +709,1,1,"Cleaver, Miss. Alice",female,22.0,0,0,113781,151.55,,S +327,0,3,"Nysveen, Mr. Johan Hansen",male,61.0,0,0,345364,6.2375,,S +843,1,1,"Serepeca, Miss. Augusta",female,30.0,0,0,113798,31.0,,C +211,0,3,"Ali, Mr. Ahmed",male,24.0,0,0,SOTON/O.Q. 3101311,7.05,,S +159,0,3,"Smiljanic, Mr. Mile",male,,0,0,315037,8.6625,,S +378,0,1,"Widener, Mr. Harry Elkins",male,27.0,0,2,113503,211.5,C82,C +778,1,3,"Emanuel, Miss. Virginia Ethel",female,5.0,0,0,364516,12.475,,S +457,0,1,"Millet, Mr. Francis Davis",male,65.0,0,0,13509,26.55,E38,S +769,0,3,"Moran, Mr. Daniel J",male,,1,0,371110,24.15,,Q +362,0,2,"del Carlo, Mr. Sebastiano",male,29.0,1,0,SC/PARIS 2167,27.7208,,C +655,0,3,"Hegarty, Miss. Hanora ""Nora""",female,18.0,0,0,365226,6.75,,Q +698,1,3,"Mullens, Miss. Katherine ""Katie""",female,,0,0,35852,7.7333,,Q +444,1,2,"Reynaldo, Ms. Encarnacion",female,28.0,0,0,230434,13.0,,S +203,0,3,"Johanson, Mr. Jakob Alfred",male,34.0,0,0,3101264,6.4958,,S +606,0,3,"Lindell, Mr. Edvard Bengtsson",male,36.0,1,0,349910,15.55,,S +673,0,2,"Mitchell, Mr. Henry Michael",male,70.0,0,0,C.A. 24580,10.5,,S +846,0,3,"Abbing, Mr. Anthony",male,42.0,0,0,C.A. 5547,7.55,,S +374,0,1,"Ringhini, Mr. Sante",male,22.0,0,0,PC 17760,135.6333,,C +667,0,2,"Butler, Mr. Reginald Fenton",male,25.0,0,0,234686,13.0,,S +61,0,3,"Sirayanian, Mr. Orsen",male,22.0,0,0,2669,7.2292,,C +642,1,1,"Sagesser, Mlle. Emma",female,24.0,0,0,PC 17477,69.3,B35,C +469,0,3,"Scanlan, Mr. James",male,,0,0,36209,7.725,,Q +792,0,2,"Gaskell, Mr. Alfred",male,16.0,0,0,239865,26.0,,S +465,0,3,"Maisner, Mr. Simon",male,,0,0,A/S 2816,8.05,,S +551,1,1,"Thayer, Mr. John Borland Jr",male,17.0,0,2,17421,110.8833,C70,C +523,0,3,"Lahoud, Mr. Sarkis",male,,0,0,2624,7.225,,C +369,1,3,"Jermyn, Miss. Annie",female,,0,0,14313,7.75,,Q +864,0,3,"Sage, Miss. Dorothy Edith ""Dolly""",female,,8,2,CA. 2343,69.55,,S +839,1,3,"Chip, Mr. Chang",male,32.0,0,0,1601,56.4958,,S +590,0,3,"Murdlin, Mr. Joseph",male,,0,0,A./5. 3235,8.05,,S +9,1,3,"Johnson, Mrs. Oscar W (Elisabeth Vilhelmina Berg)",female,27.0,0,2,347742,11.1333,,S +505,1,1,"Maioni, Miss. Roberta",female,16.0,0,0,110152,86.5,B79,S +572,1,1,"Appleton, Mrs. Edward Dale (Charlotte Lamson)",female,53.0,2,0,11769,51.4792,C101,S +235,0,2,"Leyson, Mr. Robert William Norman",male,24.0,0,0,C.A. 29566,10.5,,S +345,0,2,"Fox, Mr. Stanley Hubert",male,36.0,0,0,229236,13.0,,S +714,0,3,"Larsson, Mr. August Viktor",male,29.0,0,0,7545,9.4833,,S +477,0,2,"Renouf, Mr. Peter Henry",male,34.0,1,0,31027,21.0,,S +587,0,2,"Jarvis, Mr. John Denzil",male,47.0,0,0,237565,15.0,,S +630,0,3,"O'Connell, Mr. Patrick D",male,,0,0,334912,7.7333,,Q +133,0,3,"Robins, Mrs. Alexander A (Grace Charity Laury)",female,47.0,1,0,A/5. 3337,14.5,,S +27,0,3,"Emir, Mr. Farred Chehab",male,,0,0,2631,7.225,,C +612,0,3,"Jardin, Mr. Jose Neto",male,,0,0,SOTON/O.Q. 3101305,7.05,,S +292,1,1,"Bishop, Mrs. Dickinson H (Helen Walton)",female,19.0,1,0,11967,91.0792,B49,C +293,0,2,"Levy, Mr. Rene Jacques",male,36.0,0,0,SC/Paris 2163,12.875,D,C +40,1,3,"Nicola-Yarred, Miss. Jamila",female,14.0,1,0,2651,11.2417,,C +205,1,3,"Cohen, Mr. Gurshon ""Gus""",male,18.0,0,0,A/5 3540,8.05,,S +832,1,2,"Richards, Master. George Sibley",male,0.83,1,1,29106,18.75,,S +716,0,3,"Soholt, Mr. Peter Andreas Lauritz Andersen",male,19.0,0,0,348124,7.65,F G73,S +596,0,3,"Van Impe, Mr. Jean Baptiste",male,36.0,1,1,345773,24.15,,S +344,0,2,"Sedgwick, Mr. Charles Frederick Waddington",male,25.0,0,0,244361,13.0,,S +687,0,3,"Panula, Mr. Jaako Arnold",male,14.0,4,1,3101295,39.6875,,S +662,0,3,"Badt, Mr. Mohamed",male,40.0,0,0,2623,7.225,,C +66,1,3,"Moubarek, Master. Gerios",male,,1,1,2661,15.2458,,C +820,0,3,"Skoog, Master. Karl Thorsten",male,10.0,3,2,347088,27.9,,S +865,0,2,"Gill, Mr. John William",male,24.0,0,0,233866,13.0,,S +323,1,2,"Slayter, Miss. Hilda Mary",female,30.0,0,0,234818,12.35,,Q +358,0,2,"Funk, Miss. Annie Clemmer",female,38.0,0,0,237671,13.0,,S +129,1,3,"Peter, Miss. Anna",female,,1,1,2668,22.3583,F E69,C +166,1,3,"Goldsmith, Master. Frank John William ""Frankie""",male,9.0,0,2,363291,20.525,,S +799,0,3,"Ibrahim Shawah, Mr. Yousseff",male,30.0,0,0,2685,7.2292,,C +770,0,3,"Gronnestad, Mr. Daniel Danielsen",male,32.0,0,0,8471,8.3625,,S +785,0,3,"Ali, Mr. William",male,25.0,0,0,SOTON/O.Q. 3101312,7.05,,S +399,0,2,"Pain, Dr. Alfred",male,23.0,0,0,244278,10.5,,S +746,0,1,"Crosby, Capt. Edward Gifford",male,70.0,1,1,WE/P 5735,71.0,B22,S +498,0,3,"Shellard, Mr. Frederick William",male,,0,0,C.A. 6212,15.1,,S +297,0,3,"Hanna, Mr. Mansour",male,23.5,0,0,2693,7.2292,,C +295,0,3,"Mineff, Mr. Ivan",male,24.0,0,0,349233,7.8958,,S +545,0,1,"Douglas, Mr. Walter Donald",male,50.0,1,0,PC 17761,106.425,C86,C +755,1,2,"Herman, Mrs. Samuel (Jane Laver)",female,48.0,1,2,220845,65.0,,S +305,0,3,"Williams, Mr. Howard Hugh ""Harry""",male,,0,0,A/5 2466,8.05,,S +682,1,1,"Hassab, Mr. Hammad",male,27.0,0,0,PC 17572,76.7292,D49,C +124,1,2,"Webber, Miss. Susan",female,32.5,0,0,27267,13.0,E101,S +499,0,1,"Allison, Mrs. Hudson J C (Bessie Waldo Daniels)",female,25.0,1,2,113781,151.55,C22 C26,S +870,1,3,"Johnson, Master. Harold Theodor",male,4.0,1,1,347742,11.1333,,S +72,0,3,"Goodwin, Miss. Lillian Amy",female,16.0,5,2,CA 2144,46.9,,S +120,0,3,"Andersson, Miss. Ellis Anna Maria",female,2.0,4,2,347082,31.275,,S +325,0,3,"Sage, Mr. George John Jr",male,,8,2,CA. 2343,69.55,,S +383,0,3,"Tikkanen, Mr. Juho",male,32.0,0,0,STON/O 2. 3101293,7.925,,S +628,1,1,"Longley, Miss. Gretchen Fiske",female,21.0,0,0,13502,77.9583,D9,S +744,0,3,"McNamee, Mr. Neal",male,24.0,1,0,376566,16.1,,S +684,0,3,"Goodwin, Mr. Charles Edward",male,14.0,5,2,CA 2144,46.9,,S +598,0,3,"Johnson, Mr. Alfred",male,49.0,0,0,LINE,0.0,,S +866,1,2,"Bystrom, Mrs. (Karolina)",female,42.0,0,0,236852,13.0,,S +53,1,1,"Harper, Mrs. Henry Sleeper (Myna Haxtun)",female,49.0,1,0,PC 17572,76.7292,D33,C +732,0,3,"Hassan, Mr. Houssein G N",male,11.0,0,0,2699,18.7875,,C +306,1,1,"Allison, Master. Hudson Trevor",male,0.92,1,2,113781,151.55,C22 C26,S +140,0,1,"Giglio, Mr. Victor",male,24.0,0,0,PC 17593,79.2,B86,C +814,0,3,"Andersson, Miss. Ebba Iris Alfrida",female,6.0,4,2,347082,31.275,,S +310,1,1,"Francatelli, Miss. Laura Mabel",female,30.0,0,0,PC 17485,56.9292,E36,C +71,0,2,"Jenkin, Mr. Stephen Curnow",male,32.0,0,0,C.A. 33111,10.5,,S +529,0,3,"Salonen, Mr. Johan Werner",male,39.0,0,0,3101296,7.925,,S +466,0,3,"Goncalves, Mr. Manuel Estanslas",male,38.0,0,0,SOTON/O.Q. 3101306,7.05,,S +319,1,1,"Wick, Miss. Mary Natalie",female,31.0,0,2,36928,164.8667,C7,S +259,1,1,"Ward, Miss. Anna",female,35.0,0,0,PC 17755,512.3292,,C +114,0,3,"Jussila, Miss. Katriina",female,20.0,1,0,4136,9.825,,S +625,0,3,"Bowen, Mr. David John ""Dai""",male,21.0,0,0,54636,16.1,,S +555,1,3,"Ohman, Miss. Velin",female,22.0,0,0,347085,7.775,,S +357,1,1,"Bowerman, Miss. Elsie Edith",female,22.0,0,1,113505,55.0,E33,S +837,0,3,"Pasic, Mr. Jakob",male,21.0,0,0,315097,8.6625,,S +84,0,1,"Carrau, Mr. Francisco M",male,28.0,0,0,113059,47.1,,S +184,1,2,"Becker, Master. Richard F",male,1.0,2,1,230136,39.0,F4,S +183,0,3,"Asplund, Master. Clarence Gustaf Hugo",male,9.0,4,2,347077,31.3875,,S +145,0,2,"Andrew, Mr. Edgardo Samuel",male,18.0,0,0,231945,11.5,,S +859,1,3,"Baclini, Mrs. Solomon (Latifa Qurban)",female,24.0,0,3,2666,19.2583,,C +299,1,1,"Saalfeld, Mr. Adolphe",male,,0,0,19988,30.5,C106,S +658,0,3,"Bourke, Mrs. John (Catherine)",female,32.0,1,1,364849,15.5,,Q +507,1,2,"Quick, Mrs. Frederick Charles (Jane Richards)",female,33.0,0,2,26360,26.0,,S +692,1,3,"Karun, Miss. Manca",female,4.0,0,1,349256,13.4167,,C +88,0,3,"Slocovski, Mr. Selman Francis",male,,0,0,SOTON/OQ 392086,8.05,,S +314,0,3,"Hendekovic, Mr. Ignjac",male,28.0,0,0,349243,7.8958,,S +800,0,3,"Van Impe, Mrs. Jean Baptiste (Rosalie Paula Govaert)",female,30.0,1,1,345773,24.15,,S +614,0,3,"Horgan, Mr. John",male,,0,0,370377,7.75,,Q +12,1,1,"Bonnell, Miss. Elizabeth",female,58.0,0,0,113783,26.55,C103,S +771,0,3,"Lievens, Mr. Rene Aime",male,24.0,0,0,345781,9.5,,S +365,0,3,"O'Brien, Mr. Thomas",male,,1,0,370365,15.5,,Q +876,1,3,"Najib, Miss. Adele Kiamie ""Jane""",female,15.0,0,0,2667,7.225,,C +195,1,1,"Brown, Mrs. James Joseph (Margaret Tobin)",female,44.0,0,0,PC 17610,27.7208,B4,C +594,0,3,"Bourke, Miss. Mary",female,,0,2,364848,7.75,,Q +654,1,3,"O'Leary, Miss. Hanora ""Norah""",female,,0,0,330919,7.8292,,Q +402,0,3,"Adams, Mr. John",male,26.0,0,0,341826,8.05,,S +83,1,3,"McDermott, Miss. Brigdet Delia",female,,0,0,330932,7.7875,,Q +669,0,3,"Cook, Mr. Jacob",male,43.0,0,0,A/5 3536,8.05,,S +878,0,3,"Petroff, Mr. Nedelio",male,19.0,0,0,349212,7.8958,,S +833,0,3,"Saad, Mr. Amin",male,,0,0,2671,7.2292,,C +75,1,3,"Bing, Mr. Lee",male,32.0,0,0,1601,56.4958,,S +722,0,3,"Jensen, Mr. Svend Lauritz",male,17.0,1,0,350048,7.0542,,S +251,0,3,"Reed, Mr. James George",male,,0,0,362316,7.25,,S +238,1,2,"Collyer, Miss. Marjorie ""Lottie""",female,8.0,0,2,C.A. 31921,26.25,,S +146,0,2,"Nicholls, Mr. Joseph Charles",male,19.0,1,1,C.A. 33112,36.75,,S +808,0,3,"Pettersson, Miss. Ellen Natalia",female,18.0,0,0,347087,7.775,,S +131,0,3,"Drazenoic, Mr. Jozef",male,33.0,0,0,349241,7.8958,,C +576,0,3,"Patchett, Mr. George",male,19.0,0,0,358585,14.5,,S +515,0,3,"Coleff, Mr. Satio",male,24.0,0,0,349209,7.4958,,S +847,0,3,"Sage, Mr. Douglas Bullen",male,,8,2,CA. 2343,69.55,,S +648,1,1,"Simonius-Blumer, Col. Oberst Alfons",male,56.0,0,0,13213,35.5,A26,C +443,0,3,"Petterson, Mr. Johan Emil",male,25.0,1,0,347076,7.775,,S +478,0,3,"Braund, Mr. Lewis Richard",male,29.0,1,0,3460,7.0458,,S +537,0,1,"Butt, Major. Archibald Willingham",male,45.0,0,0,113050,26.55,B38,S +169,0,1,"Baumann, Mr. John D",male,,0,0,PC 17318,25.925,,S +149,0,2,"Navratil, Mr. Michel (""Louis M Hoffman"")",male,36.5,0,2,230080,26.0,F2,S +290,1,3,"Connolly, Miss. Kate",female,22.0,0,0,370373,7.75,,Q +15,0,3,"Vestrom, Miss. Hulda Amanda Adolfina",female,14.0,0,0,350406,7.8542,,S +386,0,2,"Davies, Mr. Charles Henry",male,18.0,0,0,S.O.C. 14879,73.5,,S +811,0,3,"Alexander, Mr. William",male,26.0,0,0,3474,7.8875,,S +78,0,3,"Moutal, Mr. Rahamin Haim",male,,0,0,374746,8.05,,S +738,1,1,"Lesurer, Mr. Gustave J",male,35.0,0,0,PC 17755,512.3292,B101,C +452,0,3,"Hagland, Mr. Ingvald Olai Olsen",male,,1,0,65303,19.9667,,S +35,0,1,"Meyer, Mr. Edgar Joseph",male,28.0,1,0,PC 17604,82.1708,,C +347,1,2,"Smith, Miss. Marion Elsie",female,40.0,0,0,31418,13.0,,S +436,1,1,"Carter, Miss. Lucile Polk",female,14.0,1,2,113760,120.0,B96 B98,S +390,1,2,"Lehmann, Miss. Bertha",female,17.0,0,0,SC 1748,12.0,,C +657,0,3,"Radeff, Mr. Alexander",male,,0,0,349223,7.8958,,S +695,0,1,"Weir, Col. John",male,60.0,0,0,113800,26.55,,S +586,1,1,"Taussig, Miss. Ruth",female,18.0,0,2,110413,79.65,E68,S +384,1,1,"Holverson, Mrs. Alexander Oskar (Mary Aline Towner)",female,35.0,1,0,113789,52.0,,S +58,0,3,"Novel, Mr. Mansouer",male,28.5,0,0,2697,7.2292,,C +246,0,1,"Minahan, Dr. William Edward",male,44.0,2,0,19928,90.0,C78,Q +557,1,1,"Duff Gordon, Lady. (Lucille Christiana Sutherland) (""Mrs Morgan"")",female,48.0,1,0,11755,39.6,A16,C +605,1,1,"Homer, Mr. Harry (""Mr E Haven"")",male,35.0,0,0,111426,26.55,,C +350,0,3,"Dimic, Mr. Jovan",male,42.0,0,0,315088,8.6625,,S +659,0,2,"Eitemiller, Mr. George Floyd",male,23.0,0,0,29751,13.0,,S +415,1,3,"Sundman, Mr. Johan Julian",male,44.0,0,0,STON/O 2. 3101269,7.925,,S +713,1,1,"Taylor, Mr. Elmer Zebley",male,48.0,1,0,19996,52.0,C126,S +474,1,2,"Jerwan, Mrs. Amin S (Marie Marthe Thuillard)",female,23.0,0,0,SC/AH Basle 541,13.7917,D,C +139,0,3,"Osen, Mr. Olaf Elon",male,16.0,0,0,7534,9.2167,,S +224,0,3,"Nenkoff, Mr. Christo",male,,0,0,349234,7.8958,,S +221,1,3,"Sunderland, Mr. Victor Francis",male,16.0,0,0,SOTON/OQ 392089,8.05,,S +68,0,3,"Crease, Mr. Ernest James",male,19.0,0,0,S.P. 3464,8.1583,,S +622,1,1,"Kimball, Mr. Edwin Nelson Jr",male,42.0,1,0,11753,52.5542,D19,S +467,0,2,"Campbell, Mr. William",male,,0,0,239853,0.0,,S +525,0,3,"Kassem, Mr. Fared",male,,0,0,2700,7.2292,,C +17,0,3,"Rice, Master. Eugene",male,2.0,4,1,382652,29.125,,Q +430,1,3,"Pickard, Mr. Berk (Berk Trembisky)",male,32.0,0,0,SOTON/O.Q. 392078,8.05,E10,S +90,0,3,"Celotti, Mr. Francesco",male,24.0,0,0,343275,8.05,,S +486,0,3,"Lefebre, Miss. Jeannie",female,,3,1,4133,25.4667,,S +831,1,3,"Yasbeck, Mrs. Antoni (Selini Alexander)",female,15.0,1,0,2659,14.4542,,C +440,0,2,"Kvillner, Mr. Johan Henrik Johannesson",male,31.0,0,0,C.A. 18723,10.5,,S +244,0,3,"Maenpaa, Mr. Matti Alexanteri",male,22.0,0,0,STON/O 2. 3101275,7.125,,S +882,0,3,"Markun, Mr. Johann",male,33.0,0,0,349257,7.8958,,S +287,1,3,"de Mulder, Mr. Theodore",male,30.0,0,0,345774,9.5,,S +735,0,2,"Troupiansky, Mr. Moses Aaron",male,23.0,0,0,233639,13.0,,S +620,0,2,"Gavey, Mr. Lawrence",male,26.0,0,0,31028,10.5,,S +296,0,1,"Lewy, Mr. Ervin G",male,,0,0,PC 17612,27.7208,,C +187,1,3,"O'Brien, Mrs. Thomas (Johanna ""Hannah"" Godfrey)",female,,1,0,370365,15.5,,Q +629,0,3,"Bostandyeff, Mr. Guentcho",male,26.0,0,0,349224,7.8958,,S +123,0,2,"Nasser, Mr. Nicholas",male,32.5,1,0,237736,30.0708,,C +678,1,3,"Turja, Miss. Anna Sofia",female,18.0,0,0,4138,9.8417,,S +263,0,1,"Taussig, Mr. Emil",male,52.0,1,1,110413,79.65,E67,S +439,0,1,"Fortune, Mr. Mark",male,64.0,1,4,19950,263.0,C23 C25 C27,S +410,0,3,"Lefebre, Miss. Ida",female,,3,1,4133,25.4667,,S +497,1,1,"Eustis, Miss. Elizabeth Mussey",female,54.0,1,0,36947,78.2667,D20,C +522,0,3,"Vovk, Mr. Janko",male,22.0,0,0,349252,7.8958,,S +766,1,1,"Hogeboom, Mrs. John C (Anna Andrews)",female,51.0,1,0,13502,77.9583,D11,S +408,1,2,"Richards, Master. William Rowe",male,3.0,1,1,29106,18.75,,S +420,0,3,"Van Impe, Miss. Catharina",female,10.0,0,2,345773,24.15,,S +453,0,1,"Foreman, Mr. Benjamin Laventall",male,30.0,0,0,113051,27.75,C111,C +447,1,2,"Mellinger, Miss. Madeleine Violet",female,13.0,0,1,250644,19.5,,S +197,0,3,"Mernagh, Mr. Robert",male,,0,0,368703,7.75,,Q +227,1,2,"Mellors, Mr. William John",male,19.0,0,0,SW/PP 751,10.5,,S +852,0,3,"Svensson, Mr. Johan",male,74.0,0,0,347060,7.775,,S +763,1,3,"Barah, Mr. Hanna Assi",male,20.0,0,0,2663,7.2292,,C +257,1,1,"Thorne, Mrs. Gertrude Maybelle",female,,0,0,PC 17585,79.2,,C +407,0,3,"Widegren, Mr. Carl/Charles Peter",male,51.0,0,0,347064,7.75,,S +103,0,1,"White, Mr. Richard Frasar",male,21.0,0,1,35281,77.2875,D26,S +315,0,2,"Hart, Mr. Benjamin",male,43.0,1,1,F.C.C. 13529,26.25,,S +77,0,3,"Staneff, Mr. Ivan",male,,0,0,349208,7.8958,,S +632,0,3,"Lundahl, Mr. Johan Svensson",male,51.0,0,0,347743,7.0542,,S +750,0,3,"Connaghton, Mr. Michael",male,31.0,0,0,335097,7.75,,Q +627,0,2,"Kirkland, Rev. Charles Leonard",male,57.0,0,0,219533,12.35,,Q +96,0,3,"Shorney, Mr. Charles Joseph",male,,0,0,374910,8.05,,S +171,0,1,"Van der hoef, Mr. Wyckoff",male,61.0,0,0,111240,33.5,B19,S +881,1,2,"Shelley, Mrs. William (Imanita Parrish Hall)",female,25.0,0,1,230433,26.0,,S +95,0,3,"Coxon, Mr. Daniel",male,59.0,0,0,364500,7.25,,S +215,0,3,"Kiernan, Mr. Philip",male,,1,0,367229,7.75,,Q +39,0,3,"Vander Planke, Miss. Augusta Maria",female,18.0,2,0,345764,18.0,,S +774,0,3,"Elias, Mr. Dibo",male,,0,0,2674,7.225,,C +37,1,3,"Mamee, Mr. Hanna",male,,0,0,2677,7.2292,,C +181,0,3,"Sage, Miss. Constance Gladys",female,,8,2,CA. 2343,69.55,,S +177,0,3,"Lefebre, Master. Henry Forbes",male,,3,1,4133,25.4667,,S +812,0,3,"Lester, Mr. James",male,39.0,0,0,A/4 48871,24.15,,S +496,0,3,"Yousseff, Mr. Gerious",male,,0,0,2627,14.4583,,C +503,0,3,"O'Sullivan, Miss. Bridget Mary",female,,0,0,330909,7.6292,,Q +216,1,1,"Newell, Miss. Madeleine",female,31.0,1,0,35273,113.275,D36,C +395,1,3,"Sandstrom, Mrs. Hjalmar (Agnes Charlotta Bengtsson)",female,24.0,0,2,PP 9549,16.7,G6,S +720,0,3,"Johnson, Mr. Malkolm Joackim",male,33.0,0,0,347062,7.775,,S +213,0,3,"Perkin, Mr. John Henry",male,22.0,0,0,A/5 21174,7.25,,S +644,1,3,"Foo, Mr. Choong",male,,0,0,1601,56.4958,,S +583,0,2,"Downton, Mr. William James",male,54.0,0,0,28403,26.0,,S +132,0,3,"Coelho, Mr. Domingos Fernandeo",male,20.0,0,0,SOTON/O.Q. 3101307,7.05,,S +363,0,3,"Barbara, Mrs. (Catherine David)",female,45.0,0,1,2691,14.4542,,C +461,1,1,"Anderson, Mr. Harry",male,48.0,0,0,19952,26.55,E12,S +186,0,1,"Rood, Mr. Hugh Roscoe",male,,0,0,113767,50.0,A32,S +14,0,3,"Andersson, Mr. Anders Johan",male,39.0,1,5,347082,31.275,,S +1,0,3,"Braund, Mr. Owen Harris",male,22.0,1,0,A/5 21171,7.25,,S +694,0,3,"Saad, Mr. Khalil",male,25.0,0,0,2672,7.225,,C +476,0,1,"Clifford, Mr. George Quincy",male,,0,0,110465,52.0,A14,S +348,1,3,"Davison, Mrs. Thomas Henry (Mary E Finck)",female,,1,0,386525,16.1,,S +489,0,3,"Somerton, Mr. Francis William",male,30.0,0,0,A.5. 18509,8.05,,S +69,1,3,"Andersson, Miss. Erna Alexandra",female,17.0,4,2,3101281,7.925,,S +883,0,3,"Dahlberg, Miss. Gerda Ulrika",female,22.0,0,0,7552,10.5167,,S +18,1,2,"Williams, Mr. Charles Eugene",male,,0,0,244373,13.0,,S +31,0,1,"Uruchurtu, Don. Manuel E",male,40.0,0,0,PC 17601,27.7208,,C +619,1,2,"Becker, Miss. Marion Louise",female,4.0,2,1,230136,39.0,F4,S +526,0,3,"Farrell, Mr. James",male,40.5,0,0,367232,7.75,,Q +585,0,3,"Paulner, Mr. Uscher",male,,0,0,3411,8.7125,,C +274,0,1,"Natsch, Mr. Charles H",male,37.0,0,1,PC 17596,29.7,C118,C +715,0,2,"Greenberg, Mr. Samuel",male,52.0,0,0,250647,13.0,,S +438,1,2,"Richards, Mrs. Sidney (Emily Hocking)",female,24.0,2,3,29106,18.75,,S +193,1,3,"Andersen-Jensen, Miss. Carla Christine Nielsine",female,19.0,1,0,350046,7.8542,,S +275,1,3,"Healy, Miss. Hanora ""Nora""",female,,0,0,370375,7.75,,Q +173,1,3,"Johnson, Miss. Eleanor Ileen",female,1.0,1,1,347742,11.1333,,S +807,0,1,"Andrews, Mr. Thomas Jr",male,39.0,0,0,112050,0.0,A36,S +680,1,1,"Cardeza, Mr. Thomas Drake Martinez",male,36.0,0,1,PC 17755,512.3292,B51 B53 B55,C +304,1,2,"Keane, Miss. Nora A",female,,0,0,226593,12.35,E101,Q +370,1,1,"Aubart, Mme. Leontine Pauline",female,24.0,0,0,PC 17477,69.3,B35,C +239,0,2,"Pengelly, Mr. Frederick William",male,19.0,0,0,28665,10.5,,S +825,0,3,"Panula, Master. Urho Abraham",male,2.0,4,1,3101295,39.6875,,S +284,1,3,"Dorking, Mr. Edward Arthur",male,19.0,0,0,A/5. 10482,8.05,,S +182,0,2,"Pernot, Mr. Rene",male,,0,0,SC/PARIS 2131,15.05,,C +64,0,3,"Skoog, Master. Harald",male,4.0,3,2,347088,27.9,,S +404,0,3,"Hakkarainen, Mr. Pekka Pietari",male,28.0,1,0,STON/O2. 3101279,15.85,,S +479,0,3,"Karlsson, Mr. Nils August",male,22.0,0,0,350060,7.5208,,S +618,0,3,"Lobb, Mrs. William Arthur (Cordelia K Stanlick)",female,26.0,1,0,A/5. 3336,16.1,,S +3,1,3,"Heikkinen, Miss. Laina",female,26.0,0,0,STON/O2. 3101282,7.925,,S +337,0,1,"Pears, Mr. Thomas Clinton",male,29.0,1,0,113776,66.6,C2,S +764,1,1,"Carter, Mrs. William Ernest (Lucile Polk)",female,36.0,1,2,113760,120.0,B96 B98,S +696,0,2,"Chapman, Mr. Charles Henry",male,52.0,0,0,248731,13.5,,S +783,0,1,"Long, Mr. Milton Clyde",male,29.0,0,0,113501,30.0,D6,S +318,0,2,"Moraweck, Dr. Ernest",male,54.0,0,0,29011,14.0,,S +706,0,2,"Morley, Mr. Henry Samuel (""Mr Henry Marshall"")",male,39.0,0,0,250655,26.0,,S +432,1,3,"Thorneycroft, Mrs. Percival (Florence Kate White)",female,,1,0,376564,16.1,,S +50,0,3,"Arnold-Franchi, Mrs. Josef (Josefine Franchi)",female,18.0,1,0,349237,17.8,,S +136,0,2,"Richard, Mr. Emile",male,23.0,0,0,SC/PARIS 2133,15.0458,,C +889,0,3,"Johnston, Miss. Catherine Helen ""Carrie""",female,,1,2,W./C. 6607,23.45,,S +604,0,3,"Torber, Mr. Ernst William",male,44.0,0,0,364511,8.05,,S +5,0,3,"Allen, Mr. William Henry",male,35.0,0,0,373450,8.05,,S +613,1,3,"Murphy, Miss. Margaret Jane",female,,1,0,367230,15.5,,Q +724,0,2,"Hodges, Mr. Henry Price",male,50.0,0,0,250643,13.0,,S +758,0,2,"Bailey, Mr. Percy Andrew",male,18.0,0,0,29108,11.5,,S +142,1,3,"Nysten, Miss. Anna Sofia",female,22.0,0,0,347081,7.75,,S +416,0,3,"Meek, Mrs. Thomas (Annie Louise Rowley)",female,,0,0,343095,8.05,,S +668,0,3,"Rommetvedt, Mr. Knud Paust",male,,0,0,312993,7.775,,S +387,0,3,"Goodwin, Master. Sidney Leonard",male,1.0,5,2,CA 2144,46.9,,S +87,0,3,"Ford, Mr. William Neal",male,16.0,1,3,W./C. 6608,34.375,,S +94,0,3,"Dean, Mr. Bertram Frank",male,26.0,1,2,C.A. 2315,20.575,,S +650,1,3,"Stanley, Miss. Amy Zillah Elsie",female,23.0,0,0,CA. 2314,7.55,,S +508,1,1,"Bradley, Mr. George (""George Arthur Brayton"")",male,,0,0,111427,26.55,,S +571,1,2,"Harris, Mr. George",male,62.0,0,0,S.W./PP 752,10.5,,S +317,1,2,"Kantor, Mrs. Sinai (Miriam Sternin)",female,24.0,1,0,244367,26.0,,S +229,0,2,"Fahlstrom, Mr. Arne Jonas",male,18.0,0,0,236171,13.0,,S +656,0,2,"Hickman, Mr. Leonard Mark",male,24.0,2,0,S.O.C. 14879,73.5,,S +281,0,3,"Duane, Mr. Frank",male,65.0,0,0,336439,7.75,,Q +753,0,3,"Vande Velde, Mr. Johannes Joseph",male,33.0,0,0,345780,9.5,,S +803,1,1,"Carter, Master. William Thornton II",male,11.0,1,2,113760,120.0,B96 B98,S +527,1,2,"Ridsdale, Miss. Lucy",female,50.0,0,0,W./C. 14258,10.5,,S +739,0,3,"Ivanoff, Mr. Kanio",male,,0,0,349201,7.8958,,S +579,0,3,"Caram, Mrs. Joseph (Maria Elias)",female,,1,0,2689,14.4583,,C +54,1,2,"Faunthorpe, Mrs. Lizzie (Elizabeth Anne Wilkinson)",female,29.0,1,0,2926,26.0,,S +867,1,2,"Duran y More, Miss. Asuncion",female,27.0,1,0,SC/PARIS 2149,13.8583,,C +351,0,3,"Odahl, Mr. Nils Martin",male,23.0,0,0,7267,9.225,,S +80,1,3,"Dowdell, Miss. Elizabeth",female,30.0,0,0,364516,12.475,,S +856,1,3,"Aks, Mrs. Sam (Leah Rosen)",female,18.0,0,1,392091,9.35,,S +872,1,1,"Beckwith, Mrs. Richard Leonard (Sallie Monypeny)",female,47.0,1,1,11751,52.5542,D35,S +836,1,1,"Compton, Miss. Sara Rebecca",female,39.0,1,1,PC 17756,83.1583,E49,C +793,0,3,"Sage, Miss. Stella Anna",female,,8,2,CA. 2343,69.55,,S +521,1,1,"Perreault, Miss. Anne",female,30.0,0,0,12749,93.5,B73,S diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index 06596e05a..eb67021a5 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -142,7 +142,7 @@ "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.sh\n## Development Code\n```python\necho 'Hello World'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\necho Hello World\n## Running Output\nstandard output: \n```text\nHello World\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "The running result indicates no error. The code works fine.\n\n## File To Rewrite:\nNo file needs to be rewritten.\n\n## Status:\nPASS\n\n## Send To:\nNoOne\n\n---", "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.py\n## Development Code\n```python\npython -c \"print(1/0)\"\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\npython -c print(1/0)\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\nTraceback (most recent call last):\n File \"\", line 1, in \nZeroDivisionError: division by zero\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\nThe error is caused by the development code in file a.py, which attempts to divide by zero. To fix the error, the development code in a.py should be modified to handle the ZeroDivisionError, for example by using a try-except block.\n\nFile To Rewrite:\na.py\n\nStatus:\nFAIL\n\nSend To:\nEngineer\n---", "\nNOTICE\n1. Role: You are a Development Engineer or QA engineer;\n2. Task: You received this message from another Development Engineer or QA engineer who ran or tested your code. \nBased on the message, first, figure out your own role, i.e. Engineer or QaEngineer,\nthen rewrite the development code or the test code based on your role, the error, and the summary, such that all bugs are fixed and the code performs well.\nAttention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script and triple quotes.\nThe message is as follows:\n# Legacy Code\n```python\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n if self.score > 21 and any(card.rank == 'A' for card in self.hand):\n self.score -= 10\n return self.score\n\n```\n---\n# Unit Test Code\n```python\n\nimport unittest\nfrom blackjack_game.player import Player\nfrom blackjack_game.deck import Deck\nfrom blackjack_game.card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n\n\n```\n---\n# Console logs\n```text\n\n```\n---\nNow you should start rewriting the code:\n## file name of the code to rewrite: Write code with triple quote. Do your best to implement THIS IN ONLY ONE FILE.\n": "```python\n## Development Engineer\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n ace_count = sum(1 for card in self.hand if card.rank == 'A')\n while self.score > 21 and ace_count > 0:\n self.score -= 10\n ace_count -= 1\n return self.score\n```\n\n```python\n## QaEngineer\n\nimport unittest\nfrom player import Player\nfrom deck import Deck\nfrom card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n```", - "\n # Context:\n user: \n## User Requirement\nThis is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Perform exploratory data analysis on the train dataset to understand the features and target variable.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Preprocess the data by handling missing values, encoding categorical variables, and scaling features.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Conduct feature engineering to create new features that may help improve model performance.\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Select and train a machine learning model using the processed train dataset.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Evaluate the model's accuracy using the eval dataset and report the results.\"\n }\n]\n```", + "\n # Context:\n user: \n## User Requirement\nThis is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Perform exploratory data analysis on the train dataset to understand the features and target variable.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Preprocess the data by handling missing values, encoding categorical variables, and scaling features.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Conduct feature engineering to create new features that may help improve model performance.\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Select and train a machine learning model using the processed train dataset.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Evaluate the model's accuracy using the eval dataset and report the results.\"\n }\n]\n```", "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Perform exploratory data analysis on the train dataset to understand the features and target variable.\\nTask 2: Preprocess the data by handling missing values, encoding categorical variables, and scaling features.\\nTask 3: Conduct feature engineering to create new features that may help improve model performance.\\nTask 4: Select and train a machine learning model using the processed train dataset.\\nTask 5: Evaluate the model's accuracy using the eval dataset and report the results.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { "task_type": [ "eda", @@ -152,10 +152,10 @@ "model_evaluate" ] }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n\\n```end\\n\\n## Current Task\\nPerform exploratory data analysis on the train dataset to understand the features and target variable.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { - "code": "# Step 1: Load the training dataset\nimport pandas as pd\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\n\n# Step 2: Display the first few rows of the dataset\ndisplay(train_data.head())\n\n# Step 3: Get a summary of the dataset\ntrain_data.info()\n\n# Step 4: Describe the numerical features\ntrain_data.describe()\n\n# Step 5: Describe the categorical features\ntrain_data.describe(include=['O'])\n\n# Step 6: Check for missing values\nmissing_values = train_data.isnull().sum()\nmissing_values[missing_values > 0]\n\n# Step 7: Visualize the distribution of the target variable 'Survived'\nimport seaborn as sns\nimport matplotlib.pyplot as plt\nsns.countplot(x='Survived', data=train_data)\nplt.title('Distribution of Survival')\nplt.show()\n\n# Step 8: Explore the relationship between features and the target variable\n# Visualize the survival rate by sex\nsns.barplot(x='Sex', y='Survived', data=train_data)\nplt.title('Survival Rate by Sex')\nplt.show()\n\n# Visualize the survival rate by class\nsns.barplot(x='Pclass', y='Survived', data=train_data)\nplt.title('Survival Rate by Class')\nplt.show()\n\n# Visualize the survival rate by embarkation port\nsns.barplot(x='Embarked', y='Survived', data=train_data)\nplt.title('Survival Rate by Embarkation Port')\nplt.show()" + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n\\n```end\\n\\n## Current Task\\nPerform exploratory data analysis on the train dataset to understand the features and target variable.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { + "code": "# Step 1: Load the training dataset\nimport pandas as pd\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\n\n# Step 2: Display the first few rows of the dataset\ndisplay(train_data.head())\n\n# Step 3: Get a summary of the dataset\ntrain_data.info()\n\n# Step 4: Describe the numerical features\ntrain_data.describe()\n\n# Step 5: Describe the categorical features\ntrain_data.describe(include=['O'])\n\n# Step 6: Check for missing values\nmissing_values = train_data.isnull().sum()\nmissing_values[missing_values > 0]\n\n# Step 7: Visualize the distribution of the target variable 'Survived'\nimport seaborn as sns\nimport matplotlib.pyplot as plt\nsns.countplot(x='Survived', data=train_data)\nplt.title('Distribution of Survival')\nplt.show()\n\n# Step 8: Explore the relationship between features and the target variable\n# Visualize the survival rate by sex\nsns.barplot(x='Sex', y='Survived', data=train_data)\nplt.title('Survival Rate by Sex')\nplt.show()\n\n# Visualize the survival rate by class\nsns.barplot(x='Pclass', y='Survived', data=train_data)\nplt.title('Survival Rate by Class')\nplt.show()\n\n# Visualize the survival rate by embarkation port\nsns.barplot(x='Embarked', y='Survived', data=train_data)\nplt.title('Survival Rate by Embarkation Port')\nplt.show()" }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { "code": "from metagpt.tools.libs.data_preprocess import get_column_info\n\ncolumn_info = get_column_info(train_data)\nprint(\"column_info\")\nprint(column_info)" }, "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nPreprocess the data by handling missing values, encoding categorical variables, and scaling features.\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Completing missing values with simple strategies'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { @@ -163,10 +163,10 @@ "FillMissingValue" ] }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n```end\\n\\n## Current Task\\nPreprocess the data by handling missing values, encoding categorical variables, and scaling features.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Name', 'Sex', 'Ticket', 'Cabin', 'Embarked'], 'Numeric': ['PassengerId', 'Survived', 'Pclass', 'Age', 'SibSp', 'Parch', 'Fare'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}, 'strategy': {'type': 'str', 'description': 'the imputation strategy, notice mean/median can only be used for numeric features', 'default': 'mean', 'enum': ['mean', 'median', 'most_frequent', 'constant']}, 'fill_value': {'type': 'int', 'description': 'fill_value is used to replace all occurrences of missing_values', 'default': None}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { - "code": "# Step 1: Handle missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create copies of the datasets to avoid changing the original data\ntrain_processed = train_data.copy()\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\n\n# Fill missing values for 'Age' with median and 'Embarked' with most frequent value\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\ntrain_processed = fill_missing_age.fit_transform(train_processed)\neval_processed = fill_missing_age.transform(eval_data)\n\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\neval_processed = fill_missing_embarked.transform(eval_data)\n\n# Since 'Cabin' has a lot of missing values, we'll create a binary feature indicating whether cabin info is missing\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\n\n# Step 2: Encode categorical variables\nfrom sklearn.preprocessing import LabelEncoder\n\nlabel_encoder = LabelEncoder()\n\n# Encode 'Sex'\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\n\n# Encode 'Embarked'\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\n\n# Step 3: Scale features\nfrom sklearn.preprocessing import StandardScaler\n\nscaler = StandardScaler()\n\n# Scale 'Age' and 'Fare'\nfor feature in ['Age', 'Fare']:\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\n\n# Display the first few rows of the processed train dataset\ndisplay(train_processed.head())" + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n```end\\n\\n## Current Task\\nPreprocess the data by handling missing values, encoding categorical variables, and scaling features.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Name', 'Sex', 'Ticket', 'Cabin', 'Embarked'], 'Numeric': ['PassengerId', 'Survived', 'Pclass', 'Age', 'SibSp', 'Parch', 'Fare'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}, 'strategy': {'type': 'str', 'description': 'the imputation strategy, notice mean/median can only be used for numeric features', 'default': 'mean', 'enum': ['mean', 'median', 'most_frequent', 'constant']}, 'fill_value': {'type': 'int', 'description': 'fill_value is used to replace all occurrences of missing_values', 'default': None}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { + "code": "# Step 1: Handle missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create copies of the datasets to avoid changing the original data\ntrain_processed = train_data.copy()\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\n\n# Fill missing values for 'Age' with median and 'Embarked' with most frequent value\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\ntrain_processed = fill_missing_age.fit_transform(train_processed)\neval_processed = fill_missing_age.transform(eval_data)\n\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\neval_processed = fill_missing_embarked.transform(eval_data)\n\n# Since 'Cabin' has a lot of missing values, we'll create a binary feature indicating whether cabin info is missing\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\n\n# Step 2: Encode categorical variables\nfrom sklearn.preprocessing import LabelEncoder\n\nlabel_encoder = LabelEncoder()\n\n# Encode 'Sex'\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\n\n# Encode 'Embarked'\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\n\n# Step 3: Scale features\nfrom sklearn.preprocessing import StandardScaler\n\nscaler = StandardScaler()\n\n# Scale 'Age' and 'Fare'\nfor feature in ['Age', 'Fare']:\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\n\n# Display the first few rows of the processed train dataset\ndisplay(train_processed.head())" }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { "code": "from metagpt.tools.libs.data_preprocess import get_column_info\ncolumn_info = get_column_info(train_processed)\nprint(\"column_info\")\nprint(column_info)" }, "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nConduct feature engineering to create new features that may help improve model performance.\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'CatCross': 'Add pairwise crossed features and convert them to numerical features.'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { @@ -174,20 +174,20 @@ "CatCross" ] }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n```end\\n\\n## Current Task\\nConduct feature engineering to create new features that may help improve model performance.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Name', 'Ticket', 'Cabin'], 'Numeric': ['PassengerId', 'Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about feature engineering. when performing it, please adhere to the following principles:\\n- Generate as diverse features as possible to improve the model's performance step-by-step. \\n- If potential impactful features are not included in 'Code Steps', add new steps to generate them.\\n- Avoid creating redundant or excessively numerous features in one step.\\n- Exclude ID columns from feature generation and remove them.\\n- Each step do feature engineering to train, must do same for test separately at the same time.\\n- Avoid using the label column to create features, except for cat encoding.\\n- Use the data from previous task result if exist, do not mock or reload data yourself.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'CatCross': {'type': 'class', 'description': 'Add pairwise crossed features and convert them to numerical features.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'cols': {'type': 'list', 'description': 'Columns to be pairwise crossed, at least 2 columns.'}, 'max_cat_num': {'type': 'int', 'description': 'Maximum unique categories per crossed feature.', 'default': 100}}}, 'required': ['cols']}, 'fit': {'description': 'Fit the CatCross model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/feature_engineering.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n```end\\n\\n## Current Task\\nConduct feature engineering to create new features that may help improve model performance.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Name', 'Ticket', 'Cabin'], 'Numeric': ['PassengerId', 'Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about feature engineering. when performing it, please adhere to the following principles:\\n- Generate as diverse features as possible to improve the model's performance step-by-step. \\n- If potential impactful features are not included in 'Code Steps', add new steps to generate them.\\n- Avoid creating redundant or excessively numerous features in one step.\\n- Exclude ID columns from feature generation and remove them.\\n- Each step do feature engineering to train, must do same for test separately at the same time.\\n- Avoid using the label column to create features, except for cat encoding.\\n- Use the data from previous task result if exist, do not mock or reload data yourself.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'CatCross': {'type': 'class', 'description': 'Add pairwise crossed features and convert them to numerical features.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'cols': {'type': 'list', 'description': 'Columns to be pairwise crossed, at least 2 columns.'}, 'max_cat_num': {'type': 'int', 'description': 'Maximum unique categories per crossed feature.', 'default': 100}}}, 'required': ['cols']}, 'fit': {'description': 'Fit the CatCross model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/feature_engineering.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { "code": "# Step 1: Create a feature for the number of family members on board\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\n\n# Step 2: Create a feature to indicate whether a passenger was alone or not\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\n\n# Step 3: Create a feature for the title extracted from the passenger's name\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\.', expand=False)\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\.', expand=False)\n\n# Step 4: Create a feature for the deck extracted from the cabin number\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\n\n# Step 5: Create a feature for the ticket class and sex cross\n# Tools used: ['CatCross']\nfrom metagpt.tools.libs.feature_engineering import CatCross\n\n# Initialize CatCross with the desired columns\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\n\n# Fit and transform the train data\ntrain_processed = cat_cross.fit_transform(train_processed)\n\n# Transform the eval data\neval_processed = cat_cross.transform(eval_processed)\n\n# Remove unnecessary columns\nfor df in [train_processed, eval_processed]:\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)" }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { "code": "from metagpt.tools.libs.data_preprocess import get_column_info\ncolumn_info = get_column_info(train_processed)\nprint(\"column_info\")\nprint(column_info)" }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n## Current Task\\nSelect and train a machine learning model using the processed train dataset.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Title', 'Deck'], 'Numeric': ['Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind', 'Family_Size', 'Is_Alone', 'Pclass_Sex'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about training a model, please ensure high performance:\\n- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as XGBoost, CatBoost, etc.\\n- If non-numeric columns exist, perform label encode together with all steps.\\n- Use the data from previous task result directly, do not mock or reload data yourself.\\n- Set suitable hyperparameters for the model, make metrics as high as possible.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n## Current Task\\nSelect and train a machine learning model using the processed train dataset.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Title', 'Deck'], 'Numeric': ['Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind', 'Family_Size', 'Is_Alone', 'Pclass_Sex'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about training a model, please ensure high performance:\\n- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as XGBoost, CatBoost, etc.\\n- If non-numeric columns exist, perform label encode together with all steps.\\n- Use the data from previous task result directly, do not mock or reload data yourself.\\n- Set suitable hyperparameters for the model, make metrics as high as possible.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { "code": "from sklearn.model_selection import train_test_split\nfrom sklearn.metrics import accuracy_score\nfrom xgboost import XGBClassifier\n\n# Step 1: Prepare the data\n# Separate features and target variable\nX = train_processed.drop('Survived', axis=1)\ny = train_processed['Survived']\n\n# Step 2: Split the data into training and validation sets\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\n\n# Step 3: Initialize and train the XGBoost model\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\nmodel.fit(X_train, y_train)\n\n# Step 4: Evaluate the model\ny_pred = model.predict(X_val)\naccuracy = accuracy_score(y_val, y_pred)\nprint(f'Validation Accuracy: {accuracy:.4f}')" }, - "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\n[user: \\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n## Current Task\\nSelect and train a machine learning model using the processed train dataset.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Title', 'Deck'], 'Numeric': ['Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind', 'Family_Size', 'Is_Alone', 'Pclass_Sex'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about training a model, please ensure high performance:\\n- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as XGBoost, CatBoost, etc.\\n- If non-numeric columns exist, perform label encode together with all steps.\\n- Use the data from previous task result directly, do not mock or reload data yourself.\\n- Set suitable hyperparameters for the model, make metrics as high as possible.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n]\\n\\n[previous impl]\\nfrom sklearn.model_selection import train_test_split\\nfrom sklearn.metrics import accuracy_score\\nfrom xgboost import XGBClassifier\\n\\n# Step 1: Prepare the data\\n# Separate features and target variable\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\n\\n# Step 2: Split the data into training and validation sets\\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\\n\\n# Step 3: Initialize and train the XGBoost model\\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X_train, y_train)\\n\\n# Step 4: Evaluate the model\\ny_pred = model.predict(X_val)\\naccuracy = accuracy_score(y_val, y_pred)\\nprint(f'Validation Accuracy: {accuracy:.4f}')\\n[runtime Error]\\n[assistant: from sklearn.model_selection import train_test_split\\nfrom sklearn.metrics import accuracy_score\\nfrom xgboost import XGBClassifier\\n\\n# Step 1: Prepare the data\\n# Separate features and target variable\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\n\\n# Step 2: Split the data into training and validation sets\\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\\n\\n# Step 3: Initialize and train the XGBoost model\\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X_train, y_train)\\n\\n# Step 4: Evaluate the model\\ny_pred = model.predict(X_val)\\naccuracy = accuracy_score(y_val, y_pred)\\nprint(f'Validation Accuracy: {accuracy:.4f}'), user: Executed code failed, please reflect the cause of bug and then debug. Truncated to show only last 2000 characters\\n= self._temporary_data\\n 622 else:\\n--> 623 new, cat_codes, feature_names, feature_types = _proxy_transform(\\n 624 data,\\n 625 feature_names,\\n 626 feature_types,\\n 627 self._enable_categorical,\\n 628 )\\n 629 # Stage the data, meta info are copied inside C++ MetaInfo.\\n 630 self._temporary_data = (new, cat_codes, feature_names, feature_types)\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:1315, in _proxy_transform(data, feature_names, feature_types, enable_categorical)\\n 1313 data = pd.DataFrame(data)\\n 1314 if _is_pandas_df(data):\\n-> 1315 arr, feature_names, feature_types = _transform_pandas_df(\\n 1316 data, enable_categorical, feature_names, feature_types\\n 1317 )\\n 1318 arr, _ = _ensure_np_dtype(arr, arr.dtype)\\n 1319 return arr, None, feature_names, feature_types\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:490, in _transform_pandas_df(data, enable_categorical, feature_names, feature_types, meta, meta_type)\\n 483 for dtype in data.dtypes:\\n 484 if not (\\n 485 (dtype.name in _pandas_dtype_mapper)\\n 486 or is_pd_sparse_dtype(dtype)\\n 487 or (is_pd_cat_dtype(dtype) and enable_categorical)\\n 488 or is_pa_ext_dtype(dtype)\\n 489 ):\\n--> 490 _invalid_dataframe_dtype(data)\\n 491 if is_pa_ext_dtype(dtype):\\n 492 pyarrow_extension = True\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:308, in _invalid_dataframe_dtype(data)\\n 306 type_err = \\\"DataFrame.dtypes for data must be int, float, bool or category.\\\"\\n 307 msg = f\\\"\\\"\\\"{type_err} {_ENABLE_CAT_ERR} {err}\\\"\\\"\\\"\\n--> 308 raise ValueError(msg)\\n\\nValueError: DataFrame.dtypes for data must be int, float, bool or category. When categorical type is supplied, The experimental DMatrix parameter`enable_categorical` must be set to `True`. Invalid columns:Title: object, Deck: object\\n\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { + "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\n[user: \\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n## Current Task\\nSelect and train a machine learning model using the processed train dataset.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Title', 'Deck'], 'Numeric': ['Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind', 'Family_Size', 'Is_Alone', 'Pclass_Sex'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about training a model, please ensure high performance:\\n- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as XGBoost, CatBoost, etc.\\n- If non-numeric columns exist, perform label encode together with all steps.\\n- Use the data from previous task result directly, do not mock or reload data yourself.\\n- Set suitable hyperparameters for the model, make metrics as high as possible.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n]\\n\\n[previous impl]\\nfrom sklearn.model_selection import train_test_split\\nfrom sklearn.metrics import accuracy_score\\nfrom xgboost import XGBClassifier\\n\\n# Step 1: Prepare the data\\n# Separate features and target variable\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\n\\n# Step 2: Split the data into training and validation sets\\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\\n\\n# Step 3: Initialize and train the XGBoost model\\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X_train, y_train)\\n\\n# Step 4: Evaluate the model\\ny_pred = model.predict(X_val)\\naccuracy = accuracy_score(y_val, y_pred)\\nprint(f'Validation Accuracy: {accuracy:.4f}')\\n[runtime Error]\\n[assistant: from sklearn.model_selection import train_test_split\\nfrom sklearn.metrics import accuracy_score\\nfrom xgboost import XGBClassifier\\n\\n# Step 1: Prepare the data\\n# Separate features and target variable\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\n\\n# Step 2: Split the data into training and validation sets\\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\\n\\n# Step 3: Initialize and train the XGBoost model\\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X_train, y_train)\\n\\n# Step 4: Evaluate the model\\ny_pred = model.predict(X_val)\\naccuracy = accuracy_score(y_val, y_pred)\\nprint(f'Validation Accuracy: {accuracy:.4f}'), user: Executed code failed, please reflect the cause of bug and then debug. Truncated to show only last 2000 characters\\n= self._temporary_data\\n 622 else:\\n--> 623 new, cat_codes, feature_names, feature_types = _proxy_transform(\\n 624 data,\\n 625 feature_names,\\n 626 feature_types,\\n 627 self._enable_categorical,\\n 628 )\\n 629 # Stage the data, meta info are copied inside C++ MetaInfo.\\n 630 self._temporary_data = (new, cat_codes, feature_names, feature_types)\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:1315, in _proxy_transform(data, feature_names, feature_types, enable_categorical)\\n 1313 data = pd.DataFrame(data)\\n 1314 if _is_pandas_df(data):\\n-> 1315 arr, feature_names, feature_types = _transform_pandas_df(\\n 1316 data, enable_categorical, feature_names, feature_types\\n 1317 )\\n 1318 arr, _ = _ensure_np_dtype(arr, arr.dtype)\\n 1319 return arr, None, feature_names, feature_types\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:490, in _transform_pandas_df(data, enable_categorical, feature_names, feature_types, meta, meta_type)\\n 483 for dtype in data.dtypes:\\n 484 if not (\\n 485 (dtype.name in _pandas_dtype_mapper)\\n 486 or is_pd_sparse_dtype(dtype)\\n 487 or (is_pd_cat_dtype(dtype) and enable_categorical)\\n 488 or is_pa_ext_dtype(dtype)\\n 489 ):\\n--> 490 _invalid_dataframe_dtype(data)\\n 491 if is_pa_ext_dtype(dtype):\\n 492 pyarrow_extension = True\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:308, in _invalid_dataframe_dtype(data)\\n 306 type_err = \\\"DataFrame.dtypes for data must be int, float, bool or category.\\\"\\n 307 msg = f\\\"\\\"\\\"{type_err} {_ENABLE_CAT_ERR} {err}\\\"\\\"\\\"\\n--> 308 raise ValueError(msg)\\n\\nValueError: DataFrame.dtypes for data must be int, float, bool or category. When categorical type is supplied, The experimental DMatrix parameter`enable_categorical` must be set to `True`. Invalid columns:Title: object, Deck: object\\n\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { "reflection": "The error message indicates that the XGBoost model cannot handle columns with 'object' data types directly. The columns 'Title' and 'Deck' are of 'object' type and need to be encoded as numeric values before they can be used for training the XGBoost model. The error also suggests that if we want to use categorical data, we must set the `enable_categorical` parameter to `True` when creating the DMatrix. However, a more common practice is to convert these object types to numeric types using techniques such as label encoding or one-hot encoding. To resolve this issue, we should encode the 'Title' and 'Deck' columns using label encoding before training the model. Additionally, we should ensure that any other non-numeric columns are also converted to numeric types if they exist.", "improved_impl": "from sklearn.preprocessing import LabelEncoder\n\n# Step 1: Encode non-numeric columns\nencoder = LabelEncoder()\ntrain_processed['Title'] = encoder.fit_transform(train_processed['Title'])\neval_processed['Title'] = encoder.transform(eval_processed['Title'])\ntrain_processed['Deck'] = encoder.fit_transform(train_processed['Deck'].astype(str))\neval_processed['Deck'] = encoder.transform(eval_processed['Deck'].astype(str))\n\n# Step 2: Prepare the data\n# Separate features and target variable\nX = train_processed.drop('Survived', axis=1)\ny = train_processed['Survived']\n\n# Step 3: Train the XGBoost model\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\nmodel.fit(X, y)\n\n# Step 4: Evaluate the model on evaluation data\nX_eval = eval_processed.drop('Survived', axis=1)\ny_eval = eval_processed['Survived']\ny_pred_eval = model.predict(X_eval)\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')" }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv', eval data path: '/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('/Users/gary/Desktop/data_agents_opt/data/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n\\nfrom sklearn.preprocessing import LabelEncoder\\nencoder = LabelEncoder()\\ntrain_processed['Title'] = encoder.fit_transform(train_processed['Title'])\\neval_processed['Title'] = encoder.transform(eval_processed['Title'])\\ntrain_processed['Deck'] = encoder.fit_transform(train_processed['Deck'].astype(str))\\neval_processed['Deck'] = encoder.transform(eval_processed['Deck'].astype(str))\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X, y)\\nX_eval = eval_processed.drop('Survived', axis=1)\\ny_eval = eval_processed['Survived']\\ny_pred_eval = model.predict(X_eval)\\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')\\n```end\\n\\n## Current Task\\nEvaluate the model's accuracy using the eval dataset and report the results.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about evaluating a model, please note the following:\\n- Ensure that the evaluated data is same processed as the training data. If not, remember use object in 'Done Tasks' to transform the data.\\n- Use trained model from previous task result directly, do not mock or reload model yourself.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n\\nfrom sklearn.preprocessing import LabelEncoder\\nencoder = LabelEncoder()\\ntrain_processed['Title'] = encoder.fit_transform(train_processed['Title'])\\neval_processed['Title'] = encoder.transform(eval_processed['Title'])\\ntrain_processed['Deck'] = encoder.fit_transform(train_processed['Deck'].astype(str))\\neval_processed['Deck'] = encoder.transform(eval_processed['Deck'].astype(str))\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X, y)\\nX_eval = eval_processed.drop('Survived', axis=1)\\ny_eval = eval_processed['Survived']\\ny_pred_eval = model.predict(X_eval)\\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')\\n```end\\n\\n## Current Task\\nEvaluate the model's accuracy using the eval dataset and report the results.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about evaluating a model, please note the following:\\n- Ensure that the evaluated data is same processed as the training data. If not, remember use object in 'Done Tasks' to transform the data.\\n- Use trained model from previous task result directly, do not mock or reload model yourself.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { "code": "# Evaluate the model's accuracy using the eval dataset\nfrom sklearn.metrics import accuracy_score\n\n# Predict the target for the evaluation set\ny_pred_eval = model.predict(X_eval)\n\n# Calculate the accuracy for the evaluation set\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\n\n# Report the accuracy\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')" }, "\n # Context:\n user: \n## User Requirement\nRun data analysis on sklearn Iris dataset, include a plot\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Load the sklearn Iris dataset.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis on the Iris dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Create a plot visualizing the Iris dataset features.\"\n }\n]\n```", diff --git a/tests/metagpt/roles/test_ml_engineer.py b/tests/metagpt/roles/test_ml_engineer.py index 23570b0f1..1373213a5 100644 --- a/tests/metagpt/roles/test_ml_engineer.py +++ b/tests/metagpt/roles/test_ml_engineer.py @@ -1,6 +1,5 @@ import pytest -from metagpt.const import DATA_PATH from metagpt.logs import logger from metagpt.roles.ml_engineer import MLEngineer @@ -11,21 +10,12 @@ def test_mle_init(): @pytest.mark.asyncio -@pytest.mark.parametrize("use_tools", [(True)]) -async def test_code_interpreter(use_tools): - # requirement = "Run data analysis on sklearn Iris dataset, include a plot" - # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - data_path = f"{DATA_PATH}/titanic" +async def test_ml_engineer(): + data_path = "tests/data/ml_datasets/titanic" requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" - # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." - # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" - # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report AUC Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." - # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" - # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." tools = ["FillMissingValue", "CatCross", "dummy_tool"] - mle = MLEngineer(goal=requirement, auto_run=True, use_tools=use_tools, tools=tools) + mle = MLEngineer(goal=requirement, auto_run=True, use_tools=True, tools=tools) rsp = await mle.run(requirement) logger.info(rsp) assert len(rsp.content) > 0 From a9ef85b2824a370f138e3135f22b2f93010ce2ee Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 31 Jan 2024 11:35:05 +0800 Subject: [PATCH 519/637] add gptv config --- metagpt/config2.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/metagpt/config2.py b/metagpt/config2.py index 5a556cc52..dc53ee661 100644 --- a/metagpt/config2.py +++ b/metagpt/config2.py @@ -81,6 +81,8 @@ class Config(CLIParams, YamlModel): AZURE_TTS_SUBSCRIPTION_KEY: str = "" AZURE_TTS_REGION: str = "" mermaid_engine: str = "nodejs" + OPENAI_VISION_MODEL: str = "gpt-4-vision-preview" + VISION_MAX_TOKENS: int = 4096 @classmethod def from_home(cls, path): From 7cead197013b9e805c1b322527b3c33ffb79e3f1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 31 Jan 2024 13:35:48 +0800 Subject: [PATCH 520/637] fix: add arg for OpenAILLM in test_get_choice_function_arguments_for_aask_code. --- tests/metagpt/provider/test_openai.py | 90 +++++++++++++-------------- 1 file changed, 43 insertions(+), 47 deletions(-) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index df9355f7c..1698518f5 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -36,6 +36,48 @@ async def test_speech_to_text(): assert "你好" == resp.text +@pytest.fixture +def tool_calls_rsp(): + function_rsps = [ + Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"}', name="execute"), + Function(arguments='{\n"language": "python",\n"code": \'print("hello world")\'}', name="execute"), + Function(arguments='{\n"language": \'python\',\n"code": "print(\'hello world\')"}', name="execute"), + Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"}', name="execute"), + Function(arguments='{\n"language": "python",\n"code": ```print("hello world")```}', name="execute"), + Function(arguments='{\n"language": "python",\n"code": """print("hello world")"""}', name="execute"), + Function(arguments='\nprint("hello world")\\n', name="execute"), + # only `{` in arguments + Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), + # no `{`, `}` in arguments + Function(arguments='\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), + ] + tool_calls = [ + ChatCompletionMessageToolCall(type="function", id=f"call_{i}", function=f) for i, f in enumerate(function_rsps) + ] + messages = [ChatCompletionMessage(content=None, role="assistant", tool_calls=[t]) for t in tool_calls] + # 添加一个纯文本响应 + messages.append( + ChatCompletionMessage(content="Completed a python code for hello world!", role="assistant", tool_calls=None) + ) + # 添加 openai tool calls respond bug, code 出现在ChatCompletionMessage.content中 + messages.extend( + [ + ChatCompletionMessage(content="```python\nprint('hello world')```", role="assistant", tool_calls=None), + ChatCompletionMessage(content="'''python\nprint('hello world')'''", role="assistant", tool_calls=None), + ChatCompletionMessage(content='"""python\nprint(\'hello world\')"""', role="assistant", tool_calls=None), + ChatCompletionMessage(content="'''python\nprint(\"hello world\")'''", role="assistant", tool_calls=None), + ChatCompletionMessage(content="```python\nprint('hello world')```", role="assistant", tool_calls=None), + ] + ) + choices = [ + Choice(finish_reason="tool_calls", logprobs=None, index=i, message=msg) for i, msg in enumerate(messages) + ] + return [ + ChatCompletion(id=str(i), choices=[c], created=i, model="gpt-4", object="chat.completion") + for i, c in enumerate(choices) + ] + + class TestOpenAI: def test_make_client_kwargs_without_proxy(self): instance = OpenAILLM(mock_llm_config) @@ -50,7 +92,7 @@ class TestOpenAI: assert "http_client" in kwargs def test_get_choice_function_arguments_for_aask_code(self, tool_calls_rsp): - instance = OpenAILLM() + instance = OpenAILLM(mock_llm_config_proxy) for i, rsp in enumerate(tool_calls_rsp): code = instance.get_choice_function_arguments(rsp) logger.info(f"\ntest get function call arguments {i}: {code}") @@ -67,49 +109,3 @@ class TestOpenAI: def test_make_client_kwargs_without_proxy_azure(self, config_azure): instance = OpenAILLM() instance.config = config_azure - - @pytest.fixture - def tool_calls_rsp(self): - function_rsps = [ - Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"}', name="execute"), - Function(arguments='{\n"language": "python",\n"code": \'print("hello world")\'}', name="execute"), - Function(arguments='{\n"language": \'python\',\n"code": "print(\'hello world\')"}', name="execute"), - Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"}', name="execute"), - Function(arguments='{\n"language": "python",\n"code": ```print("hello world")```}', name="execute"), - Function(arguments='{\n"language": "python",\n"code": """print("hello world")"""}', name="execute"), - Function(arguments='\nprint("hello world")\\n', name="execute"), - # only `{` in arguments - Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), - # no `{`, `}` in arguments - Function(arguments='\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), - ] - tool_calls = [ - ChatCompletionMessageToolCall(type="function", id=f"call_{i}", function=f) - for i, f in enumerate(function_rsps) - ] - messages = [ChatCompletionMessage(content=None, role="assistant", tool_calls=[t]) for t in tool_calls] - # 添加一个纯文本响应 - messages.append( - ChatCompletionMessage(content="Completed a python code for hello world!", role="assistant", tool_calls=None) - ) - # 添加 openai tool calls respond bug, code 出现在ChatCompletionMessage.content中 - messages.extend( - [ - ChatCompletionMessage(content="```python\nprint('hello world')```", role="assistant", tool_calls=None), - ChatCompletionMessage(content="'''python\nprint('hello world')'''", role="assistant", tool_calls=None), - ChatCompletionMessage( - content='"""python\nprint(\'hello world\')"""', role="assistant", tool_calls=None - ), - ChatCompletionMessage( - content="'''python\nprint(\"hello world\")'''", role="assistant", tool_calls=None - ), - ChatCompletionMessage(content="```python\nprint('hello world')```", role="assistant", tool_calls=None), - ] - ) - choices = [ - Choice(finish_reason="tool_calls", logprobs=None, index=i, message=msg) for i, msg in enumerate(messages) - ] - return [ - ChatCompletion(id=str(i), choices=[c], created=i, model="gpt-4", object="chat.completion") - for i, c in enumerate(choices) - ] From 56f640db96816eb066a28d6b26b3945d8efb1688 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 31 Jan 2024 13:55:21 +0800 Subject: [PATCH 521/637] delete test_make_client_kwargs_without_proxy_azure. --- tests/metagpt/provider/test_openai.py | 4 ---- 1 file changed, 4 deletions(-) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index 1698518f5..a49d7e85b 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -105,7 +105,3 @@ class TestOpenAI: code["language"] == "markdown" else: code["language"] == "python" - - def test_make_client_kwargs_without_proxy_azure(self, config_azure): - instance = OpenAILLM() - instance.config = config_azure From 41907b1fe332906c542168d2f04fd93ef91bf122 Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 31 Jan 2024 13:57:32 +0800 Subject: [PATCH 522/637] add action graph, solver, search space. --- metagpt/actions/action_graph.py | 58 ++++++++ metagpt/actions/action_node.py | 159 ++++++++++++---------- metagpt/strategy/search_space.py | 20 +++ metagpt/strategy/solver.py | 77 +++++++++++ tests/data/rsp_cache.json | 37 ++++- tests/metagpt/actions/test_action_node.py | 43 +++++- tests/metagpt/strategy/test_solver.py | 47 +++++++ 7 files changed, 366 insertions(+), 75 deletions(-) create mode 100644 metagpt/actions/action_graph.py create mode 100644 metagpt/strategy/search_space.py create mode 100644 metagpt/strategy/solver.py create mode 100644 tests/metagpt/strategy/test_solver.py diff --git a/metagpt/actions/action_graph.py b/metagpt/actions/action_graph.py new file mode 100644 index 000000000..8570778c7 --- /dev/null +++ b/metagpt/actions/action_graph.py @@ -0,0 +1,58 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/1/30 13:52 +@Author : alexanderwu +@File : action_graph.py +""" +from __future__ import annotations + +# from metagpt.actions.action_node import ActionNode + + +class ActionGraph: + """ActionGraph: 用于定义一个图,图中的节点是 ActionNode 实例,节点间的依赖关系是有向边。""" + + def __init__(self): + self.nodes = {} + self.edges = {} + self.execution_order = [] + + def add_node(self, node): + """ + 添加一个节点到图中。 + :param node: ActionNode 实例 + """ + self.nodes[node.key] = node + + def add_edge(self, from_node: "ActionNode", to_node: "ActionNode"): + """ + 定义节点间的依赖关系。 + :param from_node: 节点标识 + :param to_node: 节点标识 + """ + if from_node.key not in self.edges: + self.edges[from_node.key] = [] + self.edges[from_node.key].append(to_node.key) + from_node.add_next(to_node) + to_node.add_prev(from_node) + + def topological_sort(self): + """ + 实现拓扑排序来确定执行顺序。 + """ + visited = set() + stack = [] + + def visit(k): + if k not in visited: + visited.add(k) + if k in self.edges: + for next_node in self.edges[k]: + visit(next_node) + stack.insert(0, k) + + for key in self.nodes: + visit(key) + + self.execution_order = stack diff --git a/metagpt/actions/action_node.py b/metagpt/actions/action_node.py index 162ab90eb..a3efb214e 100644 --- a/metagpt/actions/action_node.py +++ b/metagpt/actions/action_node.py @@ -9,6 +9,7 @@ NOTE: You should use typing.List instead of list to do type annotation. Because we can use typing to extract the type of the node, but we cannot use built-in list to extract. """ import json +import typing from enum import Enum from typing import Any, Dict, List, Optional, Tuple, Type, Union @@ -39,7 +40,6 @@ TAG = "CONTENT" LANGUAGE_CONSTRAINT = "Language: Please use the same language as Human INPUT." FORMAT_CONSTRAINT = f"Format: output wrapped inside [{TAG}][/{TAG}] like format example, nothing else." - SIMPLE_TEMPLATE = """ ## context {context} @@ -131,6 +131,8 @@ class ActionNode: # Action Input key: str # Product Requirement / File list / Code + func: typing.Callable # 与节点相关联的函数或LLM调用 + params: Dict[str, Type] # 输入参数的字典,键为参数名,值为参数类型 expected_type: Type # such as str / int / float etc. # context: str # everything in the history. instruction: str # the instructions should be followed. @@ -140,6 +142,10 @@ class ActionNode: content: str instruct_content: BaseModel + # For ActionGraph + prevs: List["ActionNode"] # previous nodes + nexts: List["ActionNode"] # next nodes + def __init__( self, key: str, @@ -157,6 +163,8 @@ class ActionNode: self.content = content self.children = children if children is not None else {} self.schema = schema + self.prevs = [] + self.nexts = [] def __str__(self): return ( @@ -167,6 +175,14 @@ class ActionNode: def __repr__(self): return self.__str__() + def add_prev(self, node: "ActionNode"): + """增加前置ActionNode""" + self.prevs.append(node) + + def add_next(self, node: "ActionNode"): + """增加后置ActionNode""" + self.nexts.append(node) + def add_child(self, node: "ActionNode"): """增加子ActionNode""" self.children[node.key] = node @@ -186,41 +202,38 @@ class ActionNode: obj.add_children(nodes) return obj - def get_children_mapping_old(self, exclude=None) -> Dict[str, Tuple[Type, Any]]: - """获得子ActionNode的字典,以key索引""" + def _get_children_mapping(self, exclude=None) -> Dict[str, Any]: + """获得子ActionNode的字典,以key索引,支持多级结构。""" exclude = exclude or [] - return {k: (v.expected_type, ...) for k, v in self.children.items() if k not in exclude} - def get_children_mapping(self, exclude=None) -> Dict[str, Tuple[Type, Any]]: - """获得子ActionNode的字典,以key索引,支持多级结构""" - exclude = exclude or [] - mapping = {} - - def _get_mapping(node: "ActionNode", prefix: str = ""): + def _get_mapping(node: "ActionNode") -> Dict[str, Any]: + mapping = {} for key, child in node.children.items(): if key in exclude: continue - full_key = f"{prefix}{key}" - mapping[full_key] = (child.expected_type, ...) - _get_mapping(child, prefix=f"{full_key}.") + # 对于嵌套的子节点,递归调用 _get_mapping + if child.children: + mapping[key] = _get_mapping(child) + else: + mapping[key] = (child.expected_type, Field(default=child.example, description=child.instruction)) + return mapping - _get_mapping(self) - return mapping + return _get_mapping(self) - def get_self_mapping(self) -> Dict[str, Tuple[Type, Any]]: + def _get_self_mapping(self) -> Dict[str, Tuple[Type, Any]]: """get self key: type mapping""" return {self.key: (self.expected_type, ...)} def get_mapping(self, mode="children", exclude=None) -> Dict[str, Tuple[Type, Any]]: """get key: type mapping under mode""" if mode == "children" or (mode == "auto" and self.children): - return self.get_children_mapping(exclude=exclude) - return {} if exclude and self.key in exclude else self.get_self_mapping() + return self._get_children_mapping(exclude=exclude) + return {} if exclude and self.key in exclude else self._get_self_mapping() @classmethod @register_action_outcls def create_model_class(cls, class_name: str, mapping: Dict[str, Tuple[Type, Any]]): - """基于pydantic v1的模型动态生成,用来检验结果类型正确性""" + """基于pydantic v2的模型动态生成,用来检验结果类型正确性""" def check_fields(cls, values): required_fields = set(mapping.keys()) @@ -235,7 +248,17 @@ class ActionNode: validators = {"check_missing_fields_validator": model_validator(mode="before")(check_fields)} - new_class = create_model(class_name, __validators__=validators, **mapping) + new_fields = {} + for field_name, field_value in mapping.items(): + if isinstance(field_value, dict): + # 对于嵌套结构,递归创建模型类 + nested_class_name = f"{class_name}_{field_name}" + nested_class = cls.create_model_class(nested_class_name, field_value) + new_fields[field_name] = (nested_class, ...) + else: + new_fields[field_name] = field_value + + new_class = create_model(class_name, __validators__=validators, **new_fields) return new_class def create_class(self, mode: str = "auto", class_name: str = None, exclude=None): @@ -243,39 +266,48 @@ class ActionNode: mapping = self.get_mapping(mode=mode, exclude=exclude) return self.create_model_class(class_name, mapping) - def create_children_class(self, exclude=None): + def _create_children_class(self, exclude=None): """使用object内有的字段直接生成model_class""" class_name = f"{self.key}_AN" - mapping = self.get_children_mapping(exclude=exclude) + mapping = self._get_children_mapping(exclude=exclude) return self.create_model_class(class_name, mapping) def to_dict(self, format_func=None, mode="auto", exclude=None) -> Dict: """将当前节点与子节点都按照node: format的格式组织成字典""" + nodes = self._to_dict(format_func=format_func, mode=mode, exclude=exclude) + if not isinstance(nodes, dict): + nodes = {self.key: nodes} + return nodes - # 如果没有提供格式化函数,使用默认的格式化方式 + def _to_dict(self, format_func=None, mode="auto", exclude=None) -> Dict: + """将当前节点与子节点都按照node: format的格式组织成字典""" + + # 如果没有提供格式化函数,则使用默认的格式化函数 if format_func is None: - format_func = lambda node: f"{node.instruction}" + format_func = lambda node: node.instruction # 使用提供的格式化函数来格式化当前节点的值 formatted_value = format_func(self) # 创建当前节点的键值对 - if mode == "children" or (mode == "auto" and self.children): - node_dict = {} + if (mode == "children" or mode == "auto") and self.children: + node_value = {} else: - node_dict = {self.key: formatted_value} + node_value = formatted_value if mode == "root": - return node_dict + return {self.key: node_value} - # 遍历子节点并递归调用 to_dict 方法 + # 递归处理子节点 exclude = exclude or [] - for _, child_node in self.children.items(): - if child_node.key in exclude: + for child_key, child_node in self.children.items(): + if child_key in exclude: continue - node_dict.update(child_node.to_dict(format_func)) + # 递归调用 to_dict 方法并更新节点字典 + child_dict = child_node._to_dict(format_func, mode, exclude) + node_value[child_key] = child_dict - return node_dict + return node_value def update_instruct_content(self, incre_data: dict[str, Any]): assert self.instruct_content @@ -344,6 +376,17 @@ class ActionNode: if schema == "raw": return context + "\n\n## Actions\n" + LANGUAGE_CONSTRAINT + "\n" + self.instruction + ### 直接使用 pydantic BaseModel 生成 instruction 与 example,仅限 JSON + # child_class = self._create_children_class() + # node_schema = child_class.model_json_schema() + # defaults = { + # k: str(v) + # for k, v in child_class.model_fields.items() + # if k not in exclude + # } + # instruction = node_schema + # example = json.dumps(defaults, indent=4) + # FIXME: json instruction会带来格式问题,如:"Project name": "web_2048 # 项目名称使用下划线", # compile example暂时不支持markdown instruction = self.compile_instruction(schema="markdown", mode=mode, exclude=exclude) @@ -454,7 +497,7 @@ class ActionNode: continue child = await i.simple_fill(schema=schema, mode=mode, timeout=timeout, exclude=exclude) tmp.update(child.instruct_content.model_dump()) - cls = self.create_children_class() + cls = self._create_children_class() self.instruct_content = cls(**tmp) return self @@ -645,49 +688,19 @@ class ActionNode: ActionNode: The root node of the created ActionNode tree. """ key = key or model.__name__ - root_node = cls(key=model.__name__, expected_type=Type[model], instruction="", example="") + root_node = cls(key=key, expected_type=Type[model], instruction="", example="") - for field_name, field_model in model.model_fields.items(): - # Extracting field details - expected_type = field_model.annotation - instruction = field_model.description or "" - example = field_model.default + for field_name, field_info in model.model_fields.items(): + field_type = field_info.annotation + description = field_info.description + default = field_info.default - # Check if the field is a Pydantic model itself. - # Use isinstance to avoid typing.List, typing.Dict, etc. (they are instances of type, not subclasses) - if isinstance(expected_type, type) and issubclass(expected_type, BaseModel): - # Recursively process the nested model - child_node = cls.from_pydantic(expected_type, key=field_name) + # Recursively handle nested models if needed + if not isinstance(field_type, typing._GenericAlias) and issubclass(field_type, BaseModel): + child_node = cls.from_pydantic(field_type, key=field_name) else: - child_node = cls(key=field_name, expected_type=expected_type, instruction=instruction, example=example) + child_node = cls(key=field_name, expected_type=field_type, instruction=description, example=default) root_node.add_child(child_node) return root_node - - -class ToolUse(BaseModel): - tool_name: str = Field(default="a", description="tool name", examples=[]) - - -class Task(BaseModel): - task_id: int = Field(default="1", description="task id", examples=[1, 2, 3]) - name: str = Field(default="Get data from ...", description="task name", examples=[]) - dependent_task_ids: List[int] = Field(default=[], description="dependent task ids", examples=[1, 2, 3]) - tool: ToolUse = Field(default=ToolUse(), description="tool use", examples=[]) - - -class Tasks(BaseModel): - tasks: List[Task] = Field(default=[], description="tasks", examples=[]) - - -if __name__ == "__main__": - node = ActionNode.from_pydantic(Tasks) - print("Tasks") - print(Tasks.model_json_schema()) - print("Task") - print(Task.model_json_schema()) - print(node) - prompt = node.compile(context="") - node.create_children_class() - print(prompt) diff --git a/metagpt/strategy/search_space.py b/metagpt/strategy/search_space.py new file mode 100644 index 000000000..c643a2f11 --- /dev/null +++ b/metagpt/strategy/search_space.py @@ -0,0 +1,20 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/1/30 17:15 +@Author : alexanderwu +@File : search_space.py +""" + + +class SearchSpace: + """SearchSpace: 用于定义一个搜索空间,搜索空间中的节点是 ActionNode 类。""" + + def __init__(self): + self.search_space = {} + + def add_node(self, node): + self.search_space[node.key] = node + + def get_node(self, key): + return self.search_space[key] diff --git a/metagpt/strategy/solver.py b/metagpt/strategy/solver.py new file mode 100644 index 000000000..bd21dda3e --- /dev/null +++ b/metagpt/strategy/solver.py @@ -0,0 +1,77 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/1/30 17:13 +@Author : alexanderwu +@File : solver.py +""" +from abc import abstractmethod + +from metagpt.actions.action_graph import ActionGraph +from metagpt.provider.base_llm import BaseLLM +from metagpt.strategy.search_space import SearchSpace + + +class BaseSolver: + """AbstractSolver: 用于定义一个抽象求解器,求解器中的搜索空间是 SearchSpace 实例,图是 ActionGraph 实例。""" + + def __init__(self, graph: ActionGraph, search_space: SearchSpace, llm: BaseLLM, context): + """ + :param graph: ActionGraph 实例 + :param search_space: SearchSpace 实例 + :param llm: BaseLLM + :param context: Context + """ + self.graph = graph + self.search_space = search_space + self.llm = llm + self.context = context + + @abstractmethod + async def solve(self): + """求解器的求解方法。""" + + +class NaiveSolver(BaseSolver): + """NaiveSolver: 直接循序执行给定的 graph""" + + async def solve(self): + self.graph.topological_sort() + for key in self.graph.execution_order: + op = self.graph.nodes[key] + await op.fill(self.context, self.llm, mode="root") + + +class TOTSolver(BaseSolver): + """TOTSolver: 通过拓扑排序执行给定的 graph""" + + async def solve(self): + raise NotImplementedError + + +class CodeInterpreterSolver(BaseSolver): + """CodeInterpreterSolver: 通过代码解释器执行给定的 graph""" + + async def solve(self): + raise NotImplementedError + + +class ReActSolver(BaseSolver): + """ReActSolver: 通过 ReAct 执行给定的 graph""" + + async def solve(self): + raise NotImplementedError + + +class IOSolver(BaseSolver): + """IOSolver: 通过 IO 执行给定的 graph""" + + async def solve(self): + raise NotImplementedError + + +class COTSolver(BaseSolver): + """COTSolver: 通过cot执行给定的 graph""" + + async def solve(self): + raise NotImplementedError diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index 92a600b67..998eb714f 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -277,5 +277,40 @@ "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 3:\nEnsure there is enough space around you to avoid any obstacles.\n": "Choice 3: \n\nAnalyzing this choice, ensuring there is enough space around you to avoid any obstacles is a crucial step in maintaining safety and preventing accidents. By creating a buffer zone, you can react to unexpected obstacles or changes in the environment without risking collisions or other hazards.\n\nThe best choice is 3.", "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 2:\nConsider a flat and stable surface, such as a grassy field or a gym mat.\n": "Choice 2: Considering a flat and stable surface, such as a grassy field or a gym mat, is the most promising option. This choice provides a safe and secure environment for the activity, reducing the risk of injury. A grassy field offers a natural and soft surface, while a gym mat provides cushioning and stability. Both options allow for comfortable movement and can accommodate various physical activities.\n\nThe best choice is 2.", "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 4:\nPractice near a wall or with a spotter for safety.\n": "Choice 4: \n\nPractice near a wall or with a spotter for safety.\n\nAnalyzing this choice, practicing near a wall or with a spotter provides a safety measure to prevent accidents or injuries. When practicing a new skill or exercise, having a wall nearby can provide support and stability, reducing the risk of falling or losing balance. Similarly, having a spotter can offer assistance and guidance, ensuring that the practice is done safely and effectively.\n\nThe best choice is 4.", - "### Requirements\n1. Please summarize the latest dialogue based on the reference information (secondary) and dialogue history (primary). Do not include text that is irrelevant to the conversation.\n- The context is for reference only. If it is irrelevant to the user's search request history, please reduce its reference and usage.\n2. If there are citable links in the context, annotate them in the main text in the format [main text](citation link). If there are none in the context, do not write links.\n3. The reply should be graceful, clear, non-repetitive, smoothly written, and of moderate length, in {LANG}.\n\n### Dialogue History (For example)\nA: MLOps competitors\n\n### Current Question (For example)\nA: MLOps competitors\n\n### Current Reply (For example)\n1. Alteryx Designer: etc. if any\n2. Matlab: ditto\n3. IBM SPSS Statistics\n4. RapidMiner Studio\n5. DataRobot AI Platform\n6. Databricks Lakehouse Platform\n7. Amazon SageMaker\n8. Dataiku\n#SYSTEM_MSG_END#\n### Reference Information\nABC cleanser is preferred by many with oily skin.\nL'Oreal is a popular brand with many positive reviews.\n\n### Dialogue History\n\nuser: Which facial cleanser is good for oily skin?\n\n### Current Question\nuser: Which facial cleanser is good for oily skin?\n\n### Current Reply: Based on the information, please write the reply to the Question\n\n\n": "Based on the information provided, ABC cleanser is preferred by many with oily skin. It is a popular choice for individuals with oily skin due to its effectiveness. Additionally, L'Oreal is a well-known brand with many positive reviews, and they offer a range of products suitable for oily skin. Both of these options could be good choices for individuals with oily skin." + "### Requirements\n1. Please summarize the latest dialogue based on the reference information (secondary) and dialogue history (primary). Do not include text that is irrelevant to the conversation.\n- The context is for reference only. If it is irrelevant to the user's search request history, please reduce its reference and usage.\n2. If there are citable links in the context, annotate them in the main text in the format [main text](citation link). If there are none in the context, do not write links.\n3. The reply should be graceful, clear, non-repetitive, smoothly written, and of moderate length, in {LANG}.\n\n### Dialogue History (For example)\nA: MLOps competitors\n\n### Current Question (For example)\nA: MLOps competitors\n\n### Current Reply (For example)\n1. Alteryx Designer: etc. if any\n2. Matlab: ditto\n3. IBM SPSS Statistics\n4. RapidMiner Studio\n5. DataRobot AI Platform\n6. Databricks Lakehouse Platform\n7. Amazon SageMaker\n8. Dataiku\n#SYSTEM_MSG_END#\n### Reference Information\nABC cleanser is preferred by many with oily skin.\nL'Oreal is a popular brand with many positive reviews.\n\n### Dialogue History\n\nuser: Which facial cleanser is good for oily skin?\n\n### Current Question\nuser: Which facial cleanser is good for oily skin?\n\n### Current Reply: Based on the information, please write the reply to the Question\n\n\n": "Based on the information provided, ABC cleanser is preferred by many with oily skin. It is a popular choice for individuals with oily skin due to its effectiveness. Additionally, L'Oreal is a well-known brand with many positive reviews, and they offer a range of products suitable for oily skin. Both of these options could be good choices for individuals with oily skin.", + "\n## context\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Required Python packages\": [\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"openapi: 3.0.0 ...\",\n \"Shared Knowledge\": \"`game.py` contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on how to start and initialize third-party libraries.\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Required Python packages: typing.List[str] # Provide required Python packages in requirements.txt format.\n- Required Other language third-party packages: typing.List[str] # List down the required packages for languages other than Python.\n- Logic Analysis: typing.List[typing.List[str]] # Provide a list of files with the classes/methods/functions to be implemented, including dependency analysis and imports.\n- Task list: typing.List[str] # Break down the tasks into a list of filenames, prioritized by dependency order.\n- Full API spec: # Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end and back-end communication is not required, leave it blank.\n- Shared Knowledge: # Detail any shared knowledge, like common utility functions or configuration variables.\n- Anything UNCLEAR: # Mention any unclear aspects in the project management context and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Required Python packages\": [\n \"pygame==2.0.1\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"\",\n \"Shared Knowledge\": \"`game.py` contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on the scoring system and how the game over condition will be handled.\"\n}\n[/CONTENT]", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Task\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"`game.py` contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## game.py\n\nimport pygame\nimport random\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos]\n self.direction = direction\n self.length = length\n\n def move(self):\n head = self.body[0]\n x, y = head\n if self.direction == 'UP':\n new_head = (x, y - 10)\n elif self.direction == 'DOWN':\n new_head = (x, y + 10)\n elif self.direction == 'LEFT':\n new_head = (x - 10, y)\n elif self.direction == 'RIGHT':\n new_head = (x + 10, y)\n self.body = [new_head] + self.body[:-1]\n\n def change_direction(self, new_direction: str):\n if new_direction in ['UP', 'DOWN', 'LEFT', 'RIGHT']:\n self.direction = new_direction\n\n def grow(self):\n self.length += 1\n self.body.append((0, 0)) # Set default value\n\nclass Game:\n def __init__(self):\n self.snake = Snake((100, 50), 1, 'RIGHT') # Set default value\n self.food = (200, 200) # Set default value\n self.score = 0 # Set default value\n\n def update(self):\n self.snake.move()\n if self.snake.body[0] == self.food:\n self.snake.grow()\n self.food = (random.randrange(0, 400, 10), random.randrange(0, 400, 10)) # Set default value\n\n def draw(self):\n pass # Set default value\n\n def handle_input(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n pygame.quit()\n quit()\n if event.type == pygame.KEYDOWN:\n if event.key == pygame.K_UP:\n self.snake.change_direction('UP')\n elif event.key == pygame.K_DOWN:\n self.snake.change_direction('DOWN')\n elif event.key == pygame.K_LEFT:\n self.snake.change_direction('LEFT')\n elif event.key == pygame.K_RIGHT:\n self.snake.change_direction('RIGHT')\n\n def restart(self):\n self.snake = Snake((100, 50), 1, 'RIGHT') # Set default value\n self.food = (200, 200) # Set default value\n self.score = 0 # Set default value\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Task\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"`game.py` contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nimport pygame\nimport random\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos]\n self.direction = direction\n self.length = length\n\n def move(self):\n head = self.body[0]\n x, y = head\n if self.direction == 'UP':\n new_head = (x, y - 10)\n elif self.direction == 'DOWN':\n new_head = (x, y + 10)\n elif self.direction == 'LEFT':\n new_head = (x - 10, y)\n elif self.direction == 'RIGHT':\n new_head = (x + 10, y)\n self.body = [new_head] + self.body[:-1]\n\n def change_direction(self, new_direction: str):\n if new_direction in ['UP', 'DOWN', 'LEFT', 'RIGHT']:\n self.direction = new_direction\n\n def grow(self):\n self.length += 1\n self.body.append((0, 0)) # Set default value\n\nclass Game:\n def __init__(self):\n self.snake = Snake((100, 50), 1, 'RIGHT') # Set default value\n self.food = (200, 200) # Set default value\n self.score = 0 # Set default value\n\n def update(self):\n self.snake.move()\n if self.snake.body[0] == self.food:\n self.snake.grow()\n self.food = (random.randrange(0, 400, 10), random.randrange(0, 400, 10)) # Set default value\n\n def draw(self):\n pass # Set default value\n\n def handle_input(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n pygame.quit()\n quit()\n if event.type == pygame.KEYDOWN:\n if event.key == pygame.K_UP:\n self.snake.change_direction('UP')\n elif event.key == pygame.K_DOWN:\n self.snake.change_direction('DOWN')\n elif event.key == pygame.K_LEFT:\n self.snake.change_direction('LEFT')\n elif event.key == pygame.K_RIGHT:\n self.snake.change_direction('RIGHT')\n\n def restart(self):\n self.snake = Snake((100, 50), 1, 'RIGHT') # Set default value\n self.food = (200, 200) # Set default value\n self.score = 0 # Set default value\n\n```\n\n\n\n# Format example 1\n## Code Review: game.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: game.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: game.py\n1. The code is implemented as per the requirements. The Snake and Game classes are defined, and the necessary methods for updating, drawing, handling input, and restarting the game are implemented.\n2. The code logic is correct. The Snake class methods for moving, changing direction, and growing the snake are implemented properly. The Game class methods for updating the game state, handling input, and restarting the game are also correctly implemented.\n3. The existing code follows the defined \"Data structures and interfaces\" with the Snake and Game classes and their respective methods.\n4. All necessary functions are implemented as per the requirements.\n5. The necessary pre-dependencies, including the pygame package, are imported.\n6. The methods from other files are not being reused in the current code.\n\n## Actions\nNo actions required.\n\n## Code Review Result\nLGTM", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Task\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"`game.py` contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: main.py\n```python\n## main.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: main.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.update()\n game.draw()\n game.handle_input()\n\n if game.game_over:\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Task\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"`game.py` contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.update()\n game.draw()\n game.handle_input()\n\n if game.game_over:\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n\n\n# Format example 1\n## Code Review: main.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: main.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: main.py\n1. Yes, the code is implemented as per the requirements.\n2. Yes, the code logic is completely correct.\n3. Yes, the existing code follows the \"Data structures and interfaces\".\n4. No, the `game_over` condition is not handled in the code. To achieve it, we need to add the implementation of the `game_over` condition in the `Game` class.\n5. Yes, all necessary pre-dependencies have been imported.\n6. Yes, methods from other files are being reused correctly.\n\n## Actions\n4. Implement the `game_over` condition in the `Game` class to handle the game over scenario.\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Task\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"`game.py` contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.update()\n game.draw()\n game.handle_input()\n\n if game.game_over:\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n## Code Review: main.py\n1. Yes, the code is implemented as per the requirements.\n2. Yes, the code logic is completely correct.\n3. Yes, the existing code follows the \"Data structures and interfaces\".\n4. No, the `game_over` condition is not handled in the code. To achieve it, we need to add the implementation of the `game_over` condition in the `Game` class.\n5. Yes, all necessary pre-dependencies have been imported.\n6. Yes, methods from other files are being reused correctly.\n\n## Actions\n4. Implement the `game_over` condition in the `Game` class to handle the game over scenario.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite main.py with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## main.py\n...\n```\n": "```Code\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.update()\n game.draw()\n game.handle_input()\n\n if game.game_over():\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Task\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"`game.py` contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: main.py\n```Code\n## main.py\n\nfrom game import Game\n\ndef main():\n game = Game()\n running = True\n\n while running:\n game.update()\n game.draw()\n game.handle_input()\n\n if game.game_over():\n game.restart()\n\nif __name__ == \"__main__\":\n main()\n\n```\n\n\n\n# Format example 1\n## Code Review: main.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: main.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: main.py\n1. The code is implemented as per the requirements. It initializes the game, updates the game state, draws the game, handles user input, and restarts the game when it's over.\n2. The code logic is correct, and it follows the expected flow of the game.\n3. The existing code follows the \"Data structures and interfaces\" as it uses classes and methods as described in the class diagram.\n4. All necessary functions are implemented as per the requirements.\n5. The necessary pre-dependencies have been imported, as the `pygame` package is imported.\n6. The methods from other files are being reused correctly.\n\n## Actions\npass\n\n## Code Review Result\nLGTM", + "\n## context\n{'Implementation approach': '我们将使用Python编程语言,并选择合适的开源框架来实现贪吃蛇游戏。我们将分析需求中的难点,并选择合适的开源框架来简化开发流程。', 'File list': ['main.py', 'game.py'], 'Data structures and interfaces': '\\nclassDiagram\\n class Game {\\n -int width\\n -int height\\n -int score\\n -int speed\\n -List snake\\n -Point food\\n +__init__(width: int, height: int, speed: int)\\n +start_game()\\n +change_direction(direction: str)\\n +game_over()\\n +update_snake()\\n +update_food()\\n +check_collision()\\n }\\n class Point {\\n -int x\\n -int y\\n +__init__(x: int, y: int)\\n }\\n Game --> Point\\n', 'Program call flow': '\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: start_game()\\n M->>G: change_direction(direction)\\n G->>G: update_snake()\\n G->>G: update_food()\\n G->>G: check_collision()\\n G-->>G: game_over()\\n', 'Anything UNCLEAR': ''}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Required Python packages\": [\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"openapi: 3.0.0 ...\",\n \"Shared Knowledge\": \"`game.py` contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on how to start and initialize third-party libraries.\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Required Python packages: typing.List[str] # Provide required Python packages in requirements.txt format.\n- Required Other language third-party packages: typing.List[str] # List down the required packages for languages other than Python.\n- Logic Analysis: typing.List[typing.List[str]] # Provide a list of files with the classes/methods/functions to be implemented, including dependency analysis and imports.\n- Task list: typing.List[str] # Break down the tasks into a list of filenames, prioritized by dependency order.\n- Full API spec: # Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end and back-end communication is not required, leave it blank.\n- Shared Knowledge: # Detail any shared knowledge, like common utility functions or configuration variables.\n- Anything UNCLEAR: # Mention any unclear aspects in the project management context and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Required Python packages\": [\n \"pygame==2.0.1\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and related functions\"\n ],\n [\n \"main.py\",\n \"Contains the main function and imports Game from game.py\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"\",\n \"Shared Knowledge\": \"`game.py` contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"\"\n}\n[/CONTENT]", + "\nNOTICE\nRole: You are a professional software engineer, and your main task is to review the code.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n-----\n# System Design\n```text\n\n{\"Implementation approach\": \"To develop this snake game, we will use the Python language and choose the Pygame library. Pygame is an open-source Python module collection specifically designed for writing video games. It provides functionalities such as displaying images and playing sounds, making it suitable for creating intuitive and responsive user interfaces. We will ensure efficient game logic to prevent any delays during gameplay. The scoring system will be simple, with the snake gaining points for each food it eats. We will use Pygame's event handling system to implement pause and resume functionality, as well as high-score tracking. The difficulty will increase by speeding up the snake's movement. In the initial version, we will focus on single-player mode and consider adding multiplayer mode and customizable skins in future updates. Based on the new requirement, we will also add a moving obstacle that appears randomly. If the snake eats this obstacle, the game will end. If the snake does not eat the obstacle, it will disappear after 5 seconds. For this, we need to add mechanisms for obstacle generation, movement, and disappearance in the game logic.\", \"Project_name\": \"snake_game\", \"File list\": [\"main.py\", \"game.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"constants.py\", \"assets/styles.css\", \"assets/index.html\"], \"Data structures and interfaces\": \"```mermaid\n classDiagram\n class Game{\n +int score\n +int speed\n +bool game_over\n +bool paused\n +Snake snake\n +Food food\n +Obstacle obstacle\n +Scoreboard scoreboard\n +start_game() void\n +pause_game() void\n +resume_game() void\n +end_game() void\n +increase_difficulty() void\n +update() void\n +render() void\n Game()\n }\n class Snake{\n +list body_parts\n +str direction\n +bool grow\n +move() void\n +grow() void\n +check_collision() bool\n Snake()\n }\n class Food{\n +tuple position\n +spawn() void\n Food()\n }\n class Obstacle{\n +tuple position\n +int lifetime\n +bool active\n +spawn() void\n +move() void\n +check_collision() bool\n +disappear() void\n Obstacle()\n }\n class Scoreboard{\n +int high_score\n +update_score(int) void\n +reset_score() void\n +load_high_score() void\n +save_high_score() void\n Scoreboard()\n }\n class Constants{\n }\n Game \"1\" -- \"1\" Snake: has\n Game \"1\" -- \"1\" Food: has\n Game \"1\" -- \"1\" Obstacle: has\n Game \"1\" -- \"1\" Scoreboard: has\n ```\", \"Program call flow\": \"```sequenceDiagram\n participant M as Main\n participant G as Game\n participant S as Snake\n participant F as Food\n participant O as Obstacle\n participant SB as Scoreboard\n M->>G: start_game()\n loop game loop\n G->>S: move()\n G->>S: check_collision()\n G->>F: spawn()\n G->>O: spawn()\n G->>O: move()\n G->>O: check_collision()\n G->>O: disappear()\n G->>SB: update_score(score)\n G->>G: update()\n G->>G: render()\n alt if paused\n M->>G: pause_game()\n M->>G: resume_game()\n end\n alt if game_over\n G->>M: end_game()\n end\n end\n```\", \"Anything UNCLEAR\": \"There is no need for further clarification as the requirements are already clear.\"}\n\n```\n-----\n# Task\n```text\n\n{\"Required Python third-party packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party packages required for other languages.\"], \"Full API spec\": \"\n openapi: 3.0.0\n info:\n title: Snake Game API\n version: \"1.0.0\"\n paths:\n /start:\n get:\n summary: Start the game\n responses:\n '200':\n description: Game started successfully\n /pause:\n get:\n summary: Pause the game\n responses:\n '200':\n description: Game paused successfully\n /resume:\n get:\n summary: Resume the game\n responses:\n '200':\n description: Game resumed successfully\n /end:\n get:\n summary: End the game\n responses:\n '200':\n description: Game ended successfully\n /score:\n get:\n summary: Get the current score\n responses:\n '200':\n description: Current score retrieved successfully\n /highscore:\n get:\n summary: Get the high score\n responses:\n '200':\n description: High score retrieved successfully\n components: {}\n \", \"Logic Analysis\": [[\"constants.py\", \"Contains all the constant values like screen size, colors, game speeds, etc. This should be implemented first as it provides the base values for other components.\"], [\"snake.py\", \"Contains the Snake class with methods for movement, growth, and collision detection. It is dependent on constants.py for configuration values.\"], [\"food.py\", \"Contains the Food class responsible for spawning food items on the screen. It is dependent on constants.py for configuration values.\"], [\"obstacle.py\", \"Contains the Obstacle class with methods for spawning, moving, and disappearing of obstacles, as well as collision detection with the snake. It is dependent on constants.py for configuration values.\"], [\"scoreboard.py\", \"Contains the Scoreboard class for updating, resetting, loading, and saving high scores. It may use constants.py for configuration values and depends on the game's scoring logic.\"], [\"game.py\", \"Contains the main Game class which includes the game loop and methods for starting, pausing, resuming, and ending the game. It is dependent on snake.py, food.py, obstacle.py, and scoreboard.py.\"], [\"main.py\", \"The entry point of the game that initializes the game and starts the game loop. It is dependent on game.py.\"]], \"Task list\": [\"constants.py\", \"snake.py\", \"food.py\", \"obstacle.py\", \"scoreboard.py\", \"game.py\", \"main.py\"], \"Shared Knowledge\": \"\n 'constants.py' should contain all the necessary configurations for the game, such as screen dimensions, color definitions, and speed settings. These constants will be used across multiple files, ensuring consistency and ease of updates. Ensure that the Pygame library is initialized correctly in 'main.py' before starting the game loop. Also, make sure that the game's state is managed properly when pausing and resuming the game.\n \", \"Anything UNCLEAR\": \"The interaction between the 'obstacle.py' and the game loop needs to be clearly defined to ensure obstacles appear and disappear correctly. The lifetime of the obstacle and its random movement should be implemented in a way that does not interfere with the game's performance.\"}\n\n```\n-----\n```python\n\n## game.py\nimport pygame\nfrom snake import Snake\nfrom food import Food\n\nclass Game:\n def __init__(self):\n self.score = 0\n self.level = 1\n self.snake = Snake()\n self.food = Food()\n\n def start_game(self):\n pygame.init()\n self.initialize_game()\n self.game_loop()\n\n def initialize_game(self):\n self.score = 0\n self.level = 1\n self.snake.reset()\n self.food.generate()\n\n def game_loop(self):\n game_over = False\n\n while not game_over:\n self.update()\n self.draw()\n self.handle_events()\n self.check_collision()\n self.increase_score()\n self.increase_level()\n\n if self.snake.is_collision():\n game_over = True\n self.game_over()\n\n def update(self):\n self.snake.move()\n\n def draw(self):\n self.snake.draw()\n self.food.draw()\n\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n pygame.quit()\n quit()\n elif event.type == pygame.KEYDOWN:\n if event.key == pygame.K_UP:\n self.snake.change_direction(\"UP\")\n elif event.key == pygame.K_DOWN:\n self.snake.change_direction(\"DOWN\")\n elif event.key == pygame.K_LEFT:\n self.snake.change_direction(\"LEFT\")\n elif event.key == pygame.K_RIGHT:\n self.snake.change_direction(\"RIGHT\")\n\n def check_collision(self):\n if self.snake.get_head() == self.food.get_position():\n self.snake.grow()\n self.food.generate()\n\n def increase_score(self):\n self.score += 1\n\n def increase_level(self):\n if self.score % 10 == 0:\n self.level += 1\n\n def game_over(self):\n print(\"Game Over\")\n self.initialize_game()\n\n\n```\n-----\n```python\n\n## snake.py\nimport pygame\n\nclass Snake:\n def __init__(self):\n self.body = [(0, 0)]\n self.direction = (1, 0)\n\n def move(self):\n head = self.body[0]\n dx, dy = self.direction\n new_head = (head[0] + dx, head[1] + dy)\n self.body.insert(0, new_head)\n self.body.pop()\n\n def change_direction(self, direction):\n if direction == \"UP\":\n self.direction = (0, -1)\n elif direction == \"DOWN\":\n self.direction = (0, 1)\n elif direction == \"LEFT\":\n self.direction = (-1, 0)\n elif direction == \"RIGHT\":\n self.direction = (1, 0)\n\n def grow(self):\n tail = self.body[-1]\n dx, dy = self.direction\n new_tail = (tail[0] - dx, tail[1] - dy)\n self.body.append(new_tail)\n\n def get_head(self):\n return self.body[0]\n\n def get_body(self):\n return self.body[1:]\n\n\n```\n-----\n```python\n\n## main.py\nimport pygame\nfrom game import Game\n\ndef main():\n pygame.init()\n game = Game()\n game.start_game()\n\nif __name__ == \"__main__\":\n main()\n\n\n```\n-----\n```python\n\n## food.py\nimport random\n\nclass Food:\n def __init__(self):\n self.position = (0, 0)\n\n def generate(self):\n x = random.randint(0, 9)\n y = random.randint(0, 9)\n self.position = (x, y)\n\n def get_position(self):\n return self.position\n\n\n```\n-----\n\n## Code Review All: Please read all historical files and find possible bugs in the files, such as unimplemented functions, calling errors, unreferences, etc.\n\n## Call flow: mermaid code, based on the implemented function, use mermaid to draw a complete call chain\n\n## Summary: Summary based on the implementation of historical files\n\n## TODOs: Python dict[str, str], write down the list of files that need to be modified and the reasons. We will modify them later.\n\n": "```mermaid\nclassDiagram\n class Game{\n +int score\n +int level\n +Snake snake\n +Food food\n +start_game() void\n +initialize_game() void\n +game_loop() void\n +update() void\n +draw() void\n +handle_events() void\n +check_collision() void\n +increase_score() void\n +increase_level() void\n +game_over() void\n Game()\n }\n class Snake{\n +list body\n +tuple direction\n +move() void\n +change_direction(direction: str) void\n +grow() void\n +get_head() tuple\n +get_body() list\n Snake()\n }\n class Food{\n +tuple position\n +generate() void\n +get_position() tuple\n Food()\n }\n Game \"1\" -- \"1\" Snake: has\n Game \"1\" -- \"1\" Food: has\n```\n\n```sequenceDiagram\nparticipant M as Main\nparticipant G as Game\nparticipant S as Snake\nparticipant F as Food\nM->>G: start_game()\nG->>G: initialize_game()\nG->>G: game_loop()\nG->>S: move()\nG->>S: change_direction()\nG->>S: grow()\nG->>F: generate()\nS->>S: move()\nS->>S: change_direction()\nS->>S: grow()\nF->>F: generate()\n```\n\n## Summary\nThe code consists of the main game logic, including the Game, Snake, and Food classes. The game loop is responsible for updating and drawing the game elements, handling events, checking collisions, and managing the game state. The Snake class handles the movement, growth, and direction changes of the snake, while the Food class is responsible for generating and tracking the position of food items.\n\n## TODOs\n- Modify 'game.py' to add the implementation of obstacle handling and interaction with the game loop.\n- Implement 'obstacle.py' to include the methods for spawning, moving, and disappearing of obstacles, as well as collision detection with the snake.\n- Update 'main.py' to initialize the obstacle and incorporate it into the game loop.\n- Update the mermaid call flow diagram to include the interaction with the obstacle.\n\n```python\n{\n \"files_to_modify\": {\n \"game.py\": \"Add obstacle handling and interaction with the game loop\",\n \"obstacle.py\": \"Implement obstacle class with necessary methods\",\n \"main.py\": \"Initialize the obstacle and incorporate it into the game loop\"\n }\n}\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n设计一个名为'add'的函数,该函数接受两个整数作为输入,并返回它们的和。\n\n## Task\n\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: \n```python\n## \n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: . Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: \n```python\n## Design a function called 'add' that takes two integers as input and returns their sum.\n\ndef add(num1: int, num2: int) -> int:\n \"\"\"\n Add two integers and return the sum.\n \n Args:\n num1: An integer.\n num2: An integer.\n \n Returns:\n The sum of num1 and num2 as an integer.\n \"\"\"\n return num1 + num2\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\": \"We will use the Pygame library to create the game interface and handle user input. The game logic will be implemented using Python classes and data structures.\", \"File list\": [\"main.py\", \"game.py\"], \"Data structures and interfaces\": \"classDiagram\\n class Game {\\n -grid: List[List[int]]\\n -score: int\\n -game_over: bool\\n +__init__()\\n +reset_game()\\n +move(direction: str)\\n +is_game_over() bool\\n +get_empty_cells() List[Tuple[int, int]]\\n +add_new_tile()\\n +get_score() int\\n }\\n class UI {\\n -game: Game\\n +__init__(game: Game)\\n +draw_grid()\\n +draw_score()\\n +draw_game_over()\\n +handle_input()\\n }\\n Game --> UI\", \"Program call flow\": \"sequenceDiagram\\n participant M as Main\\n participant G as Game\\n participant U as UI\\n M->>G: reset_game()\\n M->>U: draw_grid()\\n M->>U: draw_score()\\n M->>U: handle_input()\\n U->>G: move(direction)\\n G->>G: add_new_tile()\\n G->>U: draw_grid()\\n G->>U: draw_score()\\n G->>U: draw_game_over()\\n G->>G: is_game_over()\\n G->>G: get_empty_cells()\\n G->>G: get_score()\", \"Anything UNCLEAR\": \"...\"}\n\n## Task\n{\"Required Python packages\": [\"pygame==2.0.1\"], \"Required Other language third-party packages\": [\"No third-party dependencies required\"], \"Logic Analysis\": [[\"game.py\", \"Contains Game class and related functions for game logic\"], [\"main.py\", \"Contains main function, initializes the game and UI\"]], \"Task list\": [\"game.py\", \"main.py\"], \"Full API spec\": \"\", \"Shared Knowledge\": \"The game logic will be implemented using Python classes and data structures. The Pygame library will be used to create the game interface and handle user input.\", \"Anything UNCLEAR\": \"...\"}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\nE.......F\n======================================================================\nERROR: test_add_new_tile (__main__.TestGame)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n File \"/Users/xx/tests/test_game.py\", line 104, in test_add_new_tile\n self.assertIn(self.game.grid[empty_cells[0][0]][empty_cells[0][1]], [2, 4])\nIndexError: list index out of range\n\n======================================================================\nFAIL: test_reset_game (__main__.TestGame)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n File \"/Users/xx/tests/test_game.py\", line 13, in test_reset_game\n self.assertEqual(self.game.grid, [[0 for _ in range(4)] for _ in range(4)])\nAssertionError: Lists differ: [[0, 0, 0, 0], [0, 2, 0, 0], [0, 0, 0, 2], [0, 0, 0, 0]] != [[0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0]]\n\nFirst differing element 1:\n[0, 2, 0, 0]\n[0, 0, 0, 0]\n\n- [[0, 0, 0, 0], [0, 2, 0, 0], [0, 0, 0, 2], [0, 0, 0, 0]]\n? --- ^\n\n+ [[0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0], [0, 0, 0, 0]]\n? +++ ^\n\n\n----------------------------------------------------------------------\nRan 9 tests in 0.002s\n\nFAILED (failures=1, errors=1)\n\n\n{\"design_filename\": \"docs/system_design/20231221155954.json\", \"task_filename\": \"docs/tasks/20231221155954.json\", \"codes_filenames\": [\"game.py\", \"main.py\"], \"reason\": \"```json\\n{\\n \\\"game.py\\\": \\\"Add handling for no empty cells in add_new_tile function, Update score in move function\\\",\\n \\\"main.py\\\": \\\"Handle game over condition in the game loop\\\"\\n}\\n```\"}\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code: game.py\n```python\n## game.py\n\nfrom typing import List, Tuple\nimport random\n\nclass Game:\n def __init__(self):\n self.grid: List[List[int]] = [[0 for _ in range(4)] for _ in range(4)]\n self.score: int = 0\n self.game_over: bool = False\n\n def reset_game(self):\n self.grid = [[0 for _ in range(4)] for _ in range(4)]\n self.score = 0\n self.game_over = False\n\n def move(self, direction: str):\n # Implement the move logic based on the direction\n pass\n\n def is_game_over(self) -> bool:\n # Check if the game is over\n return self.game_over\n\n def get_empty_cells(self) -> List[Tuple[int, int]]:\n empty_cells = []\n for i in range(4):\n for j in range(4):\n if self.grid[i][j] == 0:\n empty_cells.append((i, j))\n return empty_cells\n\n def add_new_tile(self):\n empty_cells = self.get_empty_cells()\n if empty_cells:\n cell = random.choice(empty_cells)\n self.grid[cell[0]][cell[1]] = random.choice([2, 4])\n else:\n # Handle the case when there are no empty cells\n pass\n\n def get_score(self) -> int:\n return self.score\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n编写一个从a加b的函数,返回a+b\n\n## Task\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef add(a, b):\n return a + \n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. No, the code is not implemented as per the requirements. The function 'add' is incomplete and does not return the sum of 'a' and 'b'.\n2. No, the code logic is not completely correct. The return statement in the 'add' function is incomplete.\n3. N/A\n4. N/A\n5. N/A\n6. N/A\n\n## Actions\n1. Complete the 'add' function to return the sum of 'a' and 'b'.\n ```python\n def add(a, b):\n return a + b\n ```\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n编写一个从a加b的函数,返回a+b\n\n## Task\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef add(a, b):\n return a + \n\n```\n\n## Code Review: \n1. No, the code is not implemented as per the requirements. The function 'add' is incomplete and does not return the sum of 'a' and 'b'.\n2. No, the code logic is not completely correct. The return statement in the 'add' function is incomplete.\n3. N/A\n4. N/A\n5. N/A\n6. N/A\n\n## Actions\n1. Complete the 'add' function to return the sum of 'a' and 'b'.\n ```python\n def add(a, b):\n return a + b\n ```\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## \n...\n```\n": "```python\ndef add(a, b):\n return a + b\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n编写一个从a加b的函数,返回a+b\n\n## Task\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\ndef add(a, b):\n return a + b\n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. Yes, the code is implemented as per the requirements.\n2. Yes, the code logic is completely correct.\n3. Yes, the existing code follows the \"Data structures and interfaces\".\n4. Yes, the function is implemented.\n5. Yes, all necessary pre-dependencies have been imported.\n6. N/A\n\n## Code Review Result\nLGTM", + "\n## context\n\n### Legacy Content\n{\"Implementation approach\":\"We will analyze the difficult points of the requirements and select the appropriate open-source framework to develop the search engine. We will also integrate a large language model to provide intelligent summarization of search results.\",\"File list\":[\"main.py\",\"search_engine.py\",\"index.py\",\"ranking.py\",\"summary.py\",\"knowledge_base.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\"Anything UNCLEAR\":\"Clarification needed on third-party API integration, optimization techniques, and security measures.\"}\n\n### New Requirements\n## 原始需求\n```python\n\"\"\"\n我们希望开发一个基于大语言模型与私有知识库的搜索引擎。该搜索引擎应当能根据用户输入的查询进行智能搜索,并基于大语言模型对搜索结果进行总结,以便用户能够快速获取他们所需要的信息。该搜索引擎应当能够处理大规模的数据,同时保持搜索结果的准确性和相关性。我们希望这个产品能够降低用户在查找、筛选和理解信息时的工作负担,提高他们的工作效率。\n\"\"\"\n```\n\n## 产品目标\n```python\n[\n \"提供高准确性、高相关性的搜索结果,满足用户的查询需求\",\n \"基于大语言模型对搜索结果进行智能总结,帮助用户快速获取所需信息\",\n \"处理大规模数据,保证搜索的速度和效率,提高用户的工作效率\"\n]\n```\n\n## 用户故事\n```python\n[\n \"假设用户是一名研究员,他正在为一项关于全球气候变化的报告做研究。他输入了'全球气候变化的最新研究',我们的搜索引擎快速返回了相关的文章、报告、数据集等。并且基于大语言模型对这些信息进行了智能总结,研究员可以快速了解到最新的研究趋势和发现。\",\n \"用户是一名学生,正在为即将到来的历史考试复习。他输入了'二战的主要战役',搜索引擎返回了相关的资料,大语言模型总结出主要战役的时间、地点、结果等关键信息,帮助学生快速记忆。\",\n \"用户是一名企业家,他正在寻找关于最新的市场趋势信息。他输入了'2023年人工智能市场趋势',搜索引擎返回了各种报告、新闻和分析文章。大语言模型对这些信息进行了总结,用户能够快速了解到市场的最新动态和趋势。\"\n]\n```\n\n## 竞品分析\n```python\n[\n \"Google Search:Google搜索是市场上最主要的搜索引擎,它能够提供海量的搜索结果。但Google搜索并不提供搜索结果的总结功能,用户需要自己去阅读和理解搜索结果。\",\n \"Microsoft Bing:Bing搜索也能提供丰富的搜索结果,同样没有提供搜索结果的总结功能。\",\n \"Wolfram Alpha:Wolfram Alpha是一个基于知识库的计算型搜索引擎,能够针对某些特定类型的查询提供直接的答案和总结,但它的知识库覆盖范围有限,无法处理大规模的数据。\"\n]\n```\n\n## 开发需求池\n```python\n[\n (\"开发基于大语言模型的智能总结功能\", 5),\n (\"开发搜索引擎核心算法,包括索引构建、查询处理、结果排序等\", 7),\n (\"设计和实现用户界面,包括查询输入、搜索结果展示、总结结果展示等\", 3),\n (\"构建和维护私有知识库,包括数据采集、清洗、更新等\", 7),\n (\"优化搜索引擎性能,包括搜索速度、准确性、相关性等\", 6),\n (\"开发用户反馈机制,包括反馈界面、反馈处理等\", 2),\n (\"开发安全防护机制,防止恶意查询和攻击\", 3),\n (\"集成大语言模型,包括模型选择、优化、更新等\", 5),\n (\"进行大规模的测试,包括功能测试、性能测试、压力测试等\", 5),\n (\"开发数据监控和日志系统,用于监控搜索引擎的运行状态和性能\", 4)\n]\n```\n\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Refined Implementation Approach\": \"We will refine ...\",\n \"Refined File list\": [\n \"main.py\",\n \"game.py\",\n \"new_feature.py\"\n ],\n \"Refined Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n +search(query: str) str\\n }\\n class Index {\\n -KnowledgeBase knowledge_base\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n Main --> SearchEngine\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n Index --> KnowledgeBase\\n\",\n \"Refined Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE-->>M: return summary\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, ...\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Refined Implementation Approach: # Update and extend the original implementation approach to reflect the evolving challenges and requirements due to incremental development. Outline the steps involved in the implementation process with the detailed strategies.\n- Refined File list: typing.List[str] # Update and expand the original file list including only relative paths. Up to 2 files can be added.Ensure that the refined file list reflects the evolving structure of the project.\n- Refined Data structures and interfaces: # Update and extend the existing mermaid classDiagram code syntax to incorporate new classes, methods (including __init__), and functions with precise type annotations. Delineate additional relationships between classes, ensuring clarity and adherence to PEP8 standards.Retain content that is not related to incremental development but important for consistency and clarity.\n- Refined Program call flow: # Extend the existing sequenceDiagram code syntax with detailed information, accurately covering theCRUD and initialization of each object. Ensure correct syntax usage and reflect the incremental changes introducedin the classes and API defined above. Retain content that is not related to incremental development but important for consistency and clarity.\n- Anything UNCLEAR: # Mention unclear project aspects, then try to clarify it.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Refined Implementation Approach\": \"We will refine the existing implementation approach to incorporate the new requirements for developing a search engine based on a large language model and a private knowledge base. We will update and extend the original implementation approach to reflect the evolving challenges and requirements due to incremental development. We will outline the steps involved in the implementation process with detailed strategies to ensure the search engine meets the new goals of providing high-accuracy and high-relevance search results, intelligent summarization of search results, and efficient handling of large-scale data.\",\n \"Refined File list\": [\n \"main.py\",\n \"search_engine.py\",\n \"index.py\",\n \"ranking.py\",\n \"summary.py\",\n \"knowledge_base.py\",\n \"user_interface.py\"\n ],\n \"Refined Data structures and interfaces\": \"\\nclassDiagram\\n class Main {\\n -SearchEngine search_engine\\n -UserInterface user_interface\\n +main() str\\n }\\n class SearchEngine {\\n -Index index\\n -Ranking ranking\\n -Summary summary\\n -KnowledgeBase knowledge_base\\n +search(query: str) str\\n }\\n class Index {\\n +create_index(data: dict)\\n +query_index(query: str) list\\n }\\n class Ranking {\\n +rank_results(results: list) list\\n }\\n class Summary {\\n +summarize_results(results: list) str\\n }\\n class KnowledgeBase {\\n +update(data: dict)\\n +fetch_data(query: str) dict\\n }\\n class UserInterface {\\n +display_search_results(results: str)\\n +display_summary(summary: str)\\n }\\n Main --> SearchEngine\\n Main --> UserInterface\\n SearchEngine --> Index\\n SearchEngine --> Ranking\\n SearchEngine --> Summary\\n SearchEngine --> KnowledgeBase\\n\",\n \"Refined Program call flow\": \"\\nsequenceDiagram\\n participant M as Main\\n participant SE as SearchEngine\\n participant UI as UserInterface\\n participant I as Index\\n participant R as Ranking\\n participant S as Summary\\n participant KB as KnowledgeBase\\n M->>UI: main()\\n UI->>M: return results\\n M->>SE: search(query)\\n SE->>I: query_index(query)\\n I->>KB: fetch_data(query)\\n KB-->>I: return data\\n I-->>SE: return results\\n SE->>R: rank_results(results)\\n R-->>SE: return ranked_results\\n SE->>S: summarize_results(ranked_results)\\n S-->>SE: return summary\\n SE->>UI: display_search_results(results)\\n UI->>SE: display_summary(summary)\\n\",\n \"Anything UNCLEAR\": \"Clarification needed on third-party API integration, optimization techniques, and security measures.\"\n}\n[/CONTENT]", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/knowledge_base.py\n```python\n## smart_search_engine/knowledge_base.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/knowledge_base.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/knowledge_base.py\n\nclass KnowledgeBase:\n def __init__(self, data={}):\n self.data = data\n\n def update(self, data: dict):\n \"\"\"\n Update the knowledge base with new data.\n \n Args:\n data (dict): The new data to be added to the knowledge base.\n \"\"\"\n self.data.update(data)\n\n def fetch_data(self, query: str) -> dict:\n \"\"\"\n Fetch data from the knowledge base based on the query.\n \n Args:\n query (str): The query to search for in the knowledge base.\n \n Returns:\n dict: The data fetched from the knowledge base based on the query.\n \"\"\"\n return self.data.get(query, {})\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/index.py\n```python\n## smart_search_engine/index.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/index.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/index.py\n\nfrom smart_search_engine.knowledge_base import KnowledgeBase\n\nclass Index:\n def __init__(self, knowledge_base: KnowledgeBase):\n self.knowledge_base = knowledge_base\n self.index_data = {}\n\n def create_index(self, data: dict):\n self.index_data = data\n\n def query_index(self, query: str) -> list:\n if query in self.index_data:\n return self.index_data[query]\n else:\n return []\n\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/ranking.py\n```python\n## smart_search_engine/ranking.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/ranking.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/ranking.py\n\nclass Ranking:\n def __init__(self):\n pass\n\n def rank_results(self, results: list) -> list:\n \"\"\"\n Rank the search results based on relevance.\n\n Args:\n results (list): List of search results.\n\n Returns:\n list: Ranked list of search results.\n \"\"\"\n # Implement ranking algorithm here\n ranked_results = self._custom_ranking_algorithm(results)\n return ranked_results\n\n def _custom_ranking_algorithm(self, results: list) -> list:\n \"\"\"\n Custom ranking algorithm to rank search results.\n\n Args:\n results (list): List of search results.\n\n Returns:\n list: Ranked list of search results.\n \"\"\"\n # Implement custom ranking algorithm here\n return results\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/summary.py\n```python\n## smart_search_engine/summary.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/summary.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/summary.py\n\nclass Summary:\n def __init__(self):\n pass\n\n def summarize_results(self, results: list) -> str:\n \"\"\"\n Summarize the search results.\n\n Args:\n - results (list): List of search results.\n\n Returns:\n - str: Summarized results.\n \"\"\"\n summary = \"\"\n for result in results:\n summary += result + \"\\n\"\n return summary\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/search.py\n```python\n## smart_search_engine/search.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/search.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/search.py\n\nfrom smart_search_engine.index import Index\nfrom smart_search_engine.ranking import Ranking\nfrom smart_search_engine.summary import Summary\n\nclass SearchEngine:\n def __init__(self):\n self.index = Index()\n self.ranking = Ranking()\n self.summary = Summary()\n\n def search(self, query: str) -> str:\n results = self.index.query_index(query)\n ranked_results = self.ranking.rank_results(results)\n summary = self.summary.summarize_results(ranked_results)\n return summary\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/main.py\n```python\n## smart_search_engine/main.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/main.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/main.py\n\nfrom smart_search_engine.search import SearchEngine\n\nclass Main:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def main(self) -> str:\n query = input(\"Enter your search query: \")\n result = self.search_engine.search(query)\n return result\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/interface.py\n```python\n## smart_search_engine/interface.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/interface.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/interface.py\n\nfrom .search import SearchEngine\n\nclass UserInterface:\n def __init__(self):\n self.search_engine = SearchEngine()\n\n def get_user_input(self) -> str:\n # Get user input from the interface\n user_input = input(\"Enter your search query: \")\n return user_input\n\n def display_results(self, results: list):\n # Display search results to the user\n for result in results:\n print(result)\n\n# Ensure to follow the design and import necessary modules as needed.\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/user_feedback.py\n```python\n## smart_search_engine/user_feedback.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/user_feedback.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/user_feedback.py\n\nclass UserFeedback:\n def __init__(self):\n self.feedback_data = {}\n\n def collect_feedback(self, feedback: str):\n self.feedback_data['feedback'] = feedback\n\n def get_feedback(self):\n return self.feedback_data.get('feedback', 'No feedback provided')\n\n# Code details:\n# - Created a class UserFeedback to handle user feedback.\n# - Initialized feedback_data as an empty dictionary in the constructor.\n# - Implemented collect_feedback method to store user feedback in the feedback_data dictionary.\n# - Implemented get_feedback method to retrieve the stored feedback, with a default message if no feedback is provided.\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/security.py\n```python\n## smart_search_engine/security.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/security.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/security.py\n\nclass Security:\n def __init__(self, api_key: str = \"default_api_key\"):\n self.api_key = api_key\n\n def validate_api_key(self, api_key: str) -> bool:\n \"\"\"\n Validate the API key.\n Args:\n api_key (str): The API key to be validated.\n\n Returns:\n bool: True if the API key is valid, False otherwise.\n \"\"\"\n if api_key == self.api_key:\n return True\n else:\n return False\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/testing.py\n```python\n## smart_search_engine/testing.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/testing.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/testing.py\n\nclass Testing:\n def __init__(self, test_data: dict = {}):\n self.test_data = test_data\n\n def run_tests(self):\n # Write code to run tests based on the test_data\n pass\n```", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n## Project name\n```python\n\"smart_search_engine\"\n```\n\n## Task list:\n```python\n[\n \"smart_search_engine/__init__.py\",\n \"smart_search_engine/main.py\",\n \"smart_search_engine/search.py\",\n \"smart_search_engine/index.py\",\n \"smart_search_engine/ranking.py\",\n \"smart_search_engine/summary.py\",\n \"smart_search_engine/knowledge_base.py\",\n \"smart_search_engine/interface.py\",\n \"smart_search_engine/user_feedback.py\",\n \"smart_search_engine/security.py\",\n \"smart_search_engine/testing.py\",\n \"smart_search_engine/monitoring.py\"\n]\n```\n\n## Data structures and interfaces\n```mermaid\nclassDiagram\n class Main {\n -SearchEngine search_engine\n +main() str\n }\n class SearchEngine {\n -Index index\n -Ranking ranking\n -Summary summary\n +search(query: str) str\n }\n class Index {\n -KnowledgeBase knowledge_base\n +create_index(data: dict)\n +query_index(query: str) list\n }\n class Ranking {\n +rank_results(results: list) list\n }\n class Summary {\n +summarize_results(results: list) str\n }\n class KnowledgeBase {\n +update(data: dict)\n +fetch_data(query: str) dict\n }\n Main --> SearchEngine\n SearchEngine --> Index\n SearchEngine --> Ranking\n SearchEngine --> Summary\n Index --> KnowledgeBase\n```\n\n## Program call flow\n```mermaid\nsequenceDiagram\n participant M as Main\n participant SE as SearchEngine\n participant I as Index\n participant R as Ranking\n participant S as Summary\n participant KB as KnowledgeBase\n M->>SE: search(query)\n SE->>I: query_index(query)\n I->>KB: fetch_data(query)\n KB-->>I: return data\n I-->>SE: return results\n SE->>R: rank_results(results)\n R-->>SE: return ranked_results\n SE->>S: summarize_results(ranked_results)\n S-->>SE: return summary\n SE-->>M: return summary\n```\n\n\n## Task\n{\"Logic Analysis\": \"\\n 在这个项目中,所有的模块都依赖于“SearchEngine”类,这是主入口,其他的模块(Index、Ranking和Summary)都通过它交互。另外,\\\"Index\\\"类又依赖于\\\"KnowledgeBase\\\"类,因为它需要从知识库中获取数据。\\n\\n- \\\"main.py\\\"包含\\\"Main\\\"类,是程序的入口点,它调用\\\"SearchEngine\\\"进行搜索操作,所以在其他任何模块之前,\\\"SearchEngine\\\"必须首先被定义。\\n- \\\"search.py\\\"定义了\\\"SearchEngine\\\"类,它依赖于\\\"Index\\\"、\\\"Ranking\\\"和\\\"Summary\\\",因此,这些模块需要在\\\"search.py\\\"之前定义。\\n- \\\"index.py\\\"定义了\\\"Index\\\"类,它从\\\"knowledge_base.py\\\"获取数据来创建索引,所以\\\"knowledge_base.py\\\"需要在\\\"index.py\\\"之前定义。\\n- \\\"ranking.py\\\"和\\\"summary.py\\\"相对独立,只需确保在\\\"search.py\\\"之前定义。\\n- \\\"knowledge_base.py\\\"是独立的模块,可以优先开发。\\n- \\\"interface.py\\\"、\\\"user_feedback.py\\\"、\\\"security.py\\\"、\\\"testing.py\\\"和\\\"monitoring.py\\\"看起来像是功能辅助模块,可以在主要功能模块开发完成后并行开发。\\n \", \"Task list\": [\"smart_search_engine/knowledge_base.py\", \"smart_search_engine/index.py\", \"smart_search_engine/ranking.py\", \"smart_search_engine/summary.py\", \"smart_search_engine/search.py\", \"smart_search_engine/main.py\", \"smart_search_engine/interface.py\", \"smart_search_engine/user_feedback.py\", \"smart_search_engine/security.py\", \"smart_search_engine/testing.py\", \"smart_search_engine/monitoring.py\"]}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: smart_search_engine/monitoring.py\n```python\n## smart_search_engine/monitoring.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: smart_search_engine/monitoring.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## smart_search_engine/monitoring.py\n\nclass Monitoring:\n def __init__(self, log_file: str = \"search_log.txt\"):\n self.log_file = log_file\n\n def log_search(self, query: str, results: list, summary: str):\n with open(self.log_file, \"a\") as file:\n file.write(f\"Query: {query}\\n\")\n file.write(\"Results:\\n\")\n for result in results:\n file.write(f\"- {result}\\n\")\n file.write(f\"Summary: {summary}\\n\")\n file.write(\"\\n\")\n```\n", + "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\nwrite add function to calculate two numbers\n\n## Task\n\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: \n```python\n## \n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: . Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "## Code:\n\n```python\ndef add(num1: int, num2: int) -> int:\n \"\"\"\n Add two numbers and return the result.\n\n Args:\n num1 (int): The first number to be added.\n num2 (int): The second number to be added.\n\n Returns:\n int: The sum of num1 and num2.\n \"\"\"\n return num1 + num2\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Task\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef div(a: int, b: int = 0):\n return a / b\n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. No, the code does not handle the case where the divisor 'b' is 0, which would result in a division by zero error. This should be addressed to handle this edge case.\n2. No, the code logic is not completely correct as it does not handle the division by zero scenario.\n3. Yes, the code follows the basic function implementation.\n4. Yes, the function is implemented.\n5. No, the code does not handle the necessary pre-dependencies for error handling when 'b' is 0.\n6. No, there are no methods from other files being reused in this code.\n\n## Actions\n1. Implement error handling to check if the divisor 'b' is 0 before performing the division operation.\n2. Import the necessary pre-dependencies for error handling when 'b' is 0.\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Task\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\n\ndef div(a: int, b: int = 0):\n return a / b\n\n```\n\n## Code Review: \n1. No, the code does not handle the case where the divisor 'b' is 0, which would result in a division by zero error. This should be addressed to handle this edge case.\n2. No, the code logic is not completely correct as it does not handle the division by zero scenario.\n3. Yes, the code follows the basic function implementation.\n4. Yes, the function is implemented.\n5. No, the code does not handle the necessary pre-dependencies for error handling when 'b' is 0.\n6. No, there are no methods from other files being reused in this code.\n\n## Actions\n1. Implement error handling to check if the divisor 'b' is 0 before performing the division operation.\n2. Import the necessary pre-dependencies for error handling when 'b' is 0.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## \n...\n```\n": "```python\ndef div(a: int, b: int = 0):\n if b == 0:\n raise ValueError(\"Cannot divide by zero\")\n return a / b\n```", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Task\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\ndef div(a: int, b: int = 0):\n if b == 0:\n raise ValueError(\"Cannot divide by zero\")\n return a / b\n\n```\n\n\n\n# Format example 1\n## Code Review: \n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: \n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: \n1. The code does not fully conform to the requirements. The function signature should be updated to follow the Google style standards, and the division by zero check should be handled differently.\n2. The code logic is correct, but the exception handling for division by zero should be improved.\n3. The existing code follows the basic data structures and interfaces.\n4. The function is implemented, but the exception handling for division by zero should be added.\n5. The necessary pre-dependencies have been imported.\n6. There are no methods from other files being reused in this code.\n\n## Actions\n1. Update the function signature to follow the Google style standards and improve the division by zero check as follows:\n ```python\n def divide_numbers(dividend: int, divisor: int) -> float:\n if divisor == 0:\n raise ValueError(\"Cannot divide by zero\")\n return dividend / divisor\n ```\n2. No further action is required.\n\n## Code Review Result\nLBTM", + "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\ndivide two numbers\n\n## Task\n\n\n## Code Files\n\n\n\n## Code to be Reviewed: \n```Code\ndef div(a: int, b: int = 0):\n if b == 0:\n raise ValueError(\"Cannot divide by zero\")\n return a / b\n\n```\n\n## Code Review: \n1. The code does not fully conform to the requirements. The function signature should be updated to follow the Google style standards, and the division by zero check should be handled differently.\n2. The code logic is correct, but the exception handling for division by zero should be improved.\n3. The existing code follows the basic data structures and interfaces.\n4. The function is implemented, but the exception handling for division by zero should be added.\n5. The necessary pre-dependencies have been imported.\n6. There are no methods from other files being reused in this code.\n\n## Actions\n1. Update the function signature to follow the Google style standards and improve the division by zero check as follows:\n ```python\n def divide_numbers(dividend: int, divisor: int) -> float:\n if divisor == 0:\n raise ValueError(\"Cannot divide by zero\")\n return dividend / divisor\n ```\n2. No further action is required.\n\n## Code Review Result\nLBTM\n\n# Instruction: rewrite code based on the Code Review and Actions\n## Rewrite Code: CodeBlock. If it still has some bugs, rewrite with triple quotes. Do your utmost to optimize THIS SINGLE FILE. Return all completed codes and prohibit the return of unfinished codes.\n```Code\n## \n...\n```\n": "```python\ndef divide_numbers(dividend: int, divisor: int) -> float:\n if divisor == 0:\n raise ValueError(\"Cannot divide by zero\")\n return dividend / divisor\n```", + "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"issue_type\": \"BUG\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- issue_type: # Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"issue_type\": \"REQUIREMENT\"\n}\n[/CONTENT]", + "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Product Goals\": [\n \"Create a challenging and addictive gameplay\",\n \"Enhance accessibility and responsiveness for all users\",\n \"Implement visually appealing UI design\"\n ]\n}\n[/CONTENT]", + "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ]\n}\n[/CONTENT]", + "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code should handle user input and update the game board accordingly.\"\n ],\n [\n \"P0\",\n \"The game algorithm should handle the merging of tiles and the generation of new tiles with values of 2 or 4.\"\n ],\n [\n \"P1\",\n \"The game should end when the player achieves the 2048 tile or when there are no possible moves left.\"\n ],\n [\n \"P1\",\n \"The game should display the current score and the highest tile achieved by the player.\"\n ],\n [\n \"P2\",\n \"The game should have a smooth and visually appealing user interface.\"\n ]\n ]\n}\n[/CONTENT]" } \ No newline at end of file diff --git a/tests/metagpt/actions/test_action_node.py b/tests/metagpt/actions/test_action_node.py index 53de9cc75..1ec9f4f8d 100644 --- a/tests/metagpt/actions/test_action_node.py +++ b/tests/metagpt/actions/test_action_node.py @@ -8,7 +8,7 @@ from typing import List, Tuple import pytest -from pydantic import ValidationError +from pydantic import BaseModel, Field, ValidationError from metagpt.actions import Action from metagpt.actions.action_node import ActionNode, ReviewMode, ReviseMode @@ -241,6 +241,47 @@ def test_create_model_class_with_mapping(): assert value == ["game.py", "app.py", "static/css/styles.css", "static/js/script.js", "templates/index.html"] +class ToolDef(BaseModel): + tool_name: str = Field(default="a", description="tool name", examples=[]) + description: str = Field(default="b", description="tool description", examples=[]) + + +class Task(BaseModel): + task_id: int = Field(default=1, description="task id", examples=[1, 2, 3]) + name: str = Field(default="Get data from ...", description="task name", examples=[]) + dependent_task_ids: List[int] = Field(default=[], description="dependent task ids", examples=[1, 2, 3]) + tool: ToolDef = Field(default=ToolDef(), description="tool use", examples=[]) + + +class Tasks(BaseModel): + tasks: List[Task] = Field(default=[], description="tasks", examples=[]) + + +def test_action_node_from_pydantic_and_print_everything(): + node = ActionNode.from_pydantic(Task) + print("1. Tasks") + print(Task().model_dump_json(indent=4)) + print(Tasks.model_json_schema()) + print("2. Task") + print(Task.model_json_schema()) + print("3. ActionNode") + print(node) + print("4. node.compile prompt") + prompt = node.compile(context="") + assert "tool_name" in prompt, "tool_name should be in prompt" + print(prompt) + print("5. node.get_children_mapping") + print(node._get_children_mapping()) + print("6. node.create_children_class") + children_class = node._create_children_class() + print(children_class) + import inspect + + code = inspect.getsource(Tasks) + print(code) + assert "tasks" in code, "tasks should be in code" + + if __name__ == "__main__": test_create_model_class() test_create_model_class_with_mapping() diff --git a/tests/metagpt/strategy/test_solver.py b/tests/metagpt/strategy/test_solver.py new file mode 100644 index 000000000..eae4a5a2a --- /dev/null +++ b/tests/metagpt/strategy/test_solver.py @@ -0,0 +1,47 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/1/31 13:54 +@Author : alexanderwu +@File : test_solver.py +""" +import pytest + +from metagpt.actions.action_graph import ActionGraph +from metagpt.llm import LLM +from metagpt.strategy.search_space import SearchSpace +from metagpt.strategy.solver import NaiveSolver + + +@pytest.mark.asyncio +async def test_solver(): + from metagpt.actions.write_prd_an import ( + COMPETITIVE_ANALYSIS, + ISSUE_TYPE, + PRODUCT_GOALS, + REQUIREMENT_POOL, + ) + + graph = ActionGraph() + graph.add_node(ISSUE_TYPE) + graph.add_node(PRODUCT_GOALS) + graph.add_node(COMPETITIVE_ANALYSIS) + graph.add_node(REQUIREMENT_POOL) + graph.add_edge(ISSUE_TYPE, PRODUCT_GOALS) + graph.add_edge(PRODUCT_GOALS, COMPETITIVE_ANALYSIS) + graph.add_edge(PRODUCT_GOALS, REQUIREMENT_POOL) + graph.add_edge(COMPETITIVE_ANALYSIS, REQUIREMENT_POOL) + search_space = SearchSpace() + llm = LLM() + context = "Create a 2048 game" + solver = NaiveSolver(graph, search_space, llm, context) + await solver.solve() + + print("## graph.nodes") + print(graph.nodes) + for k, v in graph.nodes.items(): + print(f"{v.key} | prevs: {[i.key for i in v.prevs]} | nexts: {[i.key for i in v.nexts]}") + + assert len(graph.nodes) == 4 + assert len(graph.execution_order) == 4 + assert graph.execution_order == [ISSUE_TYPE.key, PRODUCT_GOALS.key, COMPETITIVE_ANALYSIS.key, REQUIREMENT_POOL.key] From 16f54abb3df52a1fb19e96f15a7938aec7ffc1c9 Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 31 Jan 2024 14:01:25 +0800 Subject: [PATCH 523/637] use gpt-4-turbo-preview as default --- config/config2.yaml | 2 +- config/config2.yaml.example | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/config/config2.yaml b/config/config2.yaml index 5e7f34809..2c4ca636f 100644 --- a/config/config2.yaml +++ b/config/config2.yaml @@ -1,3 +1,3 @@ llm: api_key: "YOUR_API_KEY" - model: "gpt-3.5-turbo-1106" \ No newline at end of file + model: "gpt-4-turbo-preview" # or gpt-3.5-turbo-1106 / gpt-4-1106-preview \ No newline at end of file diff --git a/config/config2.yaml.example b/config/config2.yaml.example index 35575e5a5..1a406e756 100644 --- a/config/config2.yaml.example +++ b/config/config2.yaml.example @@ -2,7 +2,7 @@ llm: api_type: "openai" base_url: "YOUR_BASE_URL" api_key: "YOUR_API_KEY" - model: "gpt-3.5-turbo-1106" # or gpt-4-1106-preview + model: "gpt-4-turbo-preview" # or gpt-3.5-turbo-1106 / gpt-4-1106-preview proxy: "YOUR_PROXY" From 30de3b4d6498cd8ebf2d9efdeb9e6f0a5d861a5a Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 31 Jan 2024 15:08:40 +0800 Subject: [PATCH 524/637] fix message init bug --- metagpt/schema.py | 2 +- tests/metagpt/test_schema.py | 274 +++++++++++++++++------------------ 2 files changed, 138 insertions(+), 138 deletions(-) diff --git a/metagpt/schema.py b/metagpt/schema.py index e6a447fba..08f97be94 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -327,7 +327,7 @@ class AIMessage(Message): """ def __init__(self, content: str): - super().__init__(content, "assistant") + super().__init__(content=content, role="assistant") class Task(BaseModel): diff --git a/tests/metagpt/test_schema.py b/tests/metagpt/test_schema.py index 17d2bb22c..a8fa27151 100644 --- a/tests/metagpt/test_schema.py +++ b/tests/metagpt/test_schema.py @@ -46,6 +46,143 @@ def test_messages(): assert all([i in text for i in roles]) +def test_message(): + Message("a", role="v1") + + m = Message(content="a", role="v1") + v = m.dump() + d = json.loads(v) + assert d + assert d.get("content") == "a" + assert d.get("role") == "v1" + m.role = "v2" + v = m.dump() + assert v + m = Message.load(v) + assert m.content == "a" + assert m.role == "v2" + + m = Message(content="a", role="b", cause_by="c", x="d", send_to="c") + assert m.content == "a" + assert m.role == "b" + assert m.send_to == {"c"} + assert m.cause_by == "c" + m.sent_from = "e" + assert m.sent_from == "e" + + m.cause_by = "Message" + assert m.cause_by == "Message" + m.cause_by = Action + assert m.cause_by == any_to_str(Action) + m.cause_by = Action() + assert m.cause_by == any_to_str(Action) + m.content = "b" + assert m.content == "b" + + +def test_routes(): + m = Message(content="a", role="b", cause_by="c", x="d", send_to="c") + m.send_to = "b" + assert m.send_to == {"b"} + m.send_to = {"e", Action} + assert m.send_to == {"e", any_to_str(Action)} + + +def test_message_serdeser(): + out_mapping = {"field3": (str, ...), "field4": (list[str], ...)} + out_data = {"field3": "field3 value3", "field4": ["field4 value1", "field4 value2"]} + ic_obj = ActionNode.create_model_class("code", out_mapping) + + message = Message(content="code", instruct_content=ic_obj(**out_data), role="engineer", cause_by=WriteCode) + message_dict = message.model_dump() + assert message_dict["cause_by"] == "metagpt.actions.write_code.WriteCode" + assert message_dict["instruct_content"] == { + "class": "code", + "mapping": {"field3": "(, Ellipsis)", "field4": "(list[str], Ellipsis)"}, + "value": {"field3": "field3 value3", "field4": ["field4 value1", "field4 value2"]}, + } + new_message = Message.model_validate(message_dict) + assert new_message.content == message.content + assert new_message.instruct_content.model_dump() == message.instruct_content.model_dump() + assert new_message.instruct_content == message.instruct_content # TODO + assert new_message.cause_by == message.cause_by + assert new_message.instruct_content.field3 == out_data["field3"] + + message = Message(content="code") + message_dict = message.model_dump() + new_message = Message(**message_dict) + assert new_message.instruct_content is None + assert new_message.cause_by == "metagpt.actions.add_requirement.UserRequirement" + assert not Message.load("{") + + +def test_document(): + doc = Document(root_path="a", filename="b", content="c") + meta_doc = doc.get_meta() + assert doc.root_path == meta_doc.root_path + assert doc.filename == meta_doc.filename + assert meta_doc.content == "" + + +@pytest.mark.asyncio +async def test_message_queue(): + mq = MessageQueue() + val = await mq.dump() + assert val == "[]" + mq.push(Message(content="1")) + mq.push(Message(content="2中文测试aaa")) + msg = mq.pop() + assert msg.content == "1" + + val = await mq.dump() + assert val + new_mq = MessageQueue.load(val) + assert new_mq.pop_all() == mq.pop_all() + + +@pytest.mark.parametrize( + ("file_list", "want"), + [ + ( + [f"{SYSTEM_DESIGN_FILE_REPO}/a.txt", f"{TASK_FILE_REPO}/b.txt"], + CodeSummarizeContext( + design_filename=f"{SYSTEM_DESIGN_FILE_REPO}/a.txt", task_filename=f"{TASK_FILE_REPO}/b.txt" + ), + ) + ], +) +def test_CodeSummarizeContext(file_list, want): + ctx = CodeSummarizeContext.loads(file_list) + assert ctx == want + m = {ctx: ctx} + assert want in m + + +def test_class_view(): + attr_a = ClassAttribute(name="a", value_type="int", default_value="0", visibility="+", abstraction=True) + assert attr_a.get_mermaid(align=1) == "\t+int a=0*" + attr_b = ClassAttribute(name="b", value_type="str", default_value="0", visibility="#", static=True) + assert attr_b.get_mermaid(align=0) == '#str b="0"$' + class_view = ClassView(name="A") + class_view.attributes = [attr_a, attr_b] + + method_a = ClassMethod(name="run", visibility="+", abstraction=True) + assert method_a.get_mermaid(align=1) == "\t+run()*" + method_b = ClassMethod( + name="_test", + visibility="#", + static=True, + args=[ClassAttribute(name="a", value_type="str"), ClassAttribute(name="b", value_type="int")], + return_type="str", + ) + assert method_b.get_mermaid(align=0) == "#_test(str a,int b):str$" + class_view.methods = [method_a, method_b] + assert ( + class_view.get_mermaid(align=0) + == 'class A{\n\t+int a=0*\n\t#str b="0"$\n\t+run()*\n\t#_test(str a,int b):str$\n}\n' + ) + + class TestPlan: def test_add_tasks_ordering(self): plan = Plan(goal="") @@ -214,142 +351,5 @@ class TestPlan: assert plan.current_task_id == "2" -def test_message(): - Message("a", role="v1") - - m = Message(content="a", role="v1") - v = m.dump() - d = json.loads(v) - assert d - assert d.get("content") == "a" - assert d.get("role") == "v1" - m.role = "v2" - v = m.dump() - assert v - m = Message.load(v) - assert m.content == "a" - assert m.role == "v2" - - m = Message(content="a", role="b", cause_by="c", x="d", send_to="c") - assert m.content == "a" - assert m.role == "b" - assert m.send_to == {"c"} - assert m.cause_by == "c" - m.sent_from = "e" - assert m.sent_from == "e" - - m.cause_by = "Message" - assert m.cause_by == "Message" - m.cause_by = Action - assert m.cause_by == any_to_str(Action) - m.cause_by = Action() - assert m.cause_by == any_to_str(Action) - m.content = "b" - assert m.content == "b" - - -def test_routes(): - m = Message(content="a", role="b", cause_by="c", x="d", send_to="c") - m.send_to = "b" - assert m.send_to == {"b"} - m.send_to = {"e", Action} - assert m.send_to == {"e", any_to_str(Action)} - - -def test_message_serdeser(): - out_mapping = {"field3": (str, ...), "field4": (list[str], ...)} - out_data = {"field3": "field3 value3", "field4": ["field4 value1", "field4 value2"]} - ic_obj = ActionNode.create_model_class("code", out_mapping) - - message = Message(content="code", instruct_content=ic_obj(**out_data), role="engineer", cause_by=WriteCode) - message_dict = message.model_dump() - assert message_dict["cause_by"] == "metagpt.actions.write_code.WriteCode" - assert message_dict["instruct_content"] == { - "class": "code", - "mapping": {"field3": "(, Ellipsis)", "field4": "(list[str], Ellipsis)"}, - "value": {"field3": "field3 value3", "field4": ["field4 value1", "field4 value2"]}, - } - new_message = Message.model_validate(message_dict) - assert new_message.content == message.content - assert new_message.instruct_content.model_dump() == message.instruct_content.model_dump() - assert new_message.instruct_content == message.instruct_content # TODO - assert new_message.cause_by == message.cause_by - assert new_message.instruct_content.field3 == out_data["field3"] - - message = Message(content="code") - message_dict = message.model_dump() - new_message = Message(**message_dict) - assert new_message.instruct_content is None - assert new_message.cause_by == "metagpt.actions.add_requirement.UserRequirement" - assert not Message.load("{") - - -def test_document(): - doc = Document(root_path="a", filename="b", content="c") - meta_doc = doc.get_meta() - assert doc.root_path == meta_doc.root_path - assert doc.filename == meta_doc.filename - assert meta_doc.content == "" - - -@pytest.mark.asyncio -async def test_message_queue(): - mq = MessageQueue() - val = await mq.dump() - assert val == "[]" - mq.push(Message(content="1")) - mq.push(Message(content="2中文测试aaa")) - msg = mq.pop() - assert msg.content == "1" - - val = await mq.dump() - assert val - new_mq = MessageQueue.load(val) - assert new_mq.pop_all() == mq.pop_all() - - -@pytest.mark.parametrize( - ("file_list", "want"), - [ - ( - [f"{SYSTEM_DESIGN_FILE_REPO}/a.txt", f"{TASK_FILE_REPO}/b.txt"], - CodeSummarizeContext( - design_filename=f"{SYSTEM_DESIGN_FILE_REPO}/a.txt", task_filename=f"{TASK_FILE_REPO}/b.txt" - ), - ) - ], -) -def test_CodeSummarizeContext(file_list, want): - ctx = CodeSummarizeContext.loads(file_list) - assert ctx == want - m = {ctx: ctx} - assert want in m - - -def test_class_view(): - attr_a = ClassAttribute(name="a", value_type="int", default_value="0", visibility="+", abstraction=True) - assert attr_a.get_mermaid(align=1) == "\t+int a=0*" - attr_b = ClassAttribute(name="b", value_type="str", default_value="0", visibility="#", static=True) - assert attr_b.get_mermaid(align=0) == '#str b="0"$' - class_view = ClassView(name="A") - class_view.attributes = [attr_a, attr_b] - - method_a = ClassMethod(name="run", visibility="+", abstraction=True) - assert method_a.get_mermaid(align=1) == "\t+run()*" - method_b = ClassMethod( - name="_test", - visibility="#", - static=True, - args=[ClassAttribute(name="a", value_type="str"), ClassAttribute(name="b", value_type="int")], - return_type="str", - ) - assert method_b.get_mermaid(align=0) == "#_test(str a,int b):str$" - class_view.methods = [method_a, method_b] - assert ( - class_view.get_mermaid(align=0) - == 'class A{\n\t+int a=0*\n\t#str b="0"$\n\t+run()*\n\t#_test(str a,int b):str$\n}\n' - ) - - if __name__ == "__main__": pytest.main([__file__, "-s"]) From b585064edc82a1b08cbe2537793d5d97895ccebf Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 31 Jan 2024 15:36:26 +0800 Subject: [PATCH 525/637] rm redundant --- tests/metagpt/roles/test_code_interpreter.py | 15 ++------------- 1 file changed, 2 insertions(+), 13 deletions(-) diff --git a/tests/metagpt/roles/test_code_interpreter.py b/tests/metagpt/roles/test_code_interpreter.py index aeb7070fd..b78f7a9ef 100644 --- a/tests/metagpt/roles/test_code_interpreter.py +++ b/tests/metagpt/roles/test_code_interpreter.py @@ -3,24 +3,13 @@ import pytest from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter -# from metagpt.const import DATA_PATH - @pytest.mark.asyncio -@pytest.mark.parametrize("use_tools", [(True)]) -async def test_code_interpreter(use_tools): +async def test_code_interpreter(): requirement = "Run data analysis on sklearn Iris dataset, include a plot" - # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - # data_path = f"{DATA_PATH}/titanic" - # requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" - # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." - # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" - # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." tools = [] - # tools = ["FillMissingValue", "CatCross", "a"] - ci = CodeInterpreter(auto_run=True, use_tools=use_tools, tools=tools) + ci = CodeInterpreter(auto_run=True, use_tools=True, tools=tools) rsp = await ci.run(requirement) logger.info(rsp) assert len(rsp.content) > 0 From fc412e55a3ebd58ab5e79ca98b37d700b0994f36 Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Wed, 31 Jan 2024 15:58:26 +0800 Subject: [PATCH 526/637] fix tests/metagpt/learn/test_google_search.py error --- metagpt/learn/google_search.py | 2 +- metagpt/tools/search_engine_serpapi.py | 2 + metagpt/tools/search_engine_serper.py | 2 + tests/data/search_rsp_cache.json | 118 +++++++++++++++++++++- tests/metagpt/learn/test_google_search.py | 24 ++--- tests/mock/mock_aiohttp.py | 4 + 6 files changed, 135 insertions(+), 17 deletions(-) diff --git a/metagpt/learn/google_search.py b/metagpt/learn/google_search.py index 3f356f7dd..399c14de4 100644 --- a/metagpt/learn/google_search.py +++ b/metagpt/learn/google_search.py @@ -8,5 +8,5 @@ async def google_search(query: str, max_results: int = 6, **kwargs): :param max_results: The number of search results to retrieve :return: The web search results in markdown format. """ - results = await SearchEngine().run(query, max_results=max_results, as_string=False) + results = await SearchEngine(**kwargs).run(query, max_results=max_results, as_string=False) return "\n".join(f"{i}. [{j['title']}]({j['link']}): {j['snippet']}" for i, j in enumerate(results, 1)) diff --git a/metagpt/tools/search_engine_serpapi.py b/metagpt/tools/search_engine_serpapi.py index a8d5b49d0..8d27d493d 100644 --- a/metagpt/tools/search_engine_serpapi.py +++ b/metagpt/tools/search_engine_serpapi.py @@ -61,9 +61,11 @@ class SerpAPIWrapper(BaseModel): if not self.aiosession: async with aiohttp.ClientSession() as session: async with session.get(url, params=params) as response: + response.raise_for_status() res = await response.json() else: async with self.aiosession.get(url, params=params) as response: + response.raise_for_status() res = await response.json() return res diff --git a/metagpt/tools/search_engine_serper.py b/metagpt/tools/search_engine_serper.py index 39cb936b8..71ee2f4f9 100644 --- a/metagpt/tools/search_engine_serper.py +++ b/metagpt/tools/search_engine_serper.py @@ -55,9 +55,11 @@ class SerperWrapper(BaseModel): if not self.aiosession: async with aiohttp.ClientSession() as session: async with session.post(url, data=payloads, headers=headers) as response: + response.raise_for_status() res = await response.json() else: async with self.aiosession.get.post(url, data=payloads, headers=headers) as response: + response.raise_for_status() res = await response.json() return res diff --git a/tests/data/search_rsp_cache.json b/tests/data/search_rsp_cache.json index 822fb2069..7b4cc583c 100644 --- a/tests/data/search_rsp_cache.json +++ b/tests/data/search_rsp_cache.json @@ -875,5 +875,121 @@ "curl-cffi-POST-https://duckduckgo.com-{\"data\": {\"q\": \"Dataiku vs DataRobot features\"}}": "Dataiku vs DataRobot features at DuckDuckGo
", "curl-cffi-GET-https://links.duckduckgo.com/d.js-{\"params\": {\"bing_market\": \"wt-WT\", \"df\": null, \"ex\": \"-1\", \"kl\": \"wt-wt\", \"l\": \"wt-wt\", \"q\": \"Dataiku vs DataRobot features\", \"s\": \"0\", \"sp\": \"0\", \"vqd\": \"4-334935250614046875026454141242803242982\"}}": "if (DDG.deep && DDG.deep.setUpstream) DDG.deep.setUpstream(\"bingv7aa\");DDG.deep.bn={'ivc':1};if (DDG.pageLayout) DDG.pageLayout.load('a',[{\"a\":\"\\u9ad8\\u7cbe\\u5ea6\\u306a\\u6a5f\\u68b0\\u5b66\\u7fd2\\u30e2\\u30c7\\u30eb\\u3092\\u69cb\\u7bc9\\u3001\\u5b9f\\u88c5\\u3001\\u904b\\u7528\\u3002DataRobot\\u306f\\u793e\\u5185\\u30c7\\u30fc\\u30bf\\u304b\\u3089\\u65b0\\u3057\\u3044\\u4fa1\\u5024\\u3092\\u5275\\u9020\\u3057\\u307e\\u3059. DataRobot\\u306f\\u4f01\\u696d\\u306e\\u8ab2\\u984c\\u89e3\\u6c7a\\u306b\\u7279\\u5316\\u3002\\u610f\\u601d\\u6c7a\\u5b9a\\u306e\\u81ea\\u52d5\\u5316\\u304b\\u3089\\u9700\\u8981\\u4e88\\u6e2c\\u3001\\u8981\\u56e0\\u5206\\u6790\\u307e\\u3067\\u3053\\u306a\\u3059AI\\u30c4\\u30fc\\u30eb\",\"adext\":{\"callout\":{\"t\":\"Data Science Guardrails \\u00b7 Applied AI Expertise \\u00b7 Trusted by Fortune 50\",\"tid\":\"6\"},\"filterlinks\":{\"l\":[],\"tid\":\"\"},\"sitelinks\":{\"l\":[{\"snippet\":\"Explore the DataRobot AI Platform Get Started With a 30-Day Trial\",\"targetUrl\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=uchwI3Eul8XsE%2DSUlPqxXg%3D%3D&rut=2381550f96a087800d427735905717264a1a708643136f2736a970e740068621&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8BGV0WifLHqlNArHdJt3WDTVUCUzDyrVI_ULomBTgn_xk1MKGRFElGY7vQ8fpE4l__S3CnH6%2D2cXlBQayeIz9CbLU7C4XEu8BgG6oZNQ6EtjG6vrYe5hjw1GZN7VBIkj6nn%2DsoUXy14mVbvkM5ojXVf8oeoz8pwdOc4ANH2TiL9vqJe6Lud2IZXvxJf1I%2DA935XcPQobPZKQaFNFMyygI3Y4TW8k%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZnRyaWFsJTJmJTNmdXRtX21lZGl1bSUzZHNlYXJjaCUyNnV0bV9zb3VyY2UlM2RiaW5nJTI2dXRtX2NhbXBhaWduJTNkRnJlZVRyaWFsMjAyM1dXMDgxNkdQU2FkZXh0JTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZDFmMzU0ODE0ODNmMTEyM2Y5NGMzMmRiNzdjZjk5OWFm%26rlid%3D1f35481483f1123f94c32db77cf999af&vqd=4-25671318592048362755712261648304518289&iurl=%7B1%7DIG%3D3EB403B8C4EA42F4B7FF0CE90CB46EF0%26CID%3D2F20CB6F269D6DD02331DF69279D6C12%26ID%3DDevEx%2C5064.1\",\"text\":\"DataRobot Free Trial\"},{\"snippet\":\"Unlock Your AI Success in 2023 Tips on the Path of Value-Driven AI\",\"targetUrl\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=uchwI3Eul8XsE%2DSUlPqxXg%3D%3D&rut=08def2477dd7311fbcffe4c409d28fcdbe68925a50cd2894a7502f8a11785352&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8lYdQlfjG0%2Dh77MMyzT0CuDVUCUyAuuZDH6K8NWyD2XSLoABvrUNVChVbIVOVgzl4xdT3EEUvHgd9P_FWLUDT2My42qKUP3iV87B7hLXXHLdGf7yjst8tWjp%2DcaQz3uiI0c5oom%2DRo8D7A4nohZAtS9199RQLYbNcbOpJnrNMCFmz6EiWk7JqMQ9DE1t9AjaMUWEkEV%2D3W2e8XmBq5bKtRsWnT0E%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZnJlc291cmNlcyUyZmFpc3VjY2VzczIwMjMlMmYlM2Z1dG1fbWVkaXVtJTNkc2VhcmNoJTI2dXRtX3NvdXJjZSUzZGJpbmclMjZ1dG1fY2FtcGFpZ24lM2RDb250ZW50MTBLZXlzdG9BSVN1Y2Nlc3MyMDIzV1cwNTIyR1BTYWRleHQlMjZ1dG1fdGVybSUzZGRhdGFyb2JvdCUyNnV0bV9jb250ZW50JTNkYWRfZXh0JTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZDI5Zjc0NWY1MzNiNzE2NDU5ZGY0MjA1NmNjYmYyYWU0%26rlid%3D29f745f533b716459df42056ccbf2ae4&vqd=4-333465595216651803104351585568313334233&iurl=%7B1%7DIG%3D3EB403B8C4EA42F4B7FF0CE90CB46EF0%26CID%3D2F20CB6F269D6DD02331DF69279D6C12%26ID%3DDevEx%2C5066.1\",\"text\":\"10 Keys to AI Success\"},{\"snippet\":\"Our Platform Includes Four Fully Integrated Products. Read More.\",\"targetUrl\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=uchwI3Eul8XsE%2DSUlPqxXg%3D%3D&rut=fbe7591a97a4b400635f8cfafd71893553c70fc90218355b7d5622310d9567db&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8cB2vIW6%2D5rxeC5vl08jFZjVUCUw2oN7vfXdo8rlxVmZIfw2bF94_ya9lvPQwUYXJFtTGXBslf_XCcVTiFtj2KJzp9yzLPOdWafvxxwBzn2iwextOSL%2Daq20iQ8nZNktMLYBD1xp3WjThLdejbBCFrR_RvD1YZcHcKf5y5auyV04F_V6x_D6nUwdRYFDmdyciLcpT7JO12EZkmM%2D1buahlzuiBmw%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZnByb2R1Y3QlMmYlM2ZjYW1wYWlnbmlkJTNkNTMwNzA4MDk5JTI2YWRncm91cGlkJTNkMTM1MDIwMjc3NDIxNzY5OCUyNmFkaWQlM2QlMjZtc2Nsa2lkJTNkMGZhOTg4ZjJkYWU2MWE3MGJhOTVlZDUxMjVlZWFlNDA%26rlid%3D0fa988f2dae61a70ba95ed5125eeae40&vqd=4-211419575679328898707892660118042825990&iurl=%7B1%7DIG%3D3EB403B8C4EA42F4B7FF0CE90CB46EF0%26CID%3D2F20CB6F269D6DD02331DF69279D6C12%26ID%3DDevEx%2C5068.1\",\"text\":\"Product Overview\"}],\"tid\":\"7\\t9[8]\\t11[10]\\t13[12]\",\"type\":\"EnhancedSiteLink\"},\"tid\":\"1\"},\"ae\":{\"callout\":[\"Data Science Guardrails \\u00b7 Applied AI Expertise \\u00b7 Trusted by Fortune 50\"]},\"c\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=uchwI3Eul8XsE%2DSUlPqxXg%3D%3D&rut=94a279ed1549c0107c5c13f21161fd5aaa0d3f08d19e7afd2ed4a19463b69d7d&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8XX6qufLbIkEZRIFo_zgmlDVUCUwOnCSpTtxK0dn2QInSfOGU5eU24GjiRwhmSr89Qa92PcEtK2h6KVoghC%2DNwNrkANG4L6sVirCfv5kl7GPWO9gqgcdw8x5ELjGH7N2HWgbdtH%2D7TWKtxZVdVIFwYJUQDUgM_ODwTspzwBbKKLHD4EPAO5U3RDO3R_igFUlsxkeFXA%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZmpwJTJmbHAlMmZhaS1mb3ItYnVzaW5lc3MlMmYlM2Z1dG1fbWVkaXVtJTNkc2VhcmNoJTI2dXRtX3NvdXJjZSUzZGJpbmclMjZ1dG1fY2FtcGFpZ24lM2RERU1PMjAyM0FsbFByb2R1Y3RzSlAwNjI2QlBTJTI2dXRtX3Rlcm0lM2RkYXRhcm9ib3QlMjZ1dG1fY29udGVudCUzZERSX2JyYW5kZWRfcnNhJTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZGQxMGY4ZjY4ZDYxZjFiOTg2NTc1ZWFjYjI5MTczYTQ1%26rlid%3Dd10f8f68d61f1b986575eacb29173a45&vqd=4-152568096679810917558416500867559274982&iurl=%7B1%7DIG%3D3EB403B8C4EA42F4B7FF0CE90CB46EF0%26CID%3D2F20CB6F269D6DD02331DF69279D6C12%26ID%3DDevEx%2C5059.1\",\"d\":\"datarobot.com\",\"h\":0,\"i\":\"\",\"k\":0,\"m\":0,\"o\":\"\",\"p\":1,\"relevancy\":{\"abstract\":\"%E9%AB%98%E7%B2%BE%E5%BA%A6%E3%81%AA%E6%A9%9F%E6%A2%B0%E5%AD%A6%E7%BF%92%E3%83%A2%E3%83%87%E3%83%AB%E3%82%92%E6%A7%8B%E7%AF%89%E3%80%81%E5%AE%9F%E8%A3%85%E3%80%81%E9%81%8B%E7%94%A8%E3%80%82%3Cb%3EDataRobot%3C%2Fb%3E%E3%81%AF%E7%A4%BE%E5%86%85%E3%83%87%E3%83%BC%E3%82%BF%E3%81%8B%E3%82%89%E6%96%B0%E3%81%97%E3%81%84%E4%BE%A1%E5%80%A4%E3%82%92%E5%89%B5%E9%80%A0%E3%81%97%E3%81%BE%E3%81%99.%20DataRobot%E3%81%AF%E4%BC%81%E6%A5%AD%E3%81%AE%E8%AA%B2%E9%A1%8C%E8%A7%A3%E6%B1%BA%E3%81%AB%E7%89%B9%E5%8C%96%E3%80%82%E6%84%8F%E6%80%9D%E6%B1%BA%E5%AE%9A%E3%81%AE%E8%87%AA%E5%8B%95%E5%8C%96%E3%81%8B%E3%82%89%E9%9C%80%E8%A6%81%E4%BA%88%E6%B8%AC%E3%80%81%E8%A6%81%E5%9B%A0%E5%88%86%E6%9E%90%E3%81%BE%E3%81%A7%E3%81%93%E3%81%AA%E3%81%99AI%E3%83%84%E3%83%BC%E3%83%AB\",\"adx_name\":\"none\",\"is_good_v10\":0,\"q\":\"Dataiku%20vs%20DataRobot%20features\",\"q_words\":4,\"q_words_fuzzy\":0.25,\"q_words_in_ad\":1,\"root_domain\":\"datarobot.com\",\"start\":\"0\",\"title\":\"%E3%83%87%E3%83%BC%E3%82%BF%E3%81%8B%E3%82%89%E6%96%B0%E3%81%97%E3%81%84%E4%BE%A1%E5%80%A4%E3%82%92%20%2D%20%E7%A4%BE%E5%86%85%E3%83%87%E3%83%BC%E3%82%BF%E3%81%8B%E3%82%89%E4%BE%A1%E5%80%A4%E5%89%B5%E5%87%BA\"},\"s\":\"bingv7aa\",\"t\":\"\\u30c7\\u30fc\\u30bf\\u304b\\u3089\\u65b0\\u3057\\u3044\\u4fa1\\u5024\\u3092 - \\u793e\\u5185\\u30c7\\u30fc\\u30bf\\u304b\\u3089\\u4fa1\\u5024\\u5275\\u51fa\",\"tid\":\"1,6,7,9[8],11[10],13[12]\",\"u\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=uchwI3Eul8XsE%2DSUlPqxXg%3D%3D&rut=94a279ed1549c0107c5c13f21161fd5aaa0d3f08d19e7afd2ed4a19463b69d7d&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8XX6qufLbIkEZRIFo_zgmlDVUCUwOnCSpTtxK0dn2QInSfOGU5eU24GjiRwhmSr89Qa92PcEtK2h6KVoghC%2DNwNrkANG4L6sVirCfv5kl7GPWO9gqgcdw8x5ELjGH7N2HWgbdtH%2D7TWKtxZVdVIFwYJUQDUgM_ODwTspzwBbKKLHD4EPAO5U3RDO3R_igFUlsxkeFXA%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZmpwJTJmbHAlMmZhaS1mb3ItYnVzaW5lc3MlMmYlM2Z1dG1fbWVkaXVtJTNkc2VhcmNoJTI2dXRtX3NvdXJjZSUzZGJpbmclMjZ1dG1fY2FtcGFpZ24lM2RERU1PMjAyM0FsbFByb2R1Y3RzSlAwNjI2QlBTJTI2dXRtX3Rlcm0lM2RkYXRhcm9ib3QlMjZ1dG1fY29udGVudCUzZERSX2JyYW5kZWRfcnNhJTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZGQxMGY4ZjY4ZDYxZjFiOTg2NTc1ZWFjYjI5MTczYTQ1%26rlid%3Dd10f8f68d61f1b986575eacb29173a45&vqd=4-152568096679810917558416500867559274982&iurl=%7B1%7DIG%3D3EB403B8C4EA42F4B7FF0CE90CB46EF0%26CID%3D2F20CB6F269D6DD02331DF69279D6C12%26ID%3DDevEx%2C5059.1\"}], {\"page_load_url\":\"https://duckduckgo.com/y.js?ifu=%7B3%7Dappid%3D055AAD1BA669BEB8B048128DC89A107C678B527B%26rguid%3D280881b97b9245e6a74bddebc1a6cbda&iurl=%7B2%7DIG%3D3EB403B8C4EA42F4B7FF0CE90CB46EF0%26CID%3D2F20CB6F269D6DD02331DF69279D6C12%26Type%3DEvent.CPT%26DATA%3D0\",\"visibility_url\":\"https://duckduckgo.com/y.js?ivu=%7B4%7Dtype%3Dmv%26reqver%3D1.0%26rg%3D280881b97b9245e6a74bddebc1a6cbda\"});DDG.deep.signalSummary = \"\";DDG.inject('DDG.Data.languages.resultLanguages', {\"en\":[\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\",\"https://www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\",\"https://community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\",\"https://www.g2.com/compare/datarobot-vs-dataiku-dss\",\"https://www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\",\"https://comparisons.financesonline.com/datarobot-vs-dataiku-dss\",\"https://slashdot.org/software/comparison/DataRobot-vs-Dataiku-DSS/\",\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\",\"https://www.gartner.com/reviews/market/dsml-engineering-platforms/compare/dataiku-vs-datarobot\",\"https://www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\",\"https://www.trustradius.com/products/datarobot/reviews?qs=pros-and-cons\",\"https://www.getapp.com/emerging-technology-software/a/dataiku-dss/compare/datarobot/\",\"https://www.softwarereviews.com/categories/machine-learning-platforms/compare/dataiku-vs-datarobot-ai-platform\",\"https://slashdot.org/software/comparison/Alteryx-vs-DataRobot-vs-Dataiku-DSS/\",\"https://slashdot.org/software/comparison/DataRobot-vs-Databricks-vs-Dataiku-DSS/\",\"https://valohai.com/mlops-platforms-compared/\",\"https://www.dataiku.com/product/plans-and-features/\",\"https://slashdot.org/software/comparison/DataRobot-vs-Databricks-vs-Dataiku-DSS-vs-datagym/\",\"https://sourceforge.net/software/compare/C3-AI-Suite-vs-DataRobot-vs-Dataiku-DSS/\",\"https://www.softwarereviews.com/categories/machine-learning-platforms/compare/datarobot-ai-platform-vs-dataiku\",\"https://slashdot.org/software/comparison/Amazon-SageMaker-vs-DataRobot-vs-Dataiku-DSS/\",\"https://sourceforge.net/software/compare/Analance-vs-DataRobot-vs-Dataiku-DSS/\"]});DDG.deep.pageLayoutSummary = \"a1w23r1,e1\";DDG.inject('DDG.Data.languages.adLanguages', {});if (DDG.pageLayout) DDG.pageLayout.load('d',[{\"a\":\"1 Star 0% Ratings breakdown Overall Capability Score Overall Rating 4.7 ( 504 reviews) 4.7 (20) Data Access and Manipulation 4.5 (224) Data Exploration and Visualization 4.7\",\"ae\":null,\"c\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\",\"d\":\"www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\",\"da\":\"\",\"h\":0,\"i\":\"www.gartner.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku vs DataRobot 2024 | Gartner Peer Insights\",\"u\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\"},{\"a\":\"Path to AI Success Compare Dataiku DSS vs DataRobot. 103 verified user reviews and ratings of features, pros, cons, pricing, support and more.\",\"ae\":null,\"c\":\"https://www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\",\"d\":\"www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\",\"da\":\"\",\"h\":0,\"i\":\"www.trustradius.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku DSS vs DataRobot | TrustRadius\",\"u\":\"https://www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\"},{\"a\":\"General Discussion Dataiku vs DataRobot Solved! Raja Level 2 08-22-2020 03:16 AM Please enlighten me, What distinguishes Dataiku from tools like DataRobot? They appear to be similar, trying to know how dataiku has an upper hand, would make it easy for placing option to customers. 1 Reply 2 Solutions Solutions shown first - Read whole discussion\",\"ae\":null,\"c\":\"https://community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\",\"d\":\"community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\",\"da\":\"\",\"h\":0,\"i\":\"community.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Solved: Dataiku vs DataRobot - Dataiku Community\",\"u\":\"https://community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\"},{\"a\":\"DataRobot vs Dataiku DSS When assessing the two solutions, reviewers found Dataiku DSS easier to use and administer. However, reviewers preferred the ease of set up, and doing business with DataRobot overall. Reviewers felt that DataRobot meets the needs of their business better than Dataiku DSS.\",\"ae\":null,\"c\":\"https://www.g2.com/compare/datarobot-vs-dataiku-dss\",\"d\":\"www.g2.com/compare/datarobot-vs-dataiku-dss\",\"da\":\"\",\"h\":0,\"i\":\"www.g2.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare DataRobot vs. Dataiku DSS | G2\",\"u\":\"https://www.g2.com/compare/datarobot-vs-dataiku-dss\"},{\"a\":\"Quick overview Before we get into a detailed comparison, here's a quick overview of each platform. Dataiku is a cross-platform desktop application that includes a broad range of tools, such as notebooks (similar to Jupyter Notebook), workflow management (similar to Apache Airflow), and automated machine learning.\",\"ae\":null,\"c\":\"https://www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\",\"d\":\"www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\",\"da\":\"\",\"h\":0,\"i\":\"www.datarevenue.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"ML Platforms: Dataiku vs. Alteryx vs. Sagemaker vs. Datarobot\",\"u\":\"https://www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\"},{\"a\":\"Home Predictive Analysis Software DataRobot Dataiku DSS Why is FinancesOnline free Compare DataRobot vs Dataiku DSS What is better DataRobot or Dataiku DSS? Examining products to find the best Predictive Analysis Software does not always have to be tough.\",\"ae\":null,\"c\":\"https://comparisons.financesonline.com/datarobot-vs-dataiku-dss\",\"d\":\"comparisons.financesonline.com/datarobot-vs-dataiku-dss\",\"da\":\"\",\"h\":0,\"i\":\"comparisons.financesonline.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"DataRobot vs Dataiku DSS 2024 Comparison | FinancesOnline\",\"u\":\"https://comparisons.financesonline.com/datarobot-vs-dataiku-dss\"},{\"a\":\"Machine Learning Software Dataiku vs DataRobot Dataiku vs DataRobot Share How Capterra Verifies Reviews Pricing Best for Screenshots Features Reviews Pros & Cons Deployment & Support Alternatives Company Details Dataiku VISIT PROFILE DataRobot VISIT PROFILE Pricing Starting from $ 0.01 /Year Pricing Model: Not provided by vendor Free Trial\",\"ae\":null,\"c\":\"https://www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\",\"d\":\"www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\",\"da\":\"translations\",\"h\":0,\"i\":\"www.capterra.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare Dataiku vs DataRobot 2024 | Capterra\",\"u\":\"https://www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\"},{\"a\":\"What's the difference between DataRobot and Dataiku DSS? Compare DataRobot vs. Dataiku DSS in 2023 by cost, reviews, features, integrations, deployment, target market, support options, trial offers, training options, years in business, region, and more using the chart below.\",\"ae\":null,\"b\":\"/.\\tSlashdot\\tslashdot.org\",\"c\":\"https://slashdot.org/software/comparison/DataRobot-vs-Dataiku-DSS/\",\"d\":\"slashdot.org/software/comparison/DataRobot-vs-Dataiku-DSS/\",\"da\":\"\",\"h\":0,\"i\":\"slashdot.org\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare DataRobot vs. Dataiku DSS in 2023 - Slashdot\",\"u\":\"https://slashdot.org/software/comparison/DataRobot-vs-Dataiku-DSS/\"},{\"a\":\"1 Star 0% Distribution based on 504 ratings Customer Experience Evaluation & Contracting 4.6 Integration & Deployment 4.7 Service & Support 4.8 Product Capabilities 4.8 FREE View and Download Peer Insights About Dataiku\",\"ae\":null,\"c\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\",\"d\":\"www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\",\"da\":\"\",\"h\":0,\"i\":\"www.gartner.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku Reviews, Ratings & Features 2024 | Gartner Peer Insights\",\"u\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\"},{\"a\":\"1329 reviews on 16 vendors. chevron_right. Yard Management. 25 reviews on 28 vendors. chevron_right. Zero Trust Network Access. 733 reviews on 47 vendors. chevron_right. Read the latest Gartner-verified reviews covering over 500+ software categories and find the best enterprise software or services for your organization.\",\"ae\":null,\"c\":\"https://www.gartner.com/reviews/market/dsml-engineering-platforms/compare/dataiku-vs-datarobot\",\"d\":\"www.gartner.com/reviews/market/dsml-engineering-platforms/compare/dataiku-vs-datarobot\",\"da\":\"\",\"h\":0,\"i\":\"www.gartner.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Explore Enterprise Software Categories | Gartner Peer Insights\",\"u\":\"https://www.gartner.com/reviews/market/dsml-engineering-platforms/compare/dataiku-vs-datarobot\"},{\"a\":\"1. Dataiku is a versatile desktop application comprised of a wide range of tools, including automated machine learning, notebooks, and workflow management. It aims to replace pre-existing tools...\",\"ae\":null,\"b\":\"li\\tLinkedIn\\twww.linkedin.com\",\"c\":\"https://www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\",\"d\":\"www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\",\"da\":\"\",\"e\":\"2023-08-11T00:00:00.0000000\",\"h\":0,\"i\":\"www.linkedin.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Managed Machine Learning Platforms: A Comparative Analysis - LinkedIn\",\"u\":\"https://www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\"},{\"a\":\"Dataiku DSS, H2O, and Google Cloud AI are common alternatives for DataRobot. What is DataRobot's best feature? Reviewers rate Automated Machine Learning highest, with a score of 9.3. Who uses DataRobot? The most common users of DataRobot are from Mid-sized Companies (51-1,000 employees).\",\"ae\":null,\"c\":\"https://www.trustradius.com/products/datarobot/reviews?qs=pros-and-cons\",\"d\":\"www.trustradius.com/products/datarobot/reviews?qs=pros-and-cons\",\"da\":\"\",\"h\":0,\"i\":\"www.trustradius.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Pros and Cons of DataRobot 2024 - TrustRadius\",\"u\":\"https://www.trustradius.com/products/datarobot/reviews?qs=pros-and-cons\"},{\"a\":\"Compare Dataiku and DataRobot based on features, pricing, verified reviews, integrations & more. Find out which software is best for your business today. 0. App comparison. Add up to 4 apps below to see how they compare. You can also use the "Compare" buttons while browsing.\",\"ae\":null,\"c\":\"https://www.getapp.com/emerging-technology-software/a/dataiku-dss/compare/datarobot/\",\"d\":\"www.getapp.com/emerging-technology-software/a/dataiku-dss/compare/datarobot/\",\"da\":\"\",\"h\":0,\"i\":\"www.getapp.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku vs DataRobot Comparison | GetApp\",\"u\":\"https://www.getapp.com/emerging-technology-software/a/dataiku-dss/compare/datarobot/\"},{\"a\":\"Dataiku vs DataRobot AI Platform Compare Dataiku and DataRobot AI Platform using real user data focused on features, satisfaction, business value, and the vendor relationship. What is Machine Learning Platforms (ML) Software?\",\"ae\":null,\"c\":\"https://www.softwarereviews.com/categories/machine-learning-platforms/compare/dataiku-vs-datarobot-ai-platform\",\"d\":\"www.softwarereviews.com/categories/machine-learning-platforms/compare/dataiku-vs-datarobot-ai-platform\",\"da\":\"\",\"h\":0,\"i\":\"www.softwarereviews.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku vs DataRobot AI Platform - Machine Learning Platforms\",\"u\":\"https://www.softwarereviews.com/categories/machine-learning-platforms/compare/dataiku-vs-datarobot-ai-platform\"},{\"a\":\"What's the difference between Alteryx, DataRobot, and Dataiku DSS? Compare Alteryx vs. DataRobot vs. Dataiku DSS in 2024 by cost, reviews, features, integrations, deployment, target market, support options, trial offers, training options, years in business, region, and more using the chart below. Alteryx View Product DataRobot View Product\",\"ae\":null,\"b\":\"/.\\tSlashdot\\tslashdot.org\",\"c\":\"https://slashdot.org/software/comparison/Alteryx-vs-DataRobot-vs-Dataiku-DSS/\",\"d\":\"slashdot.org/software/comparison/Alteryx-vs-DataRobot-vs-Dataiku-DSS/\",\"da\":\"\",\"h\":0,\"i\":\"slashdot.org\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare Alteryx vs. DataRobot vs. Dataiku DSS in 2023 - Slashdot\",\"u\":\"https://slashdot.org/software/comparison/Alteryx-vs-DataRobot-vs-Dataiku-DSS/\"},{\"a\":\"What's the difference between DataRobot, Databricks Lakehouse, and Dataiku DSS? Compare DataRobot vs. Databricks Lakehouse vs. Dataiku DSS in 2023 by cost, reviews, features, integrations, deployment, target market, support options, trial offers, training options, years in business, region, and more using the chart below.\",\"ae\":null,\"b\":\"/.\\tSlashdot\\tslashdot.org\",\"c\":\"https://slashdot.org/software/comparison/DataRobot-vs-Databricks-vs-Dataiku-DSS/\",\"d\":\"slashdot.org/software/comparison/DataRobot-vs-Databricks-vs-Dataiku-DSS/\",\"da\":\"\",\"h\":0,\"i\":\"slashdot.org\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare DataRobot vs. Databricks Lakehouse vs. Dataiku DSS - Slashdot\",\"u\":\"https://slashdot.org/software/comparison/DataRobot-vs-Databricks-vs-Dataiku-DSS/\"},{\"a\":\"The platforms we've chosen for our analysis are ClearML, cnvrg.io, Dataiku, Datarobot, Iguazio, Sagemaker, Seldon and Valohai from the managed side, and Flyte, Kubeflow, MLflow and Metaflow from the open-source side. This is by no means an exhaustive list of all the MLOps tools out there. Most of these are tools that describe themselves as ...\",\"ae\":null,\"c\":\"https://valohai.com/mlops-platforms-compared/\",\"d\":\"valohai.com/mlops-platforms-compared/\",\"da\":\"\",\"h\":0,\"i\":\"valohai.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"MLOps Platforms Compared - Valohai\",\"u\":\"https://valohai.com/mlops-platforms-compared/\"},{\"a\":\"Visual Machine Learning and automated features preprocessing: Builtin charts and dashboards: Code notebooks and recipes: Custom web applications and plugins: Collaboration: DEPLOYMENT OPTIONS; ... Dataiku Scores an overall 4.8 out of 5 rating Based on 249 ratings for the DSMLP market, as of March 1, 2022\",\"ae\":null,\"c\":\"https://www.dataiku.com/product/plans-and-features/\",\"d\":\"www.dataiku.com/product/plans-and-features/\",\"da\":\"\",\"h\":0,\"i\":\"www.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Explore Dataiku Plans and Features | Online or Installed\",\"u\":\"https://www.dataiku.com/product/plans-and-features/\"},{\"a\":\"What's the difference between DataRobot, Databricks Lakehouse, Dataiku DSS, and DATAGYM? Compare DataRobot vs. Databricks Lakehouse vs. Dataiku DSS vs. DATAGYM in 2024 by cost, reviews, features, integrations, deployment, target market, support options, trial offers, training options, years in business, region, and more using the chart below.\",\"ae\":null,\"b\":\"/.\\tSlashdot\\tslashdot.org\",\"c\":\"https://slashdot.org/software/comparison/DataRobot-vs-Databricks-vs-Dataiku-DSS-vs-datagym/\",\"d\":\"slashdot.org/software/comparison/DataRobot-vs-Databricks-vs-Dataiku-DSS-vs-datagym/\",\"da\":\"\",\"h\":0,\"i\":\"slashdot.org\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare DataRobot vs. Databricks Lakehouse vs. Dataiku DSS vs. DATAGYM ...\",\"u\":\"https://slashdot.org/software/comparison/DataRobot-vs-Databricks-vs-Dataiku-DSS-vs-datagym/\"},{\"a\":\"Claim Dataiku DSS and update features and information. Compare C3 AI Suite vs. DataRobot vs. Dataiku DSS using this comparison chart. Compare price, features, and reviews of the software side-by-side to make the best choice for your business.\",\"ae\":null,\"b\":\"srcforge\\tSourceForge\\tsourceforge.net\",\"c\":\"https://sourceforge.net/software/compare/C3-AI-Suite-vs-DataRobot-vs-Dataiku-DSS/\",\"d\":\"sourceforge.net/software/compare/C3-AI-Suite-vs-DataRobot-vs-Dataiku-DSS/\",\"da\":\"\",\"h\":0,\"i\":\"sourceforge.net\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"C3 AI Suite vs. DataRobot vs. Dataiku DSS Comparison - SourceForge\",\"u\":\"https://sourceforge.net/software/compare/C3-AI-Suite-vs-DataRobot-vs-Dataiku-DSS/\"},{\"a\":\"Compare DataRobot AI Platform and Dataiku using real user data focused on features, satisfaction, business value, and the vendor relationship. What is Machine Learning Platforms (ML) Software?\",\"ae\":null,\"c\":\"https://www.softwarereviews.com/categories/machine-learning-platforms/compare/datarobot-ai-platform-vs-dataiku\",\"d\":\"www.softwarereviews.com/categories/machine-learning-platforms/compare/datarobot-ai-platform-vs-dataiku\",\"da\":\"\",\"h\":0,\"i\":\"www.softwarereviews.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"DataRobot AI Platform vs Dataiku - Machine Learning Platforms\",\"u\":\"https://www.softwarereviews.com/categories/machine-learning-platforms/compare/datarobot-ai-platform-vs-dataiku\"},{\"a\":\"What's the difference between Amazon SageMaker, DataRobot, and Dataiku DSS? Compare Amazon SageMaker vs. DataRobot vs. Dataiku DSS in 2024 by cost, reviews, features, integrations, deployment, target market, support options, trial offers, training options, years in business, region, and more using the chart below.\",\"ae\":null,\"b\":\"/.\\tSlashdot\\tslashdot.org\",\"c\":\"https://slashdot.org/software/comparison/Amazon-SageMaker-vs-DataRobot-vs-Dataiku-DSS/\",\"d\":\"slashdot.org/software/comparison/Amazon-SageMaker-vs-DataRobot-vs-Dataiku-DSS/\",\"da\":\"\",\"h\":0,\"i\":\"slashdot.org\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare Amazon SageMaker vs. DataRobot vs. Dataiku DSS in 2024 - Slashdot\",\"u\":\"https://slashdot.org/software/comparison/Amazon-SageMaker-vs-DataRobot-vs-Dataiku-DSS/\"},{\"a\":\"Compare Analance vs. DataRobot vs. Dataiku DSS using this comparison chart. Compare price, features, and reviews of the software side-by-side to make the best choice for your business.\",\"ae\":null,\"b\":\"srcforge\\tSourceForge\\tsourceforge.net\",\"c\":\"https://sourceforge.net/software/compare/Analance-vs-DataRobot-vs-Dataiku-DSS/\",\"d\":\"sourceforge.net/software/compare/Analance-vs-DataRobot-vs-Dataiku-DSS/\",\"da\":\"\",\"h\":0,\"i\":\"sourceforge.net\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Analance vs. DataRobot vs. Dataiku DSS Comparison - SourceForge\",\"u\":\"https://sourceforge.net/software/compare/Analance-vs-DataRobot-vs-Dataiku-DSS/\"},{\"n\":\"/d.js?q=Dataiku%20vs%20DataRobot%20features&kl=wt-wt&l=wt-wt&p=&s=23&ex=-1&ct=US&sp=0&vqd=4-334935250614046875026454141242803242982\"}]);DDG.duckbar.load('images');DDG.duckbar.load('news');DDG.duckbar.load('videos');DDG.duckbar.loadModule('related_searches', {\"ads\":[],\"query\":\"Dataiku vs DataRobot features\",\"queryEncoded\":\"Dataiku%20vs%20DataRobot%20features\",\"response_type\":\"places\",\"results\":[{\"display_text\":\"dataiku vs datarobot review\",\"text\":\"dataiku vs datarobot review\",\"web_search_url\":\"?q=dataiku%20vs%20datarobot%20review\"},{\"display_text\":\"dataiku vs alteryx\",\"text\":\"dataiku vs alteryx\",\"web_search_url\":\"?q=dataiku%20vs%20alteryx\"},{\"display_text\":\"gartner dataiku reviews\",\"text\":\"gartner dataiku reviews\",\"web_search_url\":\"?q=gartner%20dataiku%20reviews\"},{\"display_text\":\"alteryx vs dataiku knime\",\"text\":\"alteryx vs dataiku knime\",\"web_search_url\":\"?q=alteryx%20vs%20dataiku%20knime\"},{\"display_text\":\"dataiku vs rapidminer\",\"text\":\"dataiku vs rapidminer\",\"web_search_url\":\"?q=dataiku%20vs%20rapidminer\"},{\"display_text\":\"dataiku vs azure ml\",\"text\":\"dataiku vs azure ml\",\"web_search_url\":\"?q=dataiku%20vs%20azure%20ml\"},{\"display_text\":\"sagemaker vs dataiku\",\"text\":\"sagemaker vs dataiku\",\"web_search_url\":\"?q=sagemaker%20vs%20dataiku\"},{\"display_text\":\"dataiku reviews\",\"text\":\"dataiku reviews\",\"web_search_url\":\"?q=dataiku%20reviews\"}],\"vqd\":{\"Dataiku%20vs%20DataRobot%20features\":\"4-334935250614046875026454141242803242982\"}});if (DDG.pageLayout) DDG.pageLayout.initialize({\"mainline\":{\"items\":[[\"ad\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"related_searches\"]]},\"sidebar\":{\"items\":[[\"wikipedia_fathead\"]]}}, { start: 0 });DDG.deep.emit(\"load:completed\");", "curl-cffi-POST-https://duckduckgo.com-{\"data\": {\"q\": \"Dataiku and DataRobot use cases\"}}": "Dataiku and DataRobot use cases at DuckDuckGo
", - "curl-cffi-GET-https://links.duckduckgo.com/d.js-{\"params\": {\"bing_market\": \"wt-WT\", \"df\": null, \"ex\": \"-1\", \"kl\": \"wt-wt\", \"l\": \"wt-wt\", \"q\": \"Dataiku and DataRobot use cases\", \"s\": \"0\", \"sp\": \"0\", \"vqd\": \"4-60481969350525797892441552954401970387\"}}": "if (DDG.deep && DDG.deep.setUpstream) DDG.deep.setUpstream(\"bingv7aa\");DDG.deep.bn={'ivc':1};if (DDG.pageLayout) DDG.pageLayout.load('a',[{\"a\":\"\\u9ad8\\u7cbe\\u5ea6\\u306a\\u6a5f\\u68b0\\u5b66\\u7fd2\\u30e2\\u30c7\\u30eb\\u3092\\u69cb\\u7bc9\\u3001\\u5b9f\\u88c5\\u3001\\u904b\\u7528\\u3002DataRobot\\u306f\\u793e\\u5185\\u30c7\\u30fc\\u30bf\\u304b\\u3089\\u65b0\\u3057\\u3044\\u4fa1\\u5024\\u3092\\u5275\\u9020\\u3057\\u307e\\u3059. AI\\u3092\\u6d3b\\u7528\\u3057\\u30c7\\u30fc\\u30bf\\u3092\\u5206\\u6790\\u3001\\u5b9f\\u7528\\u7684\\u306a\\u30a4\\u30f3\\u30b5\\u30a4\\u30c8\\u3092\\u660e\\u3089\\u304b\\u306b\\u3002\\u30d3\\u30b8\\u30cd\\u30b9\\u306e\\u8ab2\\u984c\\u3092\\u3088\\u308a\\u65e9\\u304f\\u89e3\\u6c7a\",\"adext\":{\"callout\":{\"t\":\"30-Day Free Trial \\u00b7 Trusted by Fortune 50 \\u00b7 No Vendor Lock-in\",\"tid\":\"6\"},\"filterlinks\":{\"l\":[],\"tid\":\"\"},\"sitelinks\":{\"l\":[{\"snippet\":\"Explore the DataRobot AI Platform Get Started With a 30-Day Trial\",\"targetUrl\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=d0faee2c8c1aae9ac3a012e21d37352a1181970dce9edeba4107839fbfbf097a&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De81Q_rqdj1ZxH5XXGh4PG6pjVUCUzdB7rGpyykWEihNc_sSp5n%2DJ9jIyTjOSnXg0OUazrpKgDJrNvBOdNa5PjBGtyLGt23nrBAabI6opJXrliWQ4o%2DTyxIsqOeCXqzLOOJ3jJb74k6KEx20zilzwKmzSg3nBop2A9JqsasC17VVDPc3_i3EzPbWeRNS4nhxXWJqBKd55GfhuEOg2RZUbmmuAUhWvM%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZnRyaWFsJTJmJTNmdXRtX21lZGl1bSUzZHNlYXJjaCUyNnV0bV9zb3VyY2UlM2RiaW5nJTI2dXRtX2NhbXBhaWduJTNkRnJlZVRyaWFsMjAyM1dXMDgxNkdQU2FkZXh0JTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZDc2YmMwNmFmNTA0NDFjOGVjOGYxNjMwY2FmNGU4ZTVk%26rlid%3D76bc06af50441c8ec8f1630caf4e8e5d&vqd=4-164177780916400746369660096493208330918&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5063.1\",\"text\":\"DataRobot Free Trial\"},{\"snippet\":\"Unlock Your AI Success in 2023 Tips on the Path of Value-Driven AI\",\"targetUrl\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=fdb107a4de6fffdec2bdf43b561b2c63ca700daaef68f0e683547361efbbc2b0&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8%2DT0j3GTQEgr%2DmtHPM1LzNzVUCUyRxvVKYHe6LbNa2mmCfCZh3Ept1NM%2DP%2DM1AAluh_OL3VQw_FWI0A3YxC3pzzqthf3gpxan_Lv7CjKenge%2DwMYUz3bRFoFyHtQBMdgqv6T7gMGfyYwN3UCj6FNYwVVn9UNN0h1dIQanHNB6Ya9gRrPBACknA8qtsf6A2oUG1xhq7AOF98NzGphnfQ_38fySnRU%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZnJlc291cmNlcyUyZmFpc3VjY2VzczIwMjMlMmYlM2Z1dG1fbWVkaXVtJTNkc2VhcmNoJTI2dXRtX3NvdXJjZSUzZGJpbmclMjZ1dG1fY2FtcGFpZ24lM2RDb250ZW50MTBLZXlzdG9BSVN1Y2Nlc3MyMDIzV1cwNTIyR1BTYWRleHQlMjZ1dG1fdGVybSUzZGRhdGFyb2JvdCUyNnV0bV9jb250ZW50JTNkYWRfZXh0JTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZGQzNmQ2MzlkMmFlNTEwMTM3ZTIwMDYzZWQ1ZWY3M2Yz%26rlid%3Dd36d639d2ae510137e20063ed5ef73f3&vqd=4-117927704271333462986714580056949079639&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5065.1\",\"text\":\"10 Keys to AI Success\"},{\"snippet\":\"Our Platform Includes Four Fully Integrated Products. Read More.\",\"targetUrl\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=4f06bd3312172b8e61d65ee2626dea6e26d941c3a16aa546b4e11b79e8bf027f&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8885tVmNmhi65Jmp3f2wYSzVUCUyFey1LCmrSNpGfkWzQnoC7QIbU3ztthJ%2DqKpgCmRfxudhbLK927YN84jvZlV2zTKo9DOULVj5wB8mcGXy_F42SnsrO1jZpY9NnMnzqMYPb5xZTTdgrTO1_w3Bgpd0e0VzO81_O3%2Dfo2z4UiLuVETFVqfACqR6NEwz0yfjzJe6ED9tvi_gPDiUL9iWATrNIrsw%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZnByb2R1Y3QlMmYlM2ZjYW1wYWlnbmlkJTNkNTMwNzA4MDk5JTI2YWRncm91cGlkJTNkMTM1MDIwMjc3NDIxNzY5OCUyNmFkaWQlM2QlMjZtc2Nsa2lkJTNkY2U4NzQ1ZDViODBlMTJmNjQ2N2QyMDc2NDcwNDY2YjI%26rlid%3Dce8745d5b80e12f6467d2076470466b2&vqd=4-169069202740993895017985472268973083525&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5067.1\",\"text\":\"Product Overview\"}],\"tid\":\"7\\t9[8]\\t11[10]\\t13[12]\",\"type\":\"EnhancedSiteLink\"},\"tid\":\"1\"},\"ae\":{\"callout\":[\"30-Day Free Trial \\u00b7 Trusted by Fortune 50 \\u00b7 No Vendor Lock-in\"]},\"c\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=e744d99a8df00b24df71f821ad4d1332080aa03267e50f0e988d284f58d9d2ef&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8tT9soRYLZabP1ukFkRsgNzVUCUzl89Y8xEqpxoqHqIlCI5wWbydNnN_PoAKHAa2Vsio83mXA_ax16t6rJ7XGkBv0Cg7_D1eg2QAuJgPKEam4VWI3rW40B03r1p11ZXN1Gd1847Vj05bAnJnPfgVyC8ZzFQxLxONmOI0Hg182z2bZUVII26BUAlUHaVZ7O_9FEXLJWw%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZmpwJTJmbHAlMmZhaS1mb3ItYnVzaW5lc3MlMmYlM2Z1dG1fbWVkaXVtJTNkc2VhcmNoJTI2dXRtX3NvdXJjZSUzZGJpbmclMjZ1dG1fY2FtcGFpZ24lM2RERU1PMjAyM0FsbFByb2R1Y3RzSlAwNjI2QlBTJTI2dXRtX3Rlcm0lM2RkYXRhcm9ib3QlMjZ1dG1fY29udGVudCUzZERSX2JyYW5kZWRfcnNhJTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZDA2MTIwYzhmMTAxNzEwYTZiNmRiNjkyY2VmMWRiOTY1%26rlid%3D06120c8f101710a6b6db692cef1db965&vqd=4-91027509783546726889708070523412001433&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5058.1\",\"d\":\"datarobot.com\",\"h\":0,\"i\":\"\",\"k\":0,\"m\":0,\"o\":\"\",\"p\":1,\"relevancy\":{\"abstract\":\"%E9%AB%98%E7%B2%BE%E5%BA%A6%E3%81%AA%E6%A9%9F%E6%A2%B0%E5%AD%A6%E7%BF%92%E3%83%A2%E3%83%87%E3%83%AB%E3%82%92%E6%A7%8B%E7%AF%89%E3%80%81%E5%AE%9F%E8%A3%85%E3%80%81%E9%81%8B%E7%94%A8%E3%80%82%3Cb%3EDataRobot%3C%2Fb%3E%E3%81%AF%E7%A4%BE%E5%86%85%E3%83%87%E3%83%BC%E3%82%BF%E3%81%8B%E3%82%89%E6%96%B0%E3%81%97%E3%81%84%E4%BE%A1%E5%80%A4%E3%82%92%E5%89%B5%E9%80%A0%E3%81%97%E3%81%BE%E3%81%99.%20AI%E3%82%92%E6%B4%BB%E7%94%A8%E3%81%97%E3%83%87%E3%83%BC%E3%82%BF%E3%82%92%E5%88%86%E6%9E%90%E3%80%81%E5%AE%9F%E7%94%A8%E7%9A%84%E3%81%AA%E3%82%A4%E3%83%B3%E3%82%B5%E3%82%A4%E3%83%88%E3%82%92%E6%98%8E%E3%82%89%E3%81%8B%E3%81%AB%E3%80%82%E3%83%93%E3%82%B8%E3%83%8D%E3%82%B9%E3%81%AE%E8%AA%B2%E9%A1%8C%E3%82%92%E3%82%88%E3%82%8A%E6%97%A9%E3%81%8F%E8%A7%A3%E6%B1%BA\",\"adx_name\":\"none\",\"is_good_v10\":0,\"organic_ranks\":[5,11,12,13],\"q\":\"Dataiku%20and%20DataRobot%20use%20cases\",\"q_words\":4,\"q_words_fuzzy\":0.25,\"q_words_in_ad\":1,\"root_domain\":\"datarobot.com\",\"start\":\"0\",\"title\":\"%E3%83%93%E3%83%83%E3%82%B0%E3%83%87%E3%83%BC%E3%82%BF%E5%88%86%E6%9E%90%E3%82%92%E9%AB%98%E9%80%9F%E5%8C%96%20%2D%20%E3%83%87%E3%83%BC%E3%82%BF%E3%81%8B%E3%82%89%E6%96%B0%E3%81%97%E3%81%84%E4%BE%A1%E5%80%A4%E3%82%92\"},\"s\":\"bingv7aa\",\"t\":\"\\u30d3\\u30c3\\u30b0\\u30c7\\u30fc\\u30bf\\u5206\\u6790\\u3092\\u9ad8\\u901f\\u5316 - \\u30c7\\u30fc\\u30bf\\u304b\\u3089\\u65b0\\u3057\\u3044\\u4fa1\\u5024\\u3092\",\"tid\":\"1,6,7,9[8],11[10],13[12]\",\"u\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=e744d99a8df00b24df71f821ad4d1332080aa03267e50f0e988d284f58d9d2ef&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8tT9soRYLZabP1ukFkRsgNzVUCUzl89Y8xEqpxoqHqIlCI5wWbydNnN_PoAKHAa2Vsio83mXA_ax16t6rJ7XGkBv0Cg7_D1eg2QAuJgPKEam4VWI3rW40B03r1p11ZXN1Gd1847Vj05bAnJnPfgVyC8ZzFQxLxONmOI0Hg182z2bZUVII26BUAlUHaVZ7O_9FEXLJWw%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZmpwJTJmbHAlMmZhaS1mb3ItYnVzaW5lc3MlMmYlM2Z1dG1fbWVkaXVtJTNkc2VhcmNoJTI2dXRtX3NvdXJjZSUzZGJpbmclMjZ1dG1fY2FtcGFpZ24lM2RERU1PMjAyM0FsbFByb2R1Y3RzSlAwNjI2QlBTJTI2dXRtX3Rlcm0lM2RkYXRhcm9ib3QlMjZ1dG1fY29udGVudCUzZERSX2JyYW5kZWRfcnNhJTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZDA2MTIwYzhmMTAxNzEwYTZiNmRiNjkyY2VmMWRiOTY1%26rlid%3D06120c8f101710a6b6db692cef1db965&vqd=4-91027509783546726889708070523412001433&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5058.1\"}], {\"page_load_url\":\"https://duckduckgo.com/y.js?ifu=%7B3%7Dappid%3D055AAD1BA669BEB8B048128DC89A107C678B527B%26rguid%3D309794dc72f748f6a2b95ce5c34fbcec&iurl=%7B2%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26Type%3DEvent.CPT%26DATA%3D0\",\"visibility_url\":\"https://duckduckgo.com/y.js?ivu=%7B4%7Dtype%3Dmv%26reqver%3D1.0%26rg%3D309794dc72f748f6a2b95ce5c34fbcec\"});DDG.deep.signalSummary = \"\";DDG.inject('DDG.Data.languages.resultLanguages', {\"en\":[\"https://knowledge.dataiku.com/latest/use-cases/index.html\",\"https://community.dataiku.com/t5/Dataiku-Use-Cases-Success/tkb-p/use-cases\",\"https://www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\",\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\",\"https://community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\",\"https://www.datarobot.com/use-cases/\",\"https://academy.dataiku.com/page/use-cases\",\"https://www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\",\"https://www.g2.com/compare/datarobot-vs-dataiku-dss\",\"https://www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\",\"https://londondataconsulting.medium.com/dataiku-what-is-it-how-to-use-it-ultimate-guide-2023-47602c85a48b\",\"https://docs.datarobot.com/en/docs/api/guide/common-case/index.html\",\"https://www.datarobot.com/blog/introducing-the-datarobot-use-case-value-tracker/\",\"https://docs.datarobot.com/en/docs/workbench/wb-usecase/wb-build-usecase.html\",\"https://blog.dataiku.com/topic/use-cases-projects\",\"https://valohai.com/mlops-platforms-compared/\",\"https://www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\",\"https://pages.dataiku.com/experience-a-dataiku-demo\",\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\",\"https://www.dataiku.com/stories/\",\"https://www.dataiku.com/\",\"https://techcrunch.com/2022/12/13/ai-and-analytics-platform-dataiku-raises-200m-at-a-reduced-valuation/\",\"https://www.globenewswire.com/news-release/2022/11/17/2558152/0/en/Ben-Taylor-Joins-Dataiku-as-Chief-AI-Strategist.html\"]});DDG.deep.pageLayoutSummary = \"a1w4v1w19,w1\";DDG.inject('DDG.Data.languages.adLanguages', {});if (DDG.pageLayout) DDG.pageLayout.load('d',[{\"a\":\"Use Cases - Dataiku Knowledge Base Use Cases # These use cases allow you to practice what you've learned by building simplified, but complete use cases in Dataiku. Topics # Data Preparation Use Cases Classification Use Cases Clustering Use Cases Plugin Use Cases\",\"ae\":null,\"c\":\"https://knowledge.dataiku.com/latest/use-cases/index.html\",\"d\":\"knowledge.dataiku.com/latest/use-cases/index.html\",\"da\":\"\",\"h\":0,\"i\":\"knowledge.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Use Cases - Dataiku Knowledge Base\",\"u\":\"https://knowledge.dataiku.com/latest/use-cases/index.html\"},{\"a\":\"Community Dataiku Use Cases & Success Stories \\u26a0\\ufe0f Discover pioneering Dataiku use cases and success stories shared by customers, partners, academics, and nonprofits participating in the Dataiku Frontrunner Awards. Use the following labels to filter submissions by industry:\",\"ae\":null,\"c\":\"https://community.dataiku.com/t5/Dataiku-Use-Cases-Success/tkb-p/use-cases\",\"d\":\"community.dataiku.com/t5/Dataiku-Use-Cases-Success/tkb-p/use-cases\",\"da\":\"\",\"h\":0,\"i\":\"community.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku Use Cases & Success Stories - Dataiku Community\",\"u\":\"https://community.dataiku.com/t5/Dataiku-Use-Cases-Success/tkb-p/use-cases\"},{\"a\":\"Dataiku is a cross-platform desktop application that includes a broad range of tools, such as notebooks (similar to Jupyter Notebook), workflow management (similar to Apache Airflow), and automated machine learning. In general, Dataiku aims to replace many of your existing tools rather than to integrate with them.\",\"ae\":null,\"c\":\"https://www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\",\"d\":\"www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\",\"da\":\"\",\"h\":0,\"i\":\"www.datarevenue.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"ML Platforms: Dataiku vs. Alteryx vs. Sagemaker vs. Datarobot\",\"u\":\"https://www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\"},{\"a\":\"Dataiku has a rating of 4.8 stars with 504 reviews. DataRobot has a rating of 4.6 stars with 508 reviews. See side-by-side comparisons of product capabilities, customer experience, pros and cons, and reviewer demographics to find the best fit for your organization. See more companies in the Data Science and Machine Learning Platforms market. PDF.\",\"ae\":null,\"c\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\",\"d\":\"www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\",\"da\":\"\",\"h\":0,\"i\":\"www.gartner.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku vs DataRobot 2024 | Gartner Peer Insights\",\"u\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\"},{\"a\":\"In my humble opinion DSS is a more a 'toolbox', where as DataRobot is an autoML platform. DataRobot is really good at what it does - if you have non-technical team who want to drop in data and leave everything to autoML then this may be the option for them.\",\"ae\":null,\"c\":\"https://community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\",\"d\":\"community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\",\"da\":\"\",\"h\":0,\"i\":\"community.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Solved: Dataiku vs DataRobot - Dataiku Community\",\"u\":\"https://community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\"},{\"a\":\"Use cases AI Use Cases AI-driven organizations around the world use DataRobot to solve their most pressing business problems. Build with Free Trial Recent Popular Filters Ready to Get Started? See how a value-driven approach to AI can accelerate time to impact. Start Free Trial\",\"ae\":null,\"c\":\"https://www.datarobot.com/use-cases/\",\"d\":\"www.datarobot.com/use-cases/\",\"da\":\"\",\"h\":0,\"i\":\"www.datarobot.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Machine Learning Use Cases | DataRobot AI Platform\",\"u\":\"https://www.datarobot.com/use-cases/\"},{\"a\":\"With Dataiku's AI Prepare assistant, you can work smarter, not harder. Simply describe the transformation you want to apply in natural language and the AI assistant automatically generates the necessary data preparation steps. The ability to modify both your prompt and the resulting steps means you can prepare data faster than ever, yet still ...\",\"ae\":null,\"c\":\"https://academy.dataiku.com/page/use-cases\",\"d\":\"academy.dataiku.com/page/use-cases\",\"da\":\"\",\"h\":0,\"i\":\"academy.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Use Cases - Dataiku\",\"u\":\"https://academy.dataiku.com/page/use-cases\"},{\"a\":\"84 Reviews and Ratings Path to AI Success Compare Dataiku DSS vs DataRobot. 103 verified user reviews and ratings of features, pros, cons, pricing, support and more.\",\"ae\":null,\"c\":\"https://www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\",\"d\":\"www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\",\"da\":\"\",\"h\":0,\"i\":\"www.trustradius.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku DSS vs DataRobot | TrustRadius\",\"u\":\"https://www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\"},{\"a\":\"side-by-side comparison of DataRobot vs. Dataiku DSS. based on preference data from user reviews. DataRobot rates 4.4/5 stars with 26 reviews. By contrast, Dataiku DSS rates 4.3/5 stars with 36 reviews. Each product's score is calculated with real-time data from verified user reviews, to help you make the best choice between these two options ...\",\"ae\":null,\"c\":\"https://www.g2.com/compare/datarobot-vs-dataiku-dss\",\"d\":\"www.g2.com/compare/datarobot-vs-dataiku-dss\",\"da\":\"\",\"h\":0,\"i\":\"www.g2.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare DataRobot vs. Dataiku DSS | G2\",\"u\":\"https://www.g2.com/compare/datarobot-vs-dataiku-dss\"},{\"a\":\"Use case: Choose Datarobot if you have data stored in spreadsheets and are seeking a platform that is the simplest, albeit one with limited flexibility, ... Dataiku vs. Datarobot .\",\"ae\":null,\"b\":\"li\\tLinkedIn\\twww.linkedin.com\",\"c\":\"https://www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\",\"d\":\"www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\",\"da\":\"\",\"e\":\"2023-08-11T00:00:00.0000000\",\"h\":0,\"i\":\"www.linkedin.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Managed Machine Learning Platforms: A Comparative Analysis - LinkedIn\",\"u\":\"https://www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\"},{\"a\":\"Jan 11, 2023 Dataiku is an artificial intelligence platform created in France in 2013. It has since become one of the world's benchmarks for data science and machine learning studios. What is...\",\"ae\":null,\"c\":\"https://londondataconsulting.medium.com/dataiku-what-is-it-how-to-use-it-ultimate-guide-2023-47602c85a48b\",\"d\":\"londondataconsulting.medium.com/dataiku-what-is-it-how-to-use-it-ultimate-guide-2023-47602c85a48b\",\"da\":\"translations\",\"e\":\"2023-01-11T00:00:00.0000000\",\"h\":0,\"i\":\"londondataconsulting.medium.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku: What is it? How to use it? Ultimate Guide 2023\",\"u\":\"https://londondataconsulting.medium.com/dataiku-what-is-it-how-to-use-it-ultimate-guide-2023-47602c85a48b\"},{\"a\":\"Use cases for version 2.x. Notebooks for uses cases that use methods for 2.x versions of DataRobot's Python client. Measure price elasticity of demand. A use case to identify relationships between price and demand, maximize revenue by properly pricing products, and monitor price elasticities for changes in price and demand. Insurance claim triage.\",\"ae\":null,\"c\":\"https://docs.datarobot.com/en/docs/api/guide/common-case/index.html\",\"d\":\"docs.datarobot.com/en/docs/api/guide/common-case/index.html\",\"da\":\"\",\"h\":0,\"i\":\"docs.datarobot.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Common use cases: DataRobot docs - DataRobot AI Platform\",\"u\":\"https://docs.datarobot.com/en/docs/api/guide/common-case/index.html\"},{\"a\":\"With the Use Case Value Tracker, you can manage the project lifecycle and understand the value associated with each step. It also enables you to associate and organize all your DataRobot artifacts (e.g., datasets, models, deployments, applications, etc.) around a given use case for a holistic view. In addition to the project management aspects ...\",\"ae\":null,\"c\":\"https://www.datarobot.com/blog/introducing-the-datarobot-use-case-value-tracker/\",\"d\":\"www.datarobot.com/blog/introducing-the-datarobot-use-case-value-tracker/\",\"da\":\"\",\"h\":0,\"i\":\"www.datarobot.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Introducing the DataRobot Use Case Value Tracker\",\"u\":\"https://www.datarobot.com/blog/introducing-the-datarobot-use-case-value-tracker/\"},{\"a\":\"Use Cases are folder-like containers inside of DataRobot Workbench that allow you to group everything related to solving a specific business problem\\u2014datasets, models, experiments, No-Code AI Apps, and notebooks\\u2014inside of a single, manageable entity. You can share whole Use Cases as well as the individual assets they contain.\",\"ae\":null,\"c\":\"https://docs.datarobot.com/en/docs/workbench/wb-usecase/wb-build-usecase.html\",\"d\":\"docs.datarobot.com/en/docs/workbench/wb-usecase/wb-build-usecase.html\",\"da\":\"\",\"e\":\"2023-09-15T00:00:00.0000000\",\"h\":0,\"i\":\"docs.datarobot.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Use Cases: DataRobot docs\",\"u\":\"https://docs.datarobot.com/en/docs/workbench/wb-usecase/wb-build-usecase.html\"},{\"a\":\"January 2, 2024 Use Cases & Projects, Featured Sophie Dionnet Leveraging AI to Cut Costs December 29, 2023 Data Basics, Featured\",\"ae\":null,\"c\":\"https://blog.dataiku.com/topic/use-cases-projects\",\"d\":\"blog.dataiku.com/topic/use-cases-projects\",\"da\":\"\",\"h\":0,\"i\":\"blog.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Blog - Dataiku | Use Cases & Projects\",\"u\":\"https://blog.dataiku.com/topic/use-cases-projects\"},{\"a\":\"The platforms we've chosen for our analysis are ClearML, cnvrg.io, Dataiku, Datarobot, Iguazio, Sagemaker, Seldon and Valohai from the managed side, and Flyte, Kubeflow, MLflow and Metaflow from the open-source side. This is by no means an exhaustive list of all the MLOps tools out there.\",\"ae\":null,\"c\":\"https://valohai.com/mlops-platforms-compared/\",\"d\":\"valohai.com/mlops-platforms-compared/\",\"da\":\"\",\"h\":0,\"i\":\"valohai.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"MLOps Platforms Compared - Valohai\",\"u\":\"https://valohai.com/mlops-platforms-compared/\"},{\"a\":\"DataRobot. DSS is for all companies, whatever their expertise, industry or size, that want to create their own data-driven strategic advantages by transforming their raw data into business impacting predictions. Cloud based machine learning platform which helps enterprises scale data science capabilities through deploying machine learning ...\",\"ae\":null,\"c\":\"https://www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\",\"d\":\"www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\",\"da\":\"translations\",\"h\":0,\"i\":\"www.capterra.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare Dataiku vs DataRobot 2024 | Capterra\",\"u\":\"https://www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\"},{\"a\":\"For Every Industry & Use Case. Organizations that use Dataiku elevate their people (whether technical and working in code or on the business side and low- or no-code) to extraordinary, arming them with the ability to make better day-to-day decisions with data across: Banking & Insurance. Pharmaceuticals. Manufacturing. Telecommunications.\",\"ae\":null,\"c\":\"https://pages.dataiku.com/experience-a-dataiku-demo\",\"d\":\"pages.dataiku.com/experience-a-dataiku-demo\",\"da\":\"\",\"h\":0,\"i\":\"pages.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Check Out This Dataiku Demo\",\"u\":\"https://pages.dataiku.com/experience-a-dataiku-demo\"},{\"a\":\"4 Star 24% 3 Star 1% 2 Star 0% 1 Star 0% Distribution based on 504 ratings Customer Experience Evaluation & Contracting 4.6 Integration & Deployment 4.7 Service & Support 4.8 Product Capabilities 4.8 FREE View and Download Peer Insights About Dataiku\",\"ae\":null,\"c\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\",\"d\":\"www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\",\"da\":\"\",\"h\":0,\"i\":\"www.gartner.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku Reviews, Ratings & Features 2024 | Gartner Peer Insights\",\"u\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\"},{\"a\":\"Read The Full Case Study U.S. Venture + Dataiku: Upskilling Analysts to Save Thousands of Hours The Data and Analytics team at U.S. Venture was built to usher the company into the future of data science and AI.\",\"ae\":null,\"c\":\"https://www.dataiku.com/stories/\",\"d\":\"www.dataiku.com/stories/\",\"da\":\"\",\"h\":0,\"i\":\"www.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Stories | Dataiku\",\"u\":\"https://www.dataiku.com/stories/\"},{\"a\":\"MLOps Deploy, monitor, and maintain machine learning models, all in a single platform. Explore the Capability Collaboration With Dataiku, teams can move beyond the lab and build real and safe Generative AI applications at enterprise scale. Explore the Capability Governance\",\"ae\":null,\"c\":\"https://www.dataiku.com/\",\"d\":\"www.dataiku.com\",\"da\":\"\",\"h\":0,\"i\":\"www.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku | Everyday AI, Extraordinary People\",\"u\":\"https://www.dataiku.com/\"},{\"a\":\"The company today announced that it raised $200 million in a Series F round led by Wellington Management at a $3.7 billion valuation, down from the $4.6 billion that Dataiku received in August ...\",\"ae\":null,\"b\":\"tc\\tTechcrunch\\ttechcrunch.com\",\"c\":\"https://techcrunch.com/2022/12/13/ai-and-analytics-platform-dataiku-raises-200m-at-a-reduced-valuation/\",\"d\":\"techcrunch.com/2022/12/13/ai-and-analytics-platform-dataiku-raises-200m-at-a-reduced-valuation/\",\"da\":\"translations\",\"e\":\"2022-12-13T17:10:00.0000000\",\"h\":0,\"i\":\"techcrunch.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"AI and analytics platform Dataiku raises $200M at a reduced valuation\",\"u\":\"https://techcrunch.com/2022/12/13/ai-and-analytics-platform-dataiku-raises-200m-at-a-reduced-valuation/\"},{\"a\":\"Today, more than 500 companies worldwide use Dataiku to integrate and streamline their use of data, analytics, and AI, driving diverse use cases from fraud detection and customer churn prevention ...\",\"ae\":null,\"c\":\"https://www.globenewswire.com/news-release/2022/11/17/2558152/0/en/Ben-Taylor-Joins-Dataiku-as-Chief-AI-Strategist.html\",\"d\":\"www.globenewswire.com/news-release/2022/11/17/2558152/0/en/Ben-Taylor-Joins-Dataiku-as-Chief-AI-Strategist.html\",\"da\":\"translations\",\"e\":\"2022-11-17T13:00:00.0000000\",\"h\":0,\"i\":\"www.globenewswire.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Ben Taylor Joins Dataiku as Chief AI Strategist - GlobeNewswire\",\"u\":\"https://www.globenewswire.com/news-release/2022/11/17/2558152/0/en/Ben-Taylor-Joins-Dataiku-as-Chief-AI-Strategist.html\"},{\"n\":\"/d.js?q=Dataiku%20and%20DataRobot%20use%20cases&kl=wt-wt&l=wt-wt&p=&s=23&ex=-1&ct=US&sp=0&vqd=4-60481969350525797892441552954401970387\"}]);DDG.duckbar.load('images');DDG.duckbar.load('news');DDG.duckbar.load('videos', {\"ads\":[],\"query\":\"Dataiku and DataRobot use cases\",\"queryEncoded\":\"Dataiku%20and%20DataRobot%20use%20cases\",\"response_type\":\"places\",\"results\":[{\"content\":\"https://www.youtube.com/watch?v=ryZRRIjQ5Z8\",\"description\":\"If you're a code-first data practitioner, Dataiku helps you efficiently build high quality data pipelines and models in a number of ways. CHECK OUT DATAIKU: https://bit.ly/36XBlpK EGG ON AIR: https://bit.ly/37GhXMY BRIGHTTALK WEBINARS: https://bit.ly/33TIRjn DATA SCIENCE PIONEERS DOCUMENTARY: https://bit.ly/36V3rBF PARTNER ECOSYSTEM: https ...\",\"duration\":\"10:43\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/ryZRRIjQ5Z8?autoplay=1\",\"image_token\":\"8a4abca8613c6680a108591849e5d7b13b86111004ae004898a7f059b64c8355\",\"images\":{\"large\":\"https://tse4.mm.bing.net/th?id=OVP.WoendyuZJ9qxql-n6jit5AEsDh&pid=Api\",\"medium\":\"https://tse4.mm.bing.net/th?id=OVP.WoendyuZJ9qxql-n6jit5AEsDh&pid=Api\",\"motion\":\"https://tse4.mm.bing.net/th?id=OM1.cmvppfhHVUeE4Q_1684256861&pid=Api\",\"small\":\"https://tse4.mm.bing.net/th?id=OVP.WoendyuZJ9qxql-n6jit5AEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2021-06-08T21:15:02.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":12391},\"title\":\"Dataiku Demo for Data Scientists and Coders\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=jKL_I0SCl_E\",\"description\":\"This video showcases the Clinical Trial Explorer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest of our ...\",\"duration\":\"1:50\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/jKL_I0SCl_E?autoplay=1\",\"image_token\":\"7b19602fe6d9b761aa3cc138448cc632ddbed31da3abf2687f36705f5945973d\",\"images\":{\"large\":\"https://tse3.mm.bing.net/th?id=OVP.wfgHp53woiVosZ37-1HtnwEsDh&pid=Api\",\"medium\":\"https://tse3.mm.bing.net/th?id=OVP.wfgHp53woiVosZ37-1HtnwEsDh&pid=Api\",\"motion\":\"https://tse3.mm.bing.net/th?id=OM2.ZX_yq0xmyCGZBg_1696372683&pid=Api\",\"small\":\"https://tse3.mm.bing.net/th?id=OVP.wfgHp53woiVosZ37-1HtnwEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:19:00.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":150},\"title\":\"Clinical Trial Explorer\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=lASesA4gNFI\",\"description\":\"This video showcases the CO2 Forecast Analyzer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest of our ...\",\"duration\":\"1:50\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/lASesA4gNFI?autoplay=1\",\"image_token\":\"a09328adca01a788783d759561c2f9c9d4d214e5a26f1462d2b6b69f21a2d478\",\"images\":{\"large\":\"https://tse2.mm.bing.net/th?id=OVP.ZhQI7z-IMlcVnyeMJuGlAQEsDh&pid=Api\",\"medium\":\"https://tse2.mm.bing.net/th?id=OVP.ZhQI7z-IMlcVnyeMJuGlAQEsDh&pid=Api\",\"motion\":\"\",\"small\":\"https://tse2.mm.bing.net/th?id=OVP.ZhQI7z-IMlcVnyeMJuGlAQEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:16:20.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":49},\"title\":\"CO2 Forecast Analyzer\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=RecpD6Vtzj4\",\"description\":\"This video showcases the LLM-Enhanced ESG Document Intelligence use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the ...\",\"duration\":\"1:20\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/RecpD6Vtzj4?autoplay=1\",\"image_token\":\"6f797accb167e2e6ff7265e35116cdeb9f1c641b1df47932d9597b61b0108614\",\"images\":{\"large\":\"https://tse2.mm.bing.net/th?id=OVP.bm4gAiJOOmKV7uup6LU9pgEsDh&pid=Api\",\"medium\":\"https://tse2.mm.bing.net/th?id=OVP.bm4gAiJOOmKV7uup6LU9pgEsDh&pid=Api\",\"motion\":\"https://tse2.mm.bing.net/th?id=OM1.M8WXwCQ79nrqEA_1691502936&pid=Api\",\"small\":\"https://tse2.mm.bing.net/th?id=OVP.bm4gAiJOOmKV7uup6LU9pgEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:30:00.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":382},\"title\":\"LLM-Enhanced ESG Document Intelligence\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=zLW0TkJoHLw\",\"description\":\"This video showcases the Demand Forecast Analyzer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest of our ...\",\"duration\":\"1:42\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/zLW0TkJoHLw?autoplay=1\",\"image_token\":\"524eb9572bf9342b859509285d39ec4661fc572cb1452307acc5341b56bab921\",\"images\":{\"large\":\"https://tse1.mm.bing.net/th?id=OVP.yIekQ2cMUesOPJTfsYIZzQHgFo&pid=Api\",\"medium\":\"https://tse1.mm.bing.net/th?id=OVP.yIekQ2cMUesOPJTfsYIZzQHgFo&pid=Api\",\"motion\":\"\",\"small\":\"https://tse1.mm.bing.net/th?id=OVP.yIekQ2cMUesOPJTfsYIZzQHgFo&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:15:02.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":4},\"title\":\"LLM-Enhanced Demand Forecast\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=L-Yys0fzuVY\",\"description\":\"This video showcases the Production Quality Data Explorer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest ...\",\"duration\":\"1:47\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/L-Yys0fzuVY?autoplay=1\",\"image_token\":\"82924713be1dba83d67124fcaa6cc6afd163900a0c40f25fcf6c144ed0e36536\",\"images\":{\"large\":\"https://tse4.mm.bing.net/th?id=OVP.teiC0mX9nKCbH8qeo52udwEsDh&pid=Api\",\"medium\":\"https://tse4.mm.bing.net/th?id=OVP.teiC0mX9nKCbH8qeo52udwEsDh&pid=Api\",\"motion\":\"https://tse4.mm.bing.net/th?id=OM2.IwRaoLRWcQXzag_1691419996&pid=Api\",\"small\":\"https://tse4.mm.bing.net/th?id=OVP.teiC0mX9nKCbH8qeo52udwEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:28:29.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":175},\"title\":\"Production Quality Data Explorer\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=FluiuHuaU8A\",\"description\":\"In this breakout session of Dataiku's Product Days 2021, you will see a demo of Dataiku's Data Science Studio, the centralized, collaborative, and end-to-end platform for data science in the enterprise. CHECK OUT DATAIKU: https://bit.ly/36XBlpK EGG ON AIR: https://bit.ly/37GhXMY BRIGHTTALK WEBINARS: https://bit.ly/33TIRjn DATA SCIENCE PIONEERS ...\",\"duration\":\"13:50\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/FluiuHuaU8A?autoplay=1\",\"image_token\":\"2943fa8c1580f2936fc11667d670c0b827b94ff3d16b897f8b5ef2e2426487b3\",\"images\":{\"large\":\"https://tse2.mm.bing.net/th?id=OVP.RIM-ftwDZjYP58RimJfgwwEsDh&pid=Api\",\"medium\":\"https://tse2.mm.bing.net/th?id=OVP.RIM-ftwDZjYP58RimJfgwwEsDh&pid=Api\",\"motion\":\"https://tse2.mm.bing.net/th?id=OM1.MIQ7BoQz1MVkNw_1662248868&pid=Api\",\"small\":\"https://tse2.mm.bing.net/th?id=OVP.RIM-ftwDZjYP58RimJfgwwEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2021-07-08T15:56:22.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":3844},\"title\":\"Introduction to Dataiku Data Science | Product Days 2021\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=6TEU5JboP7k\",\"description\":\"This video showcases the LLM-Enhanced Next Best Offer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest of ...\",\"duration\":\"1:47\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/6TEU5JboP7k?autoplay=1\",\"image_token\":\"a3bc327ff2f099462935a8979bb599655c7a88a44e25a64bfea7e5973f773158\",\"images\":{\"large\":\"https://tse2.mm.bing.net/th?id=OVP.Q6SP0MmL89M_TnLvNPG4oQEsDh&pid=Api\",\"medium\":\"https://tse2.mm.bing.net/th?id=OVP.Q6SP0MmL89M_TnLvNPG4oQEsDh&pid=Api\",\"motion\":\"https://tse2.mm.bing.net/th?id=OM2.wXNo1CUgYV4Flg_1694065487&pid=Api\",\"small\":\"https://tse2.mm.bing.net/th?id=OVP.Q6SP0MmL89M_TnLvNPG4oQEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:34:32.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":462},\"title\":\"LLM-Enhanced Next Best Offer\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=UVbrpX8Zkn8\",\"description\":\"Move beyond the lab and build real and safe Generative AI applications at enterprise scale. Dataiku brings enterprise-grade development tools, pre-built use cases, and AI-powered assistants throughout the platform. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore Generative AI use cases | https ...\",\"duration\":\"2:07\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/UVbrpX8Zkn8?autoplay=1\",\"image_token\":\"3968d2d01ff722efa156290344ab0b37164e57d7efa50905c346ea1cc1a5d369\",\"images\":{\"large\":\"https://tse1.mm.bing.net/th?id=OVP.F-xH3-wKfTjM3YMshnjWwwEsDh&pid=Api\",\"medium\":\"https://tse1.mm.bing.net/th?id=OVP.F-xH3-wKfTjM3YMshnjWwwEsDh&pid=Api\",\"motion\":\"https://tse1.mm.bing.net/th?id=OM1.z-FRwxQ_NByK_A_1689135755&pid=Api\",\"small\":\"https://tse1.mm.bing.net/th?id=OVP.F-xH3-wKfTjM3YMshnjWwwEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:25:16.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":1100},\"title\":\"Dataiku for Generative AI: Real Applications, Real Safety\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=-amc9iVauuE\",\"description\":\"Dataiku is the leading platform for Everyday AI, systemizing the use of data for exceptional business results. In today's video we will take a tour of Dataiku's end to end capabilities by exploring a real life use case around environmental impact. Let's take a look at how a data science team with different skills can work together to turn ...\",\"duration\":\"12:35\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/-amc9iVauuE?autoplay=1\",\"image_token\":\"2a05a65ad8a2727aa5c48b8daa7f9ec363a24d4336a3509016d4b200c9d003cd\",\"images\":{\"large\":\"https://tse1.mm.bing.net/th?id=OVP.Az9RhdSVwpXe56mGcs6FqQEsDh&pid=Api\",\"medium\":\"https://tse1.mm.bing.net/th?id=OVP.Az9RhdSVwpXe56mGcs6FqQEsDh&pid=Api\",\"motion\":\"https://tse1.mm.bing.net/th?id=OM1.Q2OhN9DzfowU6A_1685345657&pid=Api\",\"small\":\"https://tse1.mm.bing.net/th?id=OVP.Az9RhdSVwpXe56mGcs6FqQEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-01-09T21:12:27.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":9768},\"title\":\"End to End Demo 2023\",\"uploader\":\"Dataiku\"}],\"vqd\":{\"Dataiku%20and%20DataRobot%20use%20cases\":\"4-60481969350525797892441552954401970387\"}});DDG.duckbar.loadModule('related_searches');if (DDG.pageLayout) DDG.pageLayout.initialize({\"mainline\":{\"items\":[[\"ad\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"videos\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"]]},\"sidebar\":{\"items\":[[\"organic\"]]}}, { start: 0 });DDG.deep.emit(\"load:completed\");" + "curl-cffi-GET-https://links.duckduckgo.com/d.js-{\"params\": {\"bing_market\": \"wt-WT\", \"df\": null, \"ex\": \"-1\", \"kl\": \"wt-wt\", \"l\": \"wt-wt\", \"q\": \"Dataiku and DataRobot use cases\", \"s\": \"0\", \"sp\": \"0\", \"vqd\": \"4-60481969350525797892441552954401970387\"}}": "if (DDG.deep && DDG.deep.setUpstream) DDG.deep.setUpstream(\"bingv7aa\");DDG.deep.bn={'ivc':1};if (DDG.pageLayout) DDG.pageLayout.load('a',[{\"a\":\"\\u9ad8\\u7cbe\\u5ea6\\u306a\\u6a5f\\u68b0\\u5b66\\u7fd2\\u30e2\\u30c7\\u30eb\\u3092\\u69cb\\u7bc9\\u3001\\u5b9f\\u88c5\\u3001\\u904b\\u7528\\u3002DataRobot\\u306f\\u793e\\u5185\\u30c7\\u30fc\\u30bf\\u304b\\u3089\\u65b0\\u3057\\u3044\\u4fa1\\u5024\\u3092\\u5275\\u9020\\u3057\\u307e\\u3059. AI\\u3092\\u6d3b\\u7528\\u3057\\u30c7\\u30fc\\u30bf\\u3092\\u5206\\u6790\\u3001\\u5b9f\\u7528\\u7684\\u306a\\u30a4\\u30f3\\u30b5\\u30a4\\u30c8\\u3092\\u660e\\u3089\\u304b\\u306b\\u3002\\u30d3\\u30b8\\u30cd\\u30b9\\u306e\\u8ab2\\u984c\\u3092\\u3088\\u308a\\u65e9\\u304f\\u89e3\\u6c7a\",\"adext\":{\"callout\":{\"t\":\"30-Day Free Trial \\u00b7 Trusted by Fortune 50 \\u00b7 No Vendor Lock-in\",\"tid\":\"6\"},\"filterlinks\":{\"l\":[],\"tid\":\"\"},\"sitelinks\":{\"l\":[{\"snippet\":\"Explore the DataRobot AI Platform Get Started With a 30-Day Trial\",\"targetUrl\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=d0faee2c8c1aae9ac3a012e21d37352a1181970dce9edeba4107839fbfbf097a&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De81Q_rqdj1ZxH5XXGh4PG6pjVUCUzdB7rGpyykWEihNc_sSp5n%2DJ9jIyTjOSnXg0OUazrpKgDJrNvBOdNa5PjBGtyLGt23nrBAabI6opJXrliWQ4o%2DTyxIsqOeCXqzLOOJ3jJb74k6KEx20zilzwKmzSg3nBop2A9JqsasC17VVDPc3_i3EzPbWeRNS4nhxXWJqBKd55GfhuEOg2RZUbmmuAUhWvM%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZnRyaWFsJTJmJTNmdXRtX21lZGl1bSUzZHNlYXJjaCUyNnV0bV9zb3VyY2UlM2RiaW5nJTI2dXRtX2NhbXBhaWduJTNkRnJlZVRyaWFsMjAyM1dXMDgxNkdQU2FkZXh0JTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZDc2YmMwNmFmNTA0NDFjOGVjOGYxNjMwY2FmNGU4ZTVk%26rlid%3D76bc06af50441c8ec8f1630caf4e8e5d&vqd=4-164177780916400746369660096493208330918&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5063.1\",\"text\":\"DataRobot Free Trial\"},{\"snippet\":\"Unlock Your AI Success in 2023 Tips on the Path of Value-Driven AI\",\"targetUrl\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=fdb107a4de6fffdec2bdf43b561b2c63ca700daaef68f0e683547361efbbc2b0&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8%2DT0j3GTQEgr%2DmtHPM1LzNzVUCUyRxvVKYHe6LbNa2mmCfCZh3Ept1NM%2DP%2DM1AAluh_OL3VQw_FWI0A3YxC3pzzqthf3gpxan_Lv7CjKenge%2DwMYUz3bRFoFyHtQBMdgqv6T7gMGfyYwN3UCj6FNYwVVn9UNN0h1dIQanHNB6Ya9gRrPBACknA8qtsf6A2oUG1xhq7AOF98NzGphnfQ_38fySnRU%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZnJlc291cmNlcyUyZmFpc3VjY2VzczIwMjMlMmYlM2Z1dG1fbWVkaXVtJTNkc2VhcmNoJTI2dXRtX3NvdXJjZSUzZGJpbmclMjZ1dG1fY2FtcGFpZ24lM2RDb250ZW50MTBLZXlzdG9BSVN1Y2Nlc3MyMDIzV1cwNTIyR1BTYWRleHQlMjZ1dG1fdGVybSUzZGRhdGFyb2JvdCUyNnV0bV9jb250ZW50JTNkYWRfZXh0JTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZGQzNmQ2MzlkMmFlNTEwMTM3ZTIwMDYzZWQ1ZWY3M2Yz%26rlid%3Dd36d639d2ae510137e20063ed5ef73f3&vqd=4-117927704271333462986714580056949079639&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5065.1\",\"text\":\"10 Keys to AI Success\"},{\"snippet\":\"Our Platform Includes Four Fully Integrated Products. Read More.\",\"targetUrl\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=4f06bd3312172b8e61d65ee2626dea6e26d941c3a16aa546b4e11b79e8bf027f&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8885tVmNmhi65Jmp3f2wYSzVUCUyFey1LCmrSNpGfkWzQnoC7QIbU3ztthJ%2DqKpgCmRfxudhbLK927YN84jvZlV2zTKo9DOULVj5wB8mcGXy_F42SnsrO1jZpY9NnMnzqMYPb5xZTTdgrTO1_w3Bgpd0e0VzO81_O3%2Dfo2z4UiLuVETFVqfACqR6NEwz0yfjzJe6ED9tvi_gPDiUL9iWATrNIrsw%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZnByb2R1Y3QlMmYlM2ZjYW1wYWlnbmlkJTNkNTMwNzA4MDk5JTI2YWRncm91cGlkJTNkMTM1MDIwMjc3NDIxNzY5OCUyNmFkaWQlM2QlMjZtc2Nsa2lkJTNkY2U4NzQ1ZDViODBlMTJmNjQ2N2QyMDc2NDcwNDY2YjI%26rlid%3Dce8745d5b80e12f6467d2076470466b2&vqd=4-169069202740993895017985472268973083525&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5067.1\",\"text\":\"Product Overview\"}],\"tid\":\"7\\t9[8]\\t11[10]\\t13[12]\",\"type\":\"EnhancedSiteLink\"},\"tid\":\"1\"},\"ae\":{\"callout\":[\"30-Day Free Trial \\u00b7 Trusted by Fortune 50 \\u00b7 No Vendor Lock-in\"]},\"c\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=e744d99a8df00b24df71f821ad4d1332080aa03267e50f0e988d284f58d9d2ef&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8tT9soRYLZabP1ukFkRsgNzVUCUzl89Y8xEqpxoqHqIlCI5wWbydNnN_PoAKHAa2Vsio83mXA_ax16t6rJ7XGkBv0Cg7_D1eg2QAuJgPKEam4VWI3rW40B03r1p11ZXN1Gd1847Vj05bAnJnPfgVyC8ZzFQxLxONmOI0Hg182z2bZUVII26BUAlUHaVZ7O_9FEXLJWw%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZmpwJTJmbHAlMmZhaS1mb3ItYnVzaW5lc3MlMmYlM2Z1dG1fbWVkaXVtJTNkc2VhcmNoJTI2dXRtX3NvdXJjZSUzZGJpbmclMjZ1dG1fY2FtcGFpZ24lM2RERU1PMjAyM0FsbFByb2R1Y3RzSlAwNjI2QlBTJTI2dXRtX3Rlcm0lM2RkYXRhcm9ib3QlMjZ1dG1fY29udGVudCUzZERSX2JyYW5kZWRfcnNhJTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZDA2MTIwYzhmMTAxNzEwYTZiNmRiNjkyY2VmMWRiOTY1%26rlid%3D06120c8f101710a6b6db692cef1db965&vqd=4-91027509783546726889708070523412001433&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5058.1\",\"d\":\"datarobot.com\",\"h\":0,\"i\":\"\",\"k\":0,\"m\":0,\"o\":\"\",\"p\":1,\"relevancy\":{\"abstract\":\"%E9%AB%98%E7%B2%BE%E5%BA%A6%E3%81%AA%E6%A9%9F%E6%A2%B0%E5%AD%A6%E7%BF%92%E3%83%A2%E3%83%87%E3%83%AB%E3%82%92%E6%A7%8B%E7%AF%89%E3%80%81%E5%AE%9F%E8%A3%85%E3%80%81%E9%81%8B%E7%94%A8%E3%80%82%3Cb%3EDataRobot%3C%2Fb%3E%E3%81%AF%E7%A4%BE%E5%86%85%E3%83%87%E3%83%BC%E3%82%BF%E3%81%8B%E3%82%89%E6%96%B0%E3%81%97%E3%81%84%E4%BE%A1%E5%80%A4%E3%82%92%E5%89%B5%E9%80%A0%E3%81%97%E3%81%BE%E3%81%99.%20AI%E3%82%92%E6%B4%BB%E7%94%A8%E3%81%97%E3%83%87%E3%83%BC%E3%82%BF%E3%82%92%E5%88%86%E6%9E%90%E3%80%81%E5%AE%9F%E7%94%A8%E7%9A%84%E3%81%AA%E3%82%A4%E3%83%B3%E3%82%B5%E3%82%A4%E3%83%88%E3%82%92%E6%98%8E%E3%82%89%E3%81%8B%E3%81%AB%E3%80%82%E3%83%93%E3%82%B8%E3%83%8D%E3%82%B9%E3%81%AE%E8%AA%B2%E9%A1%8C%E3%82%92%E3%82%88%E3%82%8A%E6%97%A9%E3%81%8F%E8%A7%A3%E6%B1%BA\",\"adx_name\":\"none\",\"is_good_v10\":0,\"organic_ranks\":[5,11,12,13],\"q\":\"Dataiku%20and%20DataRobot%20use%20cases\",\"q_words\":4,\"q_words_fuzzy\":0.25,\"q_words_in_ad\":1,\"root_domain\":\"datarobot.com\",\"start\":\"0\",\"title\":\"%E3%83%93%E3%83%83%E3%82%B0%E3%83%87%E3%83%BC%E3%82%BF%E5%88%86%E6%9E%90%E3%82%92%E9%AB%98%E9%80%9F%E5%8C%96%20%2D%20%E3%83%87%E3%83%BC%E3%82%BF%E3%81%8B%E3%82%89%E6%96%B0%E3%81%97%E3%81%84%E4%BE%A1%E5%80%A4%E3%82%92\"},\"s\":\"bingv7aa\",\"t\":\"\\u30d3\\u30c3\\u30b0\\u30c7\\u30fc\\u30bf\\u5206\\u6790\\u3092\\u9ad8\\u901f\\u5316 - \\u30c7\\u30fc\\u30bf\\u304b\\u3089\\u65b0\\u3057\\u3044\\u4fa1\\u5024\\u3092\",\"tid\":\"1,6,7,9[8],11[10],13[12]\",\"u\":\"https://duckduckgo.com/y.js?ad_domain=datarobot.com&ad_provider=bingv7aa&ad_type=txad&eddgt=2_trBPli7jgDj1WnqJbnww%3D%3D&rut=e744d99a8df00b24df71f821ad4d1332080aa03267e50f0e988d284f58d9d2ef&u3=https%3A%2F%2Fwww.bing.com%2Faclick%3Fld%3De8tT9soRYLZabP1ukFkRsgNzVUCUzl89Y8xEqpxoqHqIlCI5wWbydNnN_PoAKHAa2Vsio83mXA_ax16t6rJ7XGkBv0Cg7_D1eg2QAuJgPKEam4VWI3rW40B03r1p11ZXN1Gd1847Vj05bAnJnPfgVyC8ZzFQxLxONmOI0Hg182z2bZUVII26BUAlUHaVZ7O_9FEXLJWw%26u%3DaHR0cHMlM2ElMmYlMmZ3d3cuZGF0YXJvYm90LmNvbSUyZmpwJTJmbHAlMmZhaS1mb3ItYnVzaW5lc3MlMmYlM2Z1dG1fbWVkaXVtJTNkc2VhcmNoJTI2dXRtX3NvdXJjZSUzZGJpbmclMjZ1dG1fY2FtcGFpZ24lM2RERU1PMjAyM0FsbFByb2R1Y3RzSlAwNjI2QlBTJTI2dXRtX3Rlcm0lM2RkYXRhcm9ib3QlMjZ1dG1fY29udGVudCUzZERSX2JyYW5kZWRfcnNhJTI2Y2FtcGFpZ25pZCUzZDUzMDcwODA5OSUyNmFkZ3JvdXBpZCUzZDEzNTAyMDI3NzQyMTc2OTglMjZhZGlkJTNkJTI2bXNjbGtpZCUzZDA2MTIwYzhmMTAxNzEwYTZiNmRiNjkyY2VmMWRiOTY1%26rlid%3D06120c8f101710a6b6db692cef1db965&vqd=4-91027509783546726889708070523412001433&iurl=%7B1%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26ID%3DDevEx%2C5058.1\"}], {\"page_load_url\":\"https://duckduckgo.com/y.js?ifu=%7B3%7Dappid%3D055AAD1BA669BEB8B048128DC89A107C678B527B%26rguid%3D309794dc72f748f6a2b95ce5c34fbcec&iurl=%7B2%7DIG%3D5E053B32922A4B5781ED405D9621559B%26CID%3D0176CEA622686E9D34D1DAA0235D6F30%26Type%3DEvent.CPT%26DATA%3D0\",\"visibility_url\":\"https://duckduckgo.com/y.js?ivu=%7B4%7Dtype%3Dmv%26reqver%3D1.0%26rg%3D309794dc72f748f6a2b95ce5c34fbcec\"});DDG.deep.signalSummary = \"\";DDG.inject('DDG.Data.languages.resultLanguages', {\"en\":[\"https://knowledge.dataiku.com/latest/use-cases/index.html\",\"https://community.dataiku.com/t5/Dataiku-Use-Cases-Success/tkb-p/use-cases\",\"https://www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\",\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\",\"https://community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\",\"https://www.datarobot.com/use-cases/\",\"https://academy.dataiku.com/page/use-cases\",\"https://www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\",\"https://www.g2.com/compare/datarobot-vs-dataiku-dss\",\"https://www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\",\"https://londondataconsulting.medium.com/dataiku-what-is-it-how-to-use-it-ultimate-guide-2023-47602c85a48b\",\"https://docs.datarobot.com/en/docs/api/guide/common-case/index.html\",\"https://www.datarobot.com/blog/introducing-the-datarobot-use-case-value-tracker/\",\"https://docs.datarobot.com/en/docs/workbench/wb-usecase/wb-build-usecase.html\",\"https://blog.dataiku.com/topic/use-cases-projects\",\"https://valohai.com/mlops-platforms-compared/\",\"https://www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\",\"https://pages.dataiku.com/experience-a-dataiku-demo\",\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\",\"https://www.dataiku.com/stories/\",\"https://www.dataiku.com/\",\"https://techcrunch.com/2022/12/13/ai-and-analytics-platform-dataiku-raises-200m-at-a-reduced-valuation/\",\"https://www.globenewswire.com/news-release/2022/11/17/2558152/0/en/Ben-Taylor-Joins-Dataiku-as-Chief-AI-Strategist.html\"]});DDG.deep.pageLayoutSummary = \"a1w4v1w19,w1\";DDG.inject('DDG.Data.languages.adLanguages', {});if (DDG.pageLayout) DDG.pageLayout.load('d',[{\"a\":\"Use Cases - Dataiku Knowledge Base Use Cases # These use cases allow you to practice what you've learned by building simplified, but complete use cases in Dataiku. Topics # Data Preparation Use Cases Classification Use Cases Clustering Use Cases Plugin Use Cases\",\"ae\":null,\"c\":\"https://knowledge.dataiku.com/latest/use-cases/index.html\",\"d\":\"knowledge.dataiku.com/latest/use-cases/index.html\",\"da\":\"\",\"h\":0,\"i\":\"knowledge.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Use Cases - Dataiku Knowledge Base\",\"u\":\"https://knowledge.dataiku.com/latest/use-cases/index.html\"},{\"a\":\"Community Dataiku Use Cases & Success Stories \\u26a0\\ufe0f Discover pioneering Dataiku use cases and success stories shared by customers, partners, academics, and nonprofits participating in the Dataiku Frontrunner Awards. Use the following labels to filter submissions by industry:\",\"ae\":null,\"c\":\"https://community.dataiku.com/t5/Dataiku-Use-Cases-Success/tkb-p/use-cases\",\"d\":\"community.dataiku.com/t5/Dataiku-Use-Cases-Success/tkb-p/use-cases\",\"da\":\"\",\"h\":0,\"i\":\"community.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku Use Cases & Success Stories - Dataiku Community\",\"u\":\"https://community.dataiku.com/t5/Dataiku-Use-Cases-Success/tkb-p/use-cases\"},{\"a\":\"Dataiku is a cross-platform desktop application that includes a broad range of tools, such as notebooks (similar to Jupyter Notebook), workflow management (similar to Apache Airflow), and automated machine learning. In general, Dataiku aims to replace many of your existing tools rather than to integrate with them.\",\"ae\":null,\"c\":\"https://www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\",\"d\":\"www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\",\"da\":\"\",\"h\":0,\"i\":\"www.datarevenue.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"ML Platforms: Dataiku vs. Alteryx vs. Sagemaker vs. Datarobot\",\"u\":\"https://www.datarevenue.com/en-blog/ml-platforms-dataiku-vs-alteryx-vs-sagemaker\"},{\"a\":\"Dataiku has a rating of 4.8 stars with 504 reviews. DataRobot has a rating of 4.6 stars with 508 reviews. See side-by-side comparisons of product capabilities, customer experience, pros and cons, and reviewer demographics to find the best fit for your organization. See more companies in the Data Science and Machine Learning Platforms market. PDF.\",\"ae\":null,\"c\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\",\"d\":\"www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\",\"da\":\"\",\"h\":0,\"i\":\"www.gartner.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku vs DataRobot 2024 | Gartner Peer Insights\",\"u\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/compare/dataiku-vs-datarobot\"},{\"a\":\"In my humble opinion DSS is a more a 'toolbox', where as DataRobot is an autoML platform. DataRobot is really good at what it does - if you have non-technical team who want to drop in data and leave everything to autoML then this may be the option for them.\",\"ae\":null,\"c\":\"https://community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\",\"d\":\"community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\",\"da\":\"\",\"h\":0,\"i\":\"community.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Solved: Dataiku vs DataRobot - Dataiku Community\",\"u\":\"https://community.dataiku.com/t5/General-Discussion/Dataiku-vs-DataRobot/m-p/9315\"},{\"a\":\"Use cases AI Use Cases AI-driven organizations around the world use DataRobot to solve their most pressing business problems. Build with Free Trial Recent Popular Filters Ready to Get Started? See how a value-driven approach to AI can accelerate time to impact. Start Free Trial\",\"ae\":null,\"c\":\"https://www.datarobot.com/use-cases/\",\"d\":\"www.datarobot.com/use-cases/\",\"da\":\"\",\"h\":0,\"i\":\"www.datarobot.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Machine Learning Use Cases | DataRobot AI Platform\",\"u\":\"https://www.datarobot.com/use-cases/\"},{\"a\":\"With Dataiku's AI Prepare assistant, you can work smarter, not harder. Simply describe the transformation you want to apply in natural language and the AI assistant automatically generates the necessary data preparation steps. The ability to modify both your prompt and the resulting steps means you can prepare data faster than ever, yet still ...\",\"ae\":null,\"c\":\"https://academy.dataiku.com/page/use-cases\",\"d\":\"academy.dataiku.com/page/use-cases\",\"da\":\"\",\"h\":0,\"i\":\"academy.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Use Cases - Dataiku\",\"u\":\"https://academy.dataiku.com/page/use-cases\"},{\"a\":\"84 Reviews and Ratings Path to AI Success Compare Dataiku DSS vs DataRobot. 103 verified user reviews and ratings of features, pros, cons, pricing, support and more.\",\"ae\":null,\"c\":\"https://www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\",\"d\":\"www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\",\"da\":\"\",\"h\":0,\"i\":\"www.trustradius.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku DSS vs DataRobot | TrustRadius\",\"u\":\"https://www.trustradius.com/compare-products/dataiku-dss-vs-datarobot\"},{\"a\":\"side-by-side comparison of DataRobot vs. Dataiku DSS. based on preference data from user reviews. DataRobot rates 4.4/5 stars with 26 reviews. By contrast, Dataiku DSS rates 4.3/5 stars with 36 reviews. Each product's score is calculated with real-time data from verified user reviews, to help you make the best choice between these two options ...\",\"ae\":null,\"c\":\"https://www.g2.com/compare/datarobot-vs-dataiku-dss\",\"d\":\"www.g2.com/compare/datarobot-vs-dataiku-dss\",\"da\":\"\",\"h\":0,\"i\":\"www.g2.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare DataRobot vs. Dataiku DSS | G2\",\"u\":\"https://www.g2.com/compare/datarobot-vs-dataiku-dss\"},{\"a\":\"Use case: Choose Datarobot if you have data stored in spreadsheets and are seeking a platform that is the simplest, albeit one with limited flexibility, ... Dataiku vs. Datarobot .\",\"ae\":null,\"b\":\"li\\tLinkedIn\\twww.linkedin.com\",\"c\":\"https://www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\",\"d\":\"www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\",\"da\":\"\",\"e\":\"2023-08-11T00:00:00.0000000\",\"h\":0,\"i\":\"www.linkedin.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Managed Machine Learning Platforms: A Comparative Analysis - LinkedIn\",\"u\":\"https://www.linkedin.com/pulse/managed-machine-learning-platforms-comparative-analysis/\"},{\"a\":\"Jan 11, 2023 Dataiku is an artificial intelligence platform created in France in 2013. It has since become one of the world's benchmarks for data science and machine learning studios. What is...\",\"ae\":null,\"c\":\"https://londondataconsulting.medium.com/dataiku-what-is-it-how-to-use-it-ultimate-guide-2023-47602c85a48b\",\"d\":\"londondataconsulting.medium.com/dataiku-what-is-it-how-to-use-it-ultimate-guide-2023-47602c85a48b\",\"da\":\"translations\",\"e\":\"2023-01-11T00:00:00.0000000\",\"h\":0,\"i\":\"londondataconsulting.medium.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku: What is it? How to use it? Ultimate Guide 2023\",\"u\":\"https://londondataconsulting.medium.com/dataiku-what-is-it-how-to-use-it-ultimate-guide-2023-47602c85a48b\"},{\"a\":\"Use cases for version 2.x. Notebooks for uses cases that use methods for 2.x versions of DataRobot's Python client. Measure price elasticity of demand. A use case to identify relationships between price and demand, maximize revenue by properly pricing products, and monitor price elasticities for changes in price and demand. Insurance claim triage.\",\"ae\":null,\"c\":\"https://docs.datarobot.com/en/docs/api/guide/common-case/index.html\",\"d\":\"docs.datarobot.com/en/docs/api/guide/common-case/index.html\",\"da\":\"\",\"h\":0,\"i\":\"docs.datarobot.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Common use cases: DataRobot docs - DataRobot AI Platform\",\"u\":\"https://docs.datarobot.com/en/docs/api/guide/common-case/index.html\"},{\"a\":\"With the Use Case Value Tracker, you can manage the project lifecycle and understand the value associated with each step. It also enables you to associate and organize all your DataRobot artifacts (e.g., datasets, models, deployments, applications, etc.) around a given use case for a holistic view. In addition to the project management aspects ...\",\"ae\":null,\"c\":\"https://www.datarobot.com/blog/introducing-the-datarobot-use-case-value-tracker/\",\"d\":\"www.datarobot.com/blog/introducing-the-datarobot-use-case-value-tracker/\",\"da\":\"\",\"h\":0,\"i\":\"www.datarobot.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Introducing the DataRobot Use Case Value Tracker\",\"u\":\"https://www.datarobot.com/blog/introducing-the-datarobot-use-case-value-tracker/\"},{\"a\":\"Use Cases are folder-like containers inside of DataRobot Workbench that allow you to group everything related to solving a specific business problem\\u2014datasets, models, experiments, No-Code AI Apps, and notebooks\\u2014inside of a single, manageable entity. You can share whole Use Cases as well as the individual assets they contain.\",\"ae\":null,\"c\":\"https://docs.datarobot.com/en/docs/workbench/wb-usecase/wb-build-usecase.html\",\"d\":\"docs.datarobot.com/en/docs/workbench/wb-usecase/wb-build-usecase.html\",\"da\":\"\",\"e\":\"2023-09-15T00:00:00.0000000\",\"h\":0,\"i\":\"docs.datarobot.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Use Cases: DataRobot docs\",\"u\":\"https://docs.datarobot.com/en/docs/workbench/wb-usecase/wb-build-usecase.html\"},{\"a\":\"January 2, 2024 Use Cases & Projects, Featured Sophie Dionnet Leveraging AI to Cut Costs December 29, 2023 Data Basics, Featured\",\"ae\":null,\"c\":\"https://blog.dataiku.com/topic/use-cases-projects\",\"d\":\"blog.dataiku.com/topic/use-cases-projects\",\"da\":\"\",\"h\":0,\"i\":\"blog.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Blog - Dataiku | Use Cases & Projects\",\"u\":\"https://blog.dataiku.com/topic/use-cases-projects\"},{\"a\":\"The platforms we've chosen for our analysis are ClearML, cnvrg.io, Dataiku, Datarobot, Iguazio, Sagemaker, Seldon and Valohai from the managed side, and Flyte, Kubeflow, MLflow and Metaflow from the open-source side. This is by no means an exhaustive list of all the MLOps tools out there.\",\"ae\":null,\"c\":\"https://valohai.com/mlops-platforms-compared/\",\"d\":\"valohai.com/mlops-platforms-compared/\",\"da\":\"\",\"h\":0,\"i\":\"valohai.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"MLOps Platforms Compared - Valohai\",\"u\":\"https://valohai.com/mlops-platforms-compared/\"},{\"a\":\"DataRobot. DSS is for all companies, whatever their expertise, industry or size, that want to create their own data-driven strategic advantages by transforming their raw data into business impacting predictions. Cloud based machine learning platform which helps enterprises scale data science capabilities through deploying machine learning ...\",\"ae\":null,\"c\":\"https://www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\",\"d\":\"www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\",\"da\":\"translations\",\"h\":0,\"i\":\"www.capterra.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Compare Dataiku vs DataRobot 2024 | Capterra\",\"u\":\"https://www.capterra.com/machine-learning-software/compare/142192-179303/Data-Science-Studio-DSS-vs-DataRobot\"},{\"a\":\"For Every Industry & Use Case. Organizations that use Dataiku elevate their people (whether technical and working in code or on the business side and low- or no-code) to extraordinary, arming them with the ability to make better day-to-day decisions with data across: Banking & Insurance. Pharmaceuticals. Manufacturing. Telecommunications.\",\"ae\":null,\"c\":\"https://pages.dataiku.com/experience-a-dataiku-demo\",\"d\":\"pages.dataiku.com/experience-a-dataiku-demo\",\"da\":\"\",\"h\":0,\"i\":\"pages.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Check Out This Dataiku Demo\",\"u\":\"https://pages.dataiku.com/experience-a-dataiku-demo\"},{\"a\":\"4 Star 24% 3 Star 1% 2 Star 0% 1 Star 0% Distribution based on 504 ratings Customer Experience Evaluation & Contracting 4.6 Integration & Deployment 4.7 Service & Support 4.8 Product Capabilities 4.8 FREE View and Download Peer Insights About Dataiku\",\"ae\":null,\"c\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\",\"d\":\"www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\",\"da\":\"\",\"h\":0,\"i\":\"www.gartner.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku Reviews, Ratings & Features 2024 | Gartner Peer Insights\",\"u\":\"https://www.gartner.com/reviews/market/data-science-and-machine-learning-platforms/vendor/dataiku/product/dataiku\"},{\"a\":\"Read The Full Case Study U.S. Venture + Dataiku: Upskilling Analysts to Save Thousands of Hours The Data and Analytics team at U.S. Venture was built to usher the company into the future of data science and AI.\",\"ae\":null,\"c\":\"https://www.dataiku.com/stories/\",\"d\":\"www.dataiku.com/stories/\",\"da\":\"\",\"h\":0,\"i\":\"www.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Stories | Dataiku\",\"u\":\"https://www.dataiku.com/stories/\"},{\"a\":\"MLOps Deploy, monitor, and maintain machine learning models, all in a single platform. Explore the Capability Collaboration With Dataiku, teams can move beyond the lab and build real and safe Generative AI applications at enterprise scale. Explore the Capability Governance\",\"ae\":null,\"c\":\"https://www.dataiku.com/\",\"d\":\"www.dataiku.com\",\"da\":\"\",\"h\":0,\"i\":\"www.dataiku.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Dataiku | Everyday AI, Extraordinary People\",\"u\":\"https://www.dataiku.com/\"},{\"a\":\"The company today announced that it raised $200 million in a Series F round led by Wellington Management at a $3.7 billion valuation, down from the $4.6 billion that Dataiku received in August ...\",\"ae\":null,\"b\":\"tc\\tTechcrunch\\ttechcrunch.com\",\"c\":\"https://techcrunch.com/2022/12/13/ai-and-analytics-platform-dataiku-raises-200m-at-a-reduced-valuation/\",\"d\":\"techcrunch.com/2022/12/13/ai-and-analytics-platform-dataiku-raises-200m-at-a-reduced-valuation/\",\"da\":\"translations\",\"e\":\"2022-12-13T17:10:00.0000000\",\"h\":0,\"i\":\"techcrunch.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"AI and analytics platform Dataiku raises $200M at a reduced valuation\",\"u\":\"https://techcrunch.com/2022/12/13/ai-and-analytics-platform-dataiku-raises-200m-at-a-reduced-valuation/\"},{\"a\":\"Today, more than 500 companies worldwide use Dataiku to integrate and streamline their use of data, analytics, and AI, driving diverse use cases from fraud detection and customer churn prevention ...\",\"ae\":null,\"c\":\"https://www.globenewswire.com/news-release/2022/11/17/2558152/0/en/Ben-Taylor-Joins-Dataiku-as-Chief-AI-Strategist.html\",\"d\":\"www.globenewswire.com/news-release/2022/11/17/2558152/0/en/Ben-Taylor-Joins-Dataiku-as-Chief-AI-Strategist.html\",\"da\":\"translations\",\"e\":\"2022-11-17T13:00:00.0000000\",\"h\":0,\"i\":\"www.globenewswire.com\",\"k\":null,\"m\":0,\"o\":0,\"p\":0,\"s\":\"bingv7aa\",\"t\":\"Ben Taylor Joins Dataiku as Chief AI Strategist - GlobeNewswire\",\"u\":\"https://www.globenewswire.com/news-release/2022/11/17/2558152/0/en/Ben-Taylor-Joins-Dataiku-as-Chief-AI-Strategist.html\"},{\"n\":\"/d.js?q=Dataiku%20and%20DataRobot%20use%20cases&kl=wt-wt&l=wt-wt&p=&s=23&ex=-1&ct=US&sp=0&vqd=4-60481969350525797892441552954401970387\"}]);DDG.duckbar.load('images');DDG.duckbar.load('news');DDG.duckbar.load('videos', {\"ads\":[],\"query\":\"Dataiku and DataRobot use cases\",\"queryEncoded\":\"Dataiku%20and%20DataRobot%20use%20cases\",\"response_type\":\"places\",\"results\":[{\"content\":\"https://www.youtube.com/watch?v=ryZRRIjQ5Z8\",\"description\":\"If you're a code-first data practitioner, Dataiku helps you efficiently build high quality data pipelines and models in a number of ways. CHECK OUT DATAIKU: https://bit.ly/36XBlpK EGG ON AIR: https://bit.ly/37GhXMY BRIGHTTALK WEBINARS: https://bit.ly/33TIRjn DATA SCIENCE PIONEERS DOCUMENTARY: https://bit.ly/36V3rBF PARTNER ECOSYSTEM: https ...\",\"duration\":\"10:43\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/ryZRRIjQ5Z8?autoplay=1\",\"image_token\":\"8a4abca8613c6680a108591849e5d7b13b86111004ae004898a7f059b64c8355\",\"images\":{\"large\":\"https://tse4.mm.bing.net/th?id=OVP.WoendyuZJ9qxql-n6jit5AEsDh&pid=Api\",\"medium\":\"https://tse4.mm.bing.net/th?id=OVP.WoendyuZJ9qxql-n6jit5AEsDh&pid=Api\",\"motion\":\"https://tse4.mm.bing.net/th?id=OM1.cmvppfhHVUeE4Q_1684256861&pid=Api\",\"small\":\"https://tse4.mm.bing.net/th?id=OVP.WoendyuZJ9qxql-n6jit5AEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2021-06-08T21:15:02.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":12391},\"title\":\"Dataiku Demo for Data Scientists and Coders\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=jKL_I0SCl_E\",\"description\":\"This video showcases the Clinical Trial Explorer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest of our ...\",\"duration\":\"1:50\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/jKL_I0SCl_E?autoplay=1\",\"image_token\":\"7b19602fe6d9b761aa3cc138448cc632ddbed31da3abf2687f36705f5945973d\",\"images\":{\"large\":\"https://tse3.mm.bing.net/th?id=OVP.wfgHp53woiVosZ37-1HtnwEsDh&pid=Api\",\"medium\":\"https://tse3.mm.bing.net/th?id=OVP.wfgHp53woiVosZ37-1HtnwEsDh&pid=Api\",\"motion\":\"https://tse3.mm.bing.net/th?id=OM2.ZX_yq0xmyCGZBg_1696372683&pid=Api\",\"small\":\"https://tse3.mm.bing.net/th?id=OVP.wfgHp53woiVosZ37-1HtnwEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:19:00.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":150},\"title\":\"Clinical Trial Explorer\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=lASesA4gNFI\",\"description\":\"This video showcases the CO2 Forecast Analyzer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest of our ...\",\"duration\":\"1:50\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/lASesA4gNFI?autoplay=1\",\"image_token\":\"a09328adca01a788783d759561c2f9c9d4d214e5a26f1462d2b6b69f21a2d478\",\"images\":{\"large\":\"https://tse2.mm.bing.net/th?id=OVP.ZhQI7z-IMlcVnyeMJuGlAQEsDh&pid=Api\",\"medium\":\"https://tse2.mm.bing.net/th?id=OVP.ZhQI7z-IMlcVnyeMJuGlAQEsDh&pid=Api\",\"motion\":\"\",\"small\":\"https://tse2.mm.bing.net/th?id=OVP.ZhQI7z-IMlcVnyeMJuGlAQEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:16:20.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":49},\"title\":\"CO2 Forecast Analyzer\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=RecpD6Vtzj4\",\"description\":\"This video showcases the LLM-Enhanced ESG Document Intelligence use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the ...\",\"duration\":\"1:20\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/RecpD6Vtzj4?autoplay=1\",\"image_token\":\"6f797accb167e2e6ff7265e35116cdeb9f1c641b1df47932d9597b61b0108614\",\"images\":{\"large\":\"https://tse2.mm.bing.net/th?id=OVP.bm4gAiJOOmKV7uup6LU9pgEsDh&pid=Api\",\"medium\":\"https://tse2.mm.bing.net/th?id=OVP.bm4gAiJOOmKV7uup6LU9pgEsDh&pid=Api\",\"motion\":\"https://tse2.mm.bing.net/th?id=OM1.M8WXwCQ79nrqEA_1691502936&pid=Api\",\"small\":\"https://tse2.mm.bing.net/th?id=OVP.bm4gAiJOOmKV7uup6LU9pgEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:30:00.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":382},\"title\":\"LLM-Enhanced ESG Document Intelligence\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=zLW0TkJoHLw\",\"description\":\"This video showcases the Demand Forecast Analyzer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest of our ...\",\"duration\":\"1:42\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/zLW0TkJoHLw?autoplay=1\",\"image_token\":\"524eb9572bf9342b859509285d39ec4661fc572cb1452307acc5341b56bab921\",\"images\":{\"large\":\"https://tse1.mm.bing.net/th?id=OVP.yIekQ2cMUesOPJTfsYIZzQHgFo&pid=Api\",\"medium\":\"https://tse1.mm.bing.net/th?id=OVP.yIekQ2cMUesOPJTfsYIZzQHgFo&pid=Api\",\"motion\":\"\",\"small\":\"https://tse1.mm.bing.net/th?id=OVP.yIekQ2cMUesOPJTfsYIZzQHgFo&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:15:02.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":4},\"title\":\"LLM-Enhanced Demand Forecast\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=L-Yys0fzuVY\",\"description\":\"This video showcases the Production Quality Data Explorer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest ...\",\"duration\":\"1:47\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/L-Yys0fzuVY?autoplay=1\",\"image_token\":\"82924713be1dba83d67124fcaa6cc6afd163900a0c40f25fcf6c144ed0e36536\",\"images\":{\"large\":\"https://tse4.mm.bing.net/th?id=OVP.teiC0mX9nKCbH8qeo52udwEsDh&pid=Api\",\"medium\":\"https://tse4.mm.bing.net/th?id=OVP.teiC0mX9nKCbH8qeo52udwEsDh&pid=Api\",\"motion\":\"https://tse4.mm.bing.net/th?id=OM2.IwRaoLRWcQXzag_1691419996&pid=Api\",\"small\":\"https://tse4.mm.bing.net/th?id=OVP.teiC0mX9nKCbH8qeo52udwEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:28:29.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":175},\"title\":\"Production Quality Data Explorer\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=FluiuHuaU8A\",\"description\":\"In this breakout session of Dataiku's Product Days 2021, you will see a demo of Dataiku's Data Science Studio, the centralized, collaborative, and end-to-end platform for data science in the enterprise. CHECK OUT DATAIKU: https://bit.ly/36XBlpK EGG ON AIR: https://bit.ly/37GhXMY BRIGHTTALK WEBINARS: https://bit.ly/33TIRjn DATA SCIENCE PIONEERS ...\",\"duration\":\"13:50\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/FluiuHuaU8A?autoplay=1\",\"image_token\":\"2943fa8c1580f2936fc11667d670c0b827b94ff3d16b897f8b5ef2e2426487b3\",\"images\":{\"large\":\"https://tse2.mm.bing.net/th?id=OVP.RIM-ftwDZjYP58RimJfgwwEsDh&pid=Api\",\"medium\":\"https://tse2.mm.bing.net/th?id=OVP.RIM-ftwDZjYP58RimJfgwwEsDh&pid=Api\",\"motion\":\"https://tse2.mm.bing.net/th?id=OM1.MIQ7BoQz1MVkNw_1662248868&pid=Api\",\"small\":\"https://tse2.mm.bing.net/th?id=OVP.RIM-ftwDZjYP58RimJfgwwEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2021-07-08T15:56:22.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":3844},\"title\":\"Introduction to Dataiku Data Science | Product Days 2021\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=6TEU5JboP7k\",\"description\":\"This video showcases the LLM-Enhanced Next Best Offer use case from the Dataiku Generative AI Use Case Collection. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore more Generative AI use cases | https://experience.dataiku.com/generative-ai To explore more about Dataiku, check out the rest of ...\",\"duration\":\"1:47\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/6TEU5JboP7k?autoplay=1\",\"image_token\":\"a3bc327ff2f099462935a8979bb599655c7a88a44e25a64bfea7e5973f773158\",\"images\":{\"large\":\"https://tse2.mm.bing.net/th?id=OVP.Q6SP0MmL89M_TnLvNPG4oQEsDh&pid=Api\",\"medium\":\"https://tse2.mm.bing.net/th?id=OVP.Q6SP0MmL89M_TnLvNPG4oQEsDh&pid=Api\",\"motion\":\"https://tse2.mm.bing.net/th?id=OM2.wXNo1CUgYV4Flg_1694065487&pid=Api\",\"small\":\"https://tse2.mm.bing.net/th?id=OVP.Q6SP0MmL89M_TnLvNPG4oQEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:34:32.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":462},\"title\":\"LLM-Enhanced Next Best Offer\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=UVbrpX8Zkn8\",\"description\":\"Move beyond the lab and build real and safe Generative AI applications at enterprise scale. Dataiku brings enterprise-grade development tools, pre-built use cases, and AI-powered assistants throughout the platform. Learn more about Dataiku for Generative AI | https://www.dataiku.com/product/generative-ai/ Explore Generative AI use cases | https ...\",\"duration\":\"2:07\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/UVbrpX8Zkn8?autoplay=1\",\"image_token\":\"3968d2d01ff722efa156290344ab0b37164e57d7efa50905c346ea1cc1a5d369\",\"images\":{\"large\":\"https://tse1.mm.bing.net/th?id=OVP.F-xH3-wKfTjM3YMshnjWwwEsDh&pid=Api\",\"medium\":\"https://tse1.mm.bing.net/th?id=OVP.F-xH3-wKfTjM3YMshnjWwwEsDh&pid=Api\",\"motion\":\"https://tse1.mm.bing.net/th?id=OM1.z-FRwxQ_NByK_A_1689135755&pid=Api\",\"small\":\"https://tse1.mm.bing.net/th?id=OVP.F-xH3-wKfTjM3YMshnjWwwEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-06-22T12:25:16.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":1100},\"title\":\"Dataiku for Generative AI: Real Applications, Real Safety\",\"uploader\":\"Dataiku\"},{\"content\":\"https://www.youtube.com/watch?v=-amc9iVauuE\",\"description\":\"Dataiku is the leading platform for Everyday AI, systemizing the use of data for exceptional business results. In today's video we will take a tour of Dataiku's end to end capabilities by exploring a real life use case around environmental impact. Let's take a look at how a data science team with different skills can work together to turn ...\",\"duration\":\"12:35\",\"embed_html\":\"\",\"embed_url\":\"http://www.youtube.com/embed/-amc9iVauuE?autoplay=1\",\"image_token\":\"2a05a65ad8a2727aa5c48b8daa7f9ec363a24d4336a3509016d4b200c9d003cd\",\"images\":{\"large\":\"https://tse1.mm.bing.net/th?id=OVP.Az9RhdSVwpXe56mGcs6FqQEsDh&pid=Api\",\"medium\":\"https://tse1.mm.bing.net/th?id=OVP.Az9RhdSVwpXe56mGcs6FqQEsDh&pid=Api\",\"motion\":\"https://tse1.mm.bing.net/th?id=OM1.Q2OhN9DzfowU6A_1685345657&pid=Api\",\"small\":\"https://tse1.mm.bing.net/th?id=OVP.Az9RhdSVwpXe56mGcs6FqQEsDh&pid=Api\"},\"provider\":\"Bing\",\"published\":\"2023-01-09T21:12:27.0000000\",\"publisher\":\"YouTube\",\"statistics\":{\"viewCount\":9768},\"title\":\"End to End Demo 2023\",\"uploader\":\"Dataiku\"}],\"vqd\":{\"Dataiku%20and%20DataRobot%20use%20cases\":\"4-60481969350525797892441552954401970387\"}});DDG.duckbar.loadModule('related_searches');if (DDG.pageLayout) DDG.pageLayout.initialize({\"mainline\":{\"items\":[[\"ad\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"videos\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"],[\"organic\"]]},\"sidebar\":{\"items\":[[\"organic\"]]}}, { start: 0 });DDG.deep.emit(\"load:completed\");", + "aiohttp-post-https://google.serper.dev/search-{\"data\": \"[{\\\"num\\\": 6, \\\"page\\\": 1, \\\"q\\\": \\\"ai agent\\\"}]\", \"headers\": {\"Content-Type\": \"application/json\", \"X-API-KEY\": \"mock-serper-key\"}}": [ + { + "searchParameters": { + "q": "ai agent", + "num": 6, + "page": 1, + "type": "search", + "engine": "google" + }, + "organic": [ + { + "title": "AI Agent • Supercharge Your Workflows with AI", + "link": "https://aiagent.app/", + "snippet": "A web app that makes choices and performs tasks on its own, based on the goals set by you. How Does it Work?", + "position": 1 + }, + { + "title": "Intelligent agent - Wikipedia", + "link": "https://en.wikipedia.org/wiki/Intelligent_agent", + "snippet": "In artificial intelligence, an intelligent agent (IA) is an agent acting in an intelligent manner; It perceives its environment, takes actions autonomously ...", + "position": 2 + }, + { + "title": "What is an AI agent? | Zapier", + "link": "https://zapier.com/blog/ai-agent/", + "snippet": "AI Agent is a flexible app that lets you create your own agents, by picking a name, an objective, and the AI model it should use (GPT-3.5 Turbo ...", + "date": "Jun 1, 2023", + "position": 3 + }, + { + "title": "Google DeepMind Veteran Departs to Launch AI Agent Startup", + "link": "https://www.theinformation.com/articles/google-deepmind-veteran-departs-to-launch-ai-agent-startup", + "snippet": "The concept of AI agents—bots with the ability to plan and work toward a goal with minimal user guidance—first took off a year ago after ...", + "date": "10 hours ago", + "position": 4 + }, + { + "title": "AI Agents And The Era Of The Intelligent Interface - Forbes", + "link": "https://www.forbes.com/sites/davidarmano/2023/12/07/ai-agents-and-the-era-of-the-intelligent-interface/", + "snippet": "Conversing with several AI Agents connected to various systems is poised to become the next significant evolution of human-computer ...", + "date": "Dec 7, 2023", + "position": 5 + } + ], + "topStories": [ + { + "title": "Google DeepMind Veteran Departs to Launch AI Agent Startup", + "link": "https://www.theinformation.com/articles/google-deepmind-veteran-departs-to-launch-ai-agent-startup", + "source": "The Information", + "date": "10 hours ago", + "imageUrl": "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcSOUiQnjnTQpnKKDk0hnXhpIdVvwyifhK3VjZuTey9Uq3J1S8l7OB95iWMrKQ&s" + }, + { + "title": "Bitcoin to Become Native Currency for AI Agents, Former Meta Exec Predicts", + "link": "https://u.today/bitcoin-to-become-native-currency-for-ai-agents-former-meta-exec-predicts", + "source": "U.Today", + "date": "2 days ago", + "imageUrl": "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcRV9Ydu_Dou8HvI9E25KAn7nKmxk6Q-CB1cvT0dIxSudXhZPpGCR1vj0NCdaw&s" + }, + { + "title": "Building AI agents with Semantic Kernel", + "link": "https://www.infoworld.com/article/3712423/building-ai-agents-with-semantic-kernel.html", + "source": "InfoWorld", + "date": "5 days ago", + "imageUrl": "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcS0NAiT2vVMB14Ff56syKnS3g4jrNN5LIskrtxqqdViPyrBsLCuCrQWu9ojdA&s" + }, + { + "title": "AI agents help explain other AI systems", + "link": "https://news.mit.edu/2024/ai-agents-help-explain-other-ai-systems-0103", + "source": "MIT News", + "date": "4 weeks ago", + "imageUrl": "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcR-0QJKlbSMOP0wYSfB70p4_JCWHEkc6oAkLhBXVHX3ZVATBCRWTp08JY8x4w&s" + }, + { + "title": "CES 2024: LG announces walking, talking, 'Jetsons-esque' smart home robot", + "link": "https://mashable.com/article/ces-2024-lg-announcement-ai-agent-smart-home-robot", + "source": "Mashable", + "date": "3 weeks ago", + "imageUrl": "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcRVIiyOId49n_-CwPODLHIP9t6HioG05EeI_dvwvg6WNfFBcsLliI_Xhr6U-Q&s" + }, + { + "title": "Develop Your First AI Agent: Deep Q-Learning", + "link": "https://towardsdatascience.com/develop-your-first-ai-agent-deep-q-learning-375876ee2472", + "source": "Towards Data Science", + "date": "1 month ago", + "imageUrl": "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcSEzOgQuvwfalC2s5HasdRiv2IqdMLgtuUOBJv1xkVGH-Vg_bJmavQk88I1eA&s" + } + ], + "relatedSearches": [ + { + "query": "AI agent GPT" + }, + { + "query": "AI agent OpenAI" + }, + { + "query": "AI agent examples" + }, + { + "query": "Ai agent jobs" + }, + { + "query": "AI agents ChatGPT" + }, + { + "query": "AI agent Microsoft" + }, + { + "query": "AI agent free" + }, + { + "query": "AI Agent app" + } + ] + } + ] } \ No newline at end of file diff --git a/tests/metagpt/learn/test_google_search.py b/tests/metagpt/learn/test_google_search.py index da32e8923..7fda6436a 100644 --- a/tests/metagpt/learn/test_google_search.py +++ b/tests/metagpt/learn/test_google_search.py @@ -1,27 +1,21 @@ -import asyncio - +import pytest from pydantic import BaseModel from metagpt.learn.google_search import google_search +from metagpt.tools import SearchEngineType -async def mock_google_search(): +@pytest.mark.asyncio +async def test_google_search(search_engine_mocker): class Input(BaseModel): input: str inputs = [{"input": "ai agent"}] - for i in inputs: seed = Input(**i) - result = await google_search(seed.input) + result = await google_search( + seed.input, + engine=SearchEngineType.SERPER_GOOGLE, + serper_api_key="mock-serper-key", + ) assert result != "" - - -def test_suite(): - loop = asyncio.get_event_loop() - task = loop.create_task(mock_google_search()) - loop.run_until_complete(task) - - -if __name__ == "__main__": - test_suite() diff --git a/tests/mock/mock_aiohttp.py b/tests/mock/mock_aiohttp.py index 4690bf4b5..49dcdba79 100644 --- a/tests/mock/mock_aiohttp.py +++ b/tests/mock/mock_aiohttp.py @@ -39,3 +39,7 @@ class MockAioResponse: data = await self.response.json(*args, **kwargs) self.rsp_cache[self.key] = data return data + + def raise_for_status(self): + if self.response: + self.response.raise_for_status() From d74dab9bec1a42503984b9acd1c247d8b151b323 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 31 Jan 2024 16:03:16 +0800 Subject: [PATCH 527/637] update sd ut --- examples/imitate_webpage.py | 4 +- metagpt/tools/libs/sd_engine.py | 14 ++--- tests/metagpt/tools/libs/test_sd_engine.py | 66 +++++++++++++++++++--- 3 files changed, 63 insertions(+), 21 deletions(-) diff --git a/examples/imitate_webpage.py b/examples/imitate_webpage.py index 6c12c7eda..b69101861 100644 --- a/examples/imitate_webpage.py +++ b/examples/imitate_webpage.py @@ -9,7 +9,7 @@ from metagpt.roles.code_interpreter import CodeInterpreter async def main(): - web_url = 'https://pytorch.org/' + web_url = "https://pytorch.org/" prompt = f"""This is a URL of webpage: '{web_url}' . Firstly, utilize Selenium and WebDriver for rendering. Secondly, convert image to a webpage including HTML, CSS and JS in one go. @@ -20,7 +20,7 @@ Note: All required dependencies and environments have been fully installed and c await ci.run(prompt) -if __name__ == '__main__': +if __name__ == "__main__": import asyncio asyncio.run(main()) diff --git a/metagpt/tools/libs/sd_engine.py b/metagpt/tools/libs/sd_engine.py index 794758f77..7f182f380 100644 --- a/metagpt/tools/libs/sd_engine.py +++ b/metagpt/tools/libs/sd_engine.py @@ -13,7 +13,8 @@ import requests from aiohttp import ClientSession from PIL import Image, PngImagePlugin -from metagpt.const import SD_OUTPUT_FILE_REPO +# +from metagpt.const import SD_OUTPUT_FILE_REPO, SOURCE_ROOT from metagpt.logs import logger from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool @@ -82,7 +83,7 @@ class SDEngine: return self.payload def save(self, imgs, save_name=""): - save_dir = CONFIG.workspace_path / SD_OUTPUT_FILE_REPO + save_dir = SOURCE_ROOT / SD_OUTPUT_FILE_REPO if not save_dir.exists(): save_dir.mkdir(parents=True, exist_ok=True) batch_decode_base64_to_image(imgs, str(save_dir), save_name=save_name) @@ -113,17 +114,10 @@ class SDEngine: rsp_json = json.loads(data) imgs = rsp_json["images"] + logger.info(f"callback rsp json is {rsp_json.keys()}") return imgs - async def run_i2i(self): - # todo: 添加图生图接口调用 - raise NotImplementedError - - async def run_sam(self): - # todo:添加SAM接口调用 - raise NotImplementedError - def decode_base64_to_image(img, save_name): image = Image.open(io.BytesIO(base64.b64decode(img.split(",", 1)[0]))) diff --git a/tests/metagpt/tools/libs/test_sd_engine.py b/tests/metagpt/tools/libs/test_sd_engine.py index 363cf96b9..322976806 100644 --- a/tests/metagpt/tools/libs/test_sd_engine.py +++ b/tests/metagpt/tools/libs/test_sd_engine.py @@ -2,20 +2,51 @@ # @Date : 1/10/2024 10:07 PM # @Author : stellahong (stellahong@fuzhi.ai) # @Desc : +import base64 +import io + import pytest +from aioresponses import aioresponses +from PIL import Image, ImageDraw +from requests_mock import Mocker from metagpt.tools.libs.sd_engine import SDEngine +def generate_mock_image_data(): + # 创建一个简单的图片对象 + image = Image.new("RGB", (100, 100), color="white") + draw = ImageDraw.Draw(image) + draw.text((10, 10), "Mock Image", fill="black") + + # 将图片转换为二进制数据 + with io.BytesIO() as buffer: + image.save(buffer, format="PNG") + image_binary = buffer.getvalue() + + # 对图片二进制数据进行 base64 编码 + image_base64 = base64.b64encode(image_binary).decode("utf-8") + + return image_base64 + + def test_sd_tools(): - engine = SDEngine() - prompt = "1boy, hansom" - engine.construct_payload(prompt) - engine.simple_run_t2i(engine.payload) + engine = SDEngine(sd_url="http://localhost:7860") + # 使用 requests_mock.Mocker 替换 simple_run_t2i 的网络请求 + mock_imgs = generate_mock_image_data() + with Mocker() as mocker: + # 指定模拟请求的返回值 + mocker.post(engine.sd_t2i_url, json={"images": [mock_imgs]}) + + # 在被测试代码中调用 simple_run_t2i + result = engine.simple_run_t2i(engine.payload) + + # 断言结果是否是指定的 Mock 返回值 + assert len(result) == 1 def test_sd_construct_payload(): - engine = SDEngine() + engine = SDEngine(sd_url="http://localhost:7860") prompt = "1boy, hansom" engine.construct_payload(prompt) assert "negative_prompt" in engine.payload @@ -23,8 +54,25 @@ def test_sd_construct_payload(): @pytest.mark.asyncio async def test_sd_asyn_t2i(): - engine = SDEngine() - prompt = "1boy, hansom" + engine = SDEngine(sd_url="http://example.com/mock_sd_t2i") + + prompt = "1boy, hansom" engine.construct_payload(prompt) - await engine.run_t2i([engine.payload]) - assert "negative_prompt" in engine.payload + # 构建mock数据 + mock_imgs = generate_mock_image_data() + + mock_responses = aioresponses() + + # 手动启动模拟 + mock_responses.start() + + try: + # 指定模拟请求的返回值 + mock_responses.post("http://example.com/mock_sd_t2i/sdapi/v1/txt2img", payload={"images": [mock_imgs]}) + + # 在被测试代码中调用异步函数 run_t2i + await engine.run_t2i([engine.payload]) + + finally: + # 手动停止模拟 + mock_responses.stop() From 28b0323d7552f109f186b06bdc7505c93db5be85 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 31 Jan 2024 16:14:33 +0800 Subject: [PATCH 528/637] add package for test_sd_engine --- requirements.txt | 2 ++ 1 file changed, 2 insertions(+) diff --git a/requirements.txt b/requirements.txt index 66b3c9fc0..4a9c0ab30 100644 --- a/requirements.txt +++ b/requirements.txt @@ -66,3 +66,5 @@ google-generativeai==0.3.2 # playwright==1.40.0 # playwright extras require anytree ipywidgets==8.1.1 +aioresponses +requests_mock \ No newline at end of file From c44d08ceb05ee177915506a84fc40b021ef4698c Mon Sep 17 00:00:00 2001 From: stellahsr Date: Wed, 31 Jan 2024 16:30:50 +0800 Subject: [PATCH 529/637] rm config get in dev --- metagpt/tools/libs/sd_engine.py | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/metagpt/tools/libs/sd_engine.py b/metagpt/tools/libs/sd_engine.py index 57a025f3c..7001eadf5 100644 --- a/metagpt/tools/libs/sd_engine.py +++ b/metagpt/tools/libs/sd_engine.py @@ -56,11 +56,9 @@ default_negative_prompt = "(easynegative:0.8),black, dark,Low resolution" @register_tool(tool_type=ToolTypeEnum.STABLE_DIFFUSION.value) class SDEngine: def __init__(self, sd_url=""): - from metagpt.config2 import config - # Initialize the SDEngine with configuration - self.sd_url = sd_url if sd_url else config.get("SD_URL") - self.sd_t2i_url = f"{self.sd_url}{config.get('SD_T2I_API')}" + self.sd_url = sd_url + self.sd_t2i_url = f"{self.sd_url}/sdapi/v1/txt2img" # Define default payload settings for SD API self.payload = payload logger.info(self.sd_t2i_url) From 54388d0a8792b682883e917122e92ac8aab63118 Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 31 Jan 2024 17:34:48 +0800 Subject: [PATCH 530/637] refine comments --- metagpt/strategy/solver.py | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/metagpt/strategy/solver.py b/metagpt/strategy/solver.py index bd21dda3e..e7d61a881 100644 --- a/metagpt/strategy/solver.py +++ b/metagpt/strategy/solver.py @@ -13,12 +13,12 @@ from metagpt.strategy.search_space import SearchSpace class BaseSolver: - """AbstractSolver: 用于定义一个抽象求解器,求解器中的搜索空间是 SearchSpace 实例,图是 ActionGraph 实例。""" + """AbstractSolver: defines the interface of a solver.""" def __init__(self, graph: ActionGraph, search_space: SearchSpace, llm: BaseLLM, context): """ - :param graph: ActionGraph 实例 - :param search_space: SearchSpace 实例 + :param graph: ActionGraph + :param search_space: SearchSpace :param llm: BaseLLM :param context: Context """ @@ -29,11 +29,11 @@ class BaseSolver: @abstractmethod async def solve(self): - """求解器的求解方法。""" + """abstract method to solve the problem.""" class NaiveSolver(BaseSolver): - """NaiveSolver: 直接循序执行给定的 graph""" + """NaiveSolver: Iterate all the nodes in the graph and execute them one by one.""" async def solve(self): self.graph.topological_sort() @@ -43,35 +43,35 @@ class NaiveSolver(BaseSolver): class TOTSolver(BaseSolver): - """TOTSolver: 通过拓扑排序执行给定的 graph""" + """TOTSolver: Tree of Thought""" async def solve(self): raise NotImplementedError class CodeInterpreterSolver(BaseSolver): - """CodeInterpreterSolver: 通过代码解释器执行给定的 graph""" + """CodeInterpreterSolver: Write&Run code in the graph""" async def solve(self): raise NotImplementedError class ReActSolver(BaseSolver): - """ReActSolver: 通过 ReAct 执行给定的 graph""" + """ReActSolver: ReAct algorithm""" async def solve(self): raise NotImplementedError class IOSolver(BaseSolver): - """IOSolver: 通过 IO 执行给定的 graph""" + """IOSolver: use LLM directly to solve the problem""" async def solve(self): raise NotImplementedError class COTSolver(BaseSolver): - """COTSolver: 通过cot执行给定的 graph""" + """COTSolver: Chain of Thought""" async def solve(self): raise NotImplementedError From ad1edf60927e884b2ad0a4a4220abd6fb779a4b0 Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 31 Jan 2024 17:36:04 +0800 Subject: [PATCH 531/637] refine comments --- metagpt/actions/action_graph.py | 17 ++++------------- 1 file changed, 4 insertions(+), 13 deletions(-) diff --git a/metagpt/actions/action_graph.py b/metagpt/actions/action_graph.py index 8570778c7..893bc6d4c 100644 --- a/metagpt/actions/action_graph.py +++ b/metagpt/actions/action_graph.py @@ -11,7 +11,7 @@ from __future__ import annotations class ActionGraph: - """ActionGraph: 用于定义一个图,图中的节点是 ActionNode 实例,节点间的依赖关系是有向边。""" + """ActionGraph: a directed graph to represent the dependency between actions.""" def __init__(self): self.nodes = {} @@ -19,18 +19,11 @@ class ActionGraph: self.execution_order = [] def add_node(self, node): - """ - 添加一个节点到图中。 - :param node: ActionNode 实例 - """ + """Add a node to the graph""" self.nodes[node.key] = node def add_edge(self, from_node: "ActionNode", to_node: "ActionNode"): - """ - 定义节点间的依赖关系。 - :param from_node: 节点标识 - :param to_node: 节点标识 - """ + """Add an edge to the graph""" if from_node.key not in self.edges: self.edges[from_node.key] = [] self.edges[from_node.key].append(to_node.key) @@ -38,9 +31,7 @@ class ActionGraph: to_node.add_prev(from_node) def topological_sort(self): - """ - 实现拓扑排序来确定执行顺序。 - """ + """Topological sort the graph""" visited = set() stack = [] From a1b16b7e99acf6739db283748a697c7f5685c2c3 Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 31 Jan 2024 18:09:01 +0800 Subject: [PATCH 532/637] fix ml_engineer test --- tests/data/rsp_cache.json | 9 ++- tests/metagpt/roles/test_ml_engineer.py | 73 ++++++++++++++++++++++++- tests/mock/mock_llm.py | 5 +- 3 files changed, 84 insertions(+), 3 deletions(-) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index eb67021a5..ac19e9844 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -375,5 +375,12 @@ "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 3:\nEnsure there is enough space around you to avoid any obstacles.\n": "Choice 3: \n\nAnalyzing this choice, ensuring there is enough space around you to avoid any obstacles is a crucial step in maintaining safety and preventing accidents. By creating a buffer zone, you can react to unexpected obstacles or changes in the environment without risking collisions or other hazards.\n\nThe best choice is 3.", "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 2:\nConsider a flat and stable surface, such as a grassy field or a gym mat.\n": "Choice 2: Considering a flat and stable surface, such as a grassy field or a gym mat, is the most promising option. This choice provides a safe and secure environment for the activity, reducing the risk of injury. A grassy field offers a natural and soft surface, while a gym mat provides cushioning and stability. Both options allow for comfortable movement and can accommodate various physical activities.\n\nThe best choice is 2.", "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 4:\nPractice near a wall or with a spotter for safety.\n": "Choice 4: \n\nPractice near a wall or with a spotter for safety.\n\nAnalyzing this choice, practicing near a wall or with a spotter provides a safety measure to prevent accidents or injuries. When practicing a new skill or exercise, having a wall nearby can provide support and stability, reducing the risk of falling or losing balance. Similarly, having a spotter can offer assistance and guidance, ensuring that the practice is done safely and effectively.\n\nThe best choice is 4.", - "### Requirements\n1. Please summarize the latest dialogue based on the reference information (secondary) and dialogue history (primary). Do not include text that is irrelevant to the conversation.\n- The context is for reference only. If it is irrelevant to the user's search request history, please reduce its reference and usage.\n2. If there are citable links in the context, annotate them in the main text in the format [main text](citation link). If there are none in the context, do not write links.\n3. The reply should be graceful, clear, non-repetitive, smoothly written, and of moderate length, in {LANG}.\n\n### Dialogue History (For example)\nA: MLOps competitors\n\n### Current Question (For example)\nA: MLOps competitors\n\n### Current Reply (For example)\n1. Alteryx Designer: etc. if any\n2. Matlab: ditto\n3. IBM SPSS Statistics\n4. RapidMiner Studio\n5. DataRobot AI Platform\n6. Databricks Lakehouse Platform\n7. Amazon SageMaker\n8. Dataiku\n#SYSTEM_MSG_END#\n### Reference Information\nABC cleanser is preferred by many with oily skin.\nL'Oreal is a popular brand with many positive reviews.\n\n### Dialogue History\n\nuser: Which facial cleanser is good for oily skin?\n\n### Current Question\nuser: Which facial cleanser is good for oily skin?\n\n### Current Reply: Based on the information, please write the reply to the Question\n\n\n": "Based on the information provided, ABC cleanser is preferred by many with oily skin. It is a popular choice for individuals with oily skin due to its effectiveness. Additionally, L'Oreal is a well-known brand with many positive reviews, and they offer a range of products suitable for oily skin. Both of these options could be good choices for individuals with oily skin." + "### Requirements\n1. Please summarize the latest dialogue based on the reference information (secondary) and dialogue history (primary). Do not include text that is irrelevant to the conversation.\n- The context is for reference only. If it is irrelevant to the user's search request history, please reduce its reference and usage.\n2. If there are citable links in the context, annotate them in the main text in the format [main text](citation link). If there are none in the context, do not write links.\n3. The reply should be graceful, clear, non-repetitive, smoothly written, and of moderate length, in {LANG}.\n\n### Dialogue History (For example)\nA: MLOps competitors\n\n### Current Question (For example)\nA: MLOps competitors\n\n### Current Reply (For example)\n1. Alteryx Designer: etc. if any\n2. Matlab: ditto\n3. IBM SPSS Statistics\n4. RapidMiner Studio\n5. DataRobot AI Platform\n6. Databricks Lakehouse Platform\n7. Amazon SageMaker\n8. Dataiku\n#SYSTEM_MSG_END#\n### Reference Information\nABC cleanser is preferred by many with oily skin.\nL'Oreal is a popular brand with many positive reviews.\n\n### Dialogue History\n\nuser: Which facial cleanser is good for oily skin?\n\n### Current Question\nuser: Which facial cleanser is good for oily skin?\n\n### Current Reply: Based on the information, please write the reply to the Question\n\n\n": "Based on the information provided, ABC cleanser is preferred by many with oily skin. It is a popular choice for individuals with oily skin due to its effectiveness. Additionally, L'Oreal is a well-known brand with many positive reviews, and they offer a range of products suitable for oily skin. Both of these options could be good choices for individuals with oily skin.", + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\n\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import get_column_info\n\ncolumn_info = get_column_info(df)\nprint(\"column_info\")\nprint(column_info)" + }, + "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nSolve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\n[user: Tested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { + "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not correctly sort the array based on the number of ones in their binary representation in ascending order. The function `count_ones` correctly counts the number of ones in the binary representation of each number, but the sorting does not handle the case where two numbers have the same number of ones but different decimal values. To fix this issue, we need to ensure that the sorting is stable when the number of ones is the same, so that numbers are then sorted by their decimal value. We can use the `sorted` function with two keys, first sorting by the decimal value, and then by the number of ones, to ensure stability.", + "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(n).count('1')\n \n # First, sort the array by the decimal values to ensure stability\n arr.sort()\n # Then, sort the array using a custom key function\n # The key function returns the number of ones for each element\n # Since the previous sort was stable, elements with the same number of ones will remain sorted by their value\n sorted_arr = sorted(arr, key=count_ones)\n \n return sorted_arr\n" + } } \ No newline at end of file diff --git a/tests/metagpt/roles/test_ml_engineer.py b/tests/metagpt/roles/test_ml_engineer.py index 1373213a5..bc1626251 100644 --- a/tests/metagpt/roles/test_ml_engineer.py +++ b/tests/metagpt/roles/test_ml_engineer.py @@ -1,7 +1,11 @@ import pytest +from metagpt.actions.execute_code import ExecutePyCode from metagpt.logs import logger from metagpt.roles.ml_engineer import MLEngineer +from metagpt.schema import Message, Plan, Task +from metagpt.tools.tool_data_type import ToolTypeEnum +from tests.metagpt.actions.test_debug_code import CODE, DebugContext, ErrorStr def test_mle_init(): @@ -9,13 +13,80 @@ def test_mle_init(): assert ci.tools == [] +MockPlan = Plan( + goal="This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.", + context="", + tasks=[ + Task( + task_id="1", + dependent_task_ids=[], + instruction="Perform exploratory data analysis on the train dataset to understand the features and target variable.", + task_type="eda", + code_steps="", + code="", + result="", + is_success=False, + is_finished=False, + ) + ], + task_map={ + "1": Task( + task_id="1", + dependent_task_ids=[], + instruction="Perform exploratory data analysis on the train dataset to understand the features and target variable.", + task_type="eda", + code_steps="", + code="", + result="", + is_success=False, + is_finished=False, + ) + }, + current_task_id="1", +) + + +@pytest.mark.asyncio +async def test_mle_write_code(mocker): + data_path = "tests/data/ml_datasets/titanic" + + mle = MLEngineer(auto_run=True, use_tools=True) + mle.planner.plan = MockPlan + + code, _ = await mle._write_code() + assert data_path in code["code"] + + +@pytest.mark.asyncio +async def test_mle_update_data_columns(mocker): + mle = MLEngineer(auto_run=True, use_tools=True) + mle.planner.plan = MockPlan + + # manually update task type to test update + mle.planner.plan.current_task.task_type = ToolTypeEnum.DATA_PREPROCESS.value + + result = await mle._update_data_columns() + assert result is not None + + +@pytest.mark.asyncio +async def test_mle_debug_code(mocker): + mle = MLEngineer(auto_run=True, use_tools=True) + mle.working_memory.add(Message(content=ErrorStr, cause_by=ExecutePyCode)) + mle.latest_code = CODE + mle.debug_context = DebugContext + code, _ = await mle._write_code() + assert len(code) > 0 + + +@pytest.mark.skip @pytest.mark.asyncio async def test_ml_engineer(): data_path = "tests/data/ml_datasets/titanic" requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." tools = ["FillMissingValue", "CatCross", "dummy_tool"] - mle = MLEngineer(goal=requirement, auto_run=True, use_tools=True, tools=tools) + mle = MLEngineer(auto_run=True, use_tools=True, tools=tools) rsp = await mle.run(requirement) logger.info(rsp) assert len(rsp.content) > 0 diff --git a/tests/mock/mock_llm.py b/tests/mock/mock_llm.py index e2fff214f..8ee580b8a 100644 --- a/tests/mock/mock_llm.py +++ b/tests/mock/mock_llm.py @@ -13,7 +13,10 @@ OriginalLLM = OpenAILLM if config.llm.api_type == LLMType.OPENAI else AzureOpenA class MockLLM(OriginalLLM): def __init__(self, allow_open_api_call): - super().__init__(config.get_openai_llm()) + original_llm_config = ( + config.get_openai_llm() if config.llm.api_type == LLMType.OPENAI else config.get_azure_llm() + ) + super().__init__(original_llm_config) self.allow_open_api_call = allow_open_api_call self.rsp_cache: dict = {} self.rsp_candidates: list[dict] = [] # a test can have multiple calls with the same llm, thus a list From 487169ee6137393bb1b791cbba9925a4bab7d427 Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 31 Jan 2024 18:27:12 +0800 Subject: [PATCH 533/637] rm mle_simple for now --- metagpt/roles/ml_engineer_simple.py | 135 ---------------------------- 1 file changed, 135 deletions(-) delete mode 100644 metagpt/roles/ml_engineer_simple.py diff --git a/metagpt/roles/ml_engineer_simple.py b/metagpt/roles/ml_engineer_simple.py deleted file mode 100644 index 9ff1c9880..000000000 --- a/metagpt/roles/ml_engineer_simple.py +++ /dev/null @@ -1,135 +0,0 @@ -import re -from datetime import datetime -from typing import List - -import fire - -from metagpt.actions.ask_review import AskReview, ReviewConst -from metagpt.actions.execute_code import ExecutePyCode -from metagpt.actions.write_analysis_code import WriteCodeByGenerate -from metagpt.logs import logger -from metagpt.memory import Memory -from metagpt.roles import Role -from metagpt.roles.kaggle_manager import DownloadData -from metagpt.schema import Message -from metagpt.utils.save_code import save_code_file - -STRUCTURAL_CONTEXT_SIMPLE = """ -## User Requirement -{user_requirement} -## Data Description -{data_desc} -""" - -JUDGE_PROMPT_TEMPLATE = """ -# User Requirement -{user_requirement} ------ -# Context -{context} ------ -# State -Output "Ture" or "False". Judging from the code perspective, whether the user's needs have been completely fulfilled. -===== -# Output State("Ture" or "False") firstly, then output Thought and Next Steps for the code requirements based on the context respectively in one sentence -State: -Thought: -Next Steps: -""" - - -class MLEngineerSimple(Role): - def __init__(self, name="ABC", profile="MLEngineerSimple", goal="", auto_run: bool = False): - super().__init__(name=name, profile=profile, goal=goal) - self._set_react_mode(react_mode="react") - self._watch([DownloadData]) - self._init_actions([WriteCodeByGenerate, ExecutePyCode]) - - self.goal = goal - self.data_desc = "" - self.use_tools = False - self.use_code_steps = False - self.execute_code = ExecutePyCode() - self.auto_run = auto_run - - # memory for working on each task, discarded each time a task is done - self.working_memory = Memory() - - async def _act(self): - memories = self.get_memories() - if memories: - latest_event = memories[-1].cause_by - if latest_event == DownloadData: - self.data_desc = memories[-1].content - - await self._act_no_plan() - - # save code using datetime.now or keywords related to the goal of your project (plan.goal). - project_record = datetime.now().strftime("%Y-%m-%d_%H-%M-%S") - save_code_file(name=project_record, code_context=self.execute_code.nb, file_format="ipynb") - - async def _act_no_plan(self, max_retry: int = 20): - counter = 0 - state = False - while not state and counter < max_retry: - context = self.get_useful_memories() - print(f"memories数量:{len(context)}") - # print("===\n" +str(context) + "\n===") - code = await WriteCodeByGenerate().run(context=context, temperature=0.0, only_code=True) - cause_by = WriteCodeByGenerate - self.working_memory.add(Message(content=code, role="assistant", cause_by=cause_by)) - - result, success = await self.execute_code.run(code) - print(result) - self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) - - if "!pip" in code: - success = False - - counter += 1 - - if not success and counter >= max_retry: - logger.info("coding failed!") - review, _ = await self._ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) - if ReviewConst.CHANGE_WORD[0] in review: - counter = 0 # redo the task again with help of human suggestions - - completed_plan_memory = self.get_useful_memories() # completed plan as a outcome - self.rc.memory.add(completed_plan_memory[0]) # add to persistent memory - prompt = JUDGE_PROMPT_TEMPLATE.format(user_requirement=self.goal, context=completed_plan_memory) - rsp = await self._llm.aask(prompt) - self.working_memory.add(Message(content=rsp, role="system")) - - matches = re.findall(r"\b(True|False)\b", rsp) - state = False if "False" in matches else True - - async def _ask_review(self, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER): - auto_run = auto_run or self.auto_run - if not auto_run: - context = self.get_useful_memories() - review, confirmed = await AskReview().run(context=context[-5:], trigger=trigger) - if not confirmed: - self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) - return review, confirmed - return "", True - - def get_useful_memories(self) -> List[Message]: - """find useful memories only to reduce context length and improve performance""" - user_requirement = self.goal - context = STRUCTURAL_CONTEXT_SIMPLE.format(user_requirement=user_requirement, data_desc=self.data_desc) - context_msg = [Message(content=context, role="user")] - - return context_msg + self.get_working_memories(6) - - def get_working_memories(self, num=0) -> List[Message]: - return self.working_memory.get(num) # 默认为6 - - -if __name__ == "__main__": - requirement = "Run data analysis on sklearn Iris dataset, include a plot" - - async def main(requirement: str = requirement, auto_run: bool = True): - role = MLEngineerSimple(goal=requirement, auto_run=auto_run) - await role.run(requirement) - - fire.Fire(main) From b9663cebbd1884099de2b48540dd742918ed9788 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 31 Jan 2024 21:00:02 +0800 Subject: [PATCH 534/637] fix parse_code bug. --- metagpt/utils/common.py | 25 ++++++++++++++----------- 1 file changed, 14 insertions(+), 11 deletions(-) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 543c627a3..ec20223b8 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -265,19 +265,22 @@ class CodeParser: return block_dict @classmethod - def parse_code(cls, block: str, text: str, lang: str = "", start_ends: str = r'["\'`]{3}') -> str: + def parse_code(cls, block: str, text: str, lang: str = "") -> str: if block: text = cls.parse_block(block, text) - pattern = rf"{start_ends}{lang}.*?\s+(.*?){start_ends}" - match = re.search(pattern, text, re.DOTALL) - if match: - code = match.group(1) - else: - logger.error(f"{pattern} not match following text:") - logger.error(text) - # raise Exception - return text # just assume original text is code - return code + start_ends = ["```", '"""', "'''"] + patterns = [] + for start_end in start_ends: + pattern = rf"{start_end}{lang}.*?\s+(.*?){start_end}" + match = re.search(pattern, text, re.DOTALL) + if match: + code = match.group(1) + return code + patterns.append(pattern) + logger.error(f"{patterns} not match following text:") + logger.error(text) + # raise Exception + return text # just assume original text is code @classmethod def parse_str(cls, block: str, text: str, lang: str = ""): From 15e72ca51db617b5b0d7b4d263a0bd4fb9a6cbb7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Wed, 31 Jan 2024 21:07:40 +0800 Subject: [PATCH 535/637] chore. --- metagpt/utils/common.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index ec20223b8..7d3d47680 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -268,7 +268,7 @@ class CodeParser: def parse_code(cls, block: str, text: str, lang: str = "") -> str: if block: text = cls.parse_block(block, text) - start_ends = ["```", '"""', "'''"] + start_ends = ["```", "'''", '"""'] patterns = [] for start_end in start_ends: pattern = rf"{start_end}{lang}.*?\s+(.*?){start_end}" From 6656ebf4c41d06418916b58f23a16e919cb22527 Mon Sep 17 00:00:00 2001 From: yzlin Date: Wed, 31 Jan 2024 21:40:17 +0800 Subject: [PATCH 536/637] add ask_review, write plan, ci test --- tests/data/rsp_cache.json | 19 ++++++++++++++++++- tests/metagpt/actions/test_ask_review.py | 12 ++++++++++++ tests/metagpt/actions/test_write_plan.py | 7 +++++-- tests/metagpt/roles/test_code_interpreter.py | 8 ++++++-- 4 files changed, 41 insertions(+), 5 deletions(-) create mode 100644 tests/metagpt/actions/test_ask_review.py diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index b1f083185..e5f03d9cb 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -417,5 +417,22 @@ "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"issue_type\": \"BUG\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- issue_type: # Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"issue_type\": \"REQUIREMENT\"\n}\n[/CONTENT]", "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Product Goals\": [\n \"Create a challenging and addictive gameplay\",\n \"Enhance accessibility and responsiveness for all users\",\n \"Implement visually appealing UI design\"\n ]\n}\n[/CONTENT]", "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ]\n}\n[/CONTENT]", - "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code should handle user input and update the game board accordingly.\"\n ],\n [\n \"P0\",\n \"The game algorithm should handle the merging of tiles and the generation of new tiles with values of 2 or 4.\"\n ],\n [\n \"P1\",\n \"The game should end when the player achieves the 2048 tile or when there are no possible moves left.\"\n ],\n [\n \"P1\",\n \"The game should display the current score and the highest tile achieved by the player.\"\n ],\n [\n \"P2\",\n \"The game should have a smooth and visually appealing user interface.\"\n ]\n ]\n}\n[/CONTENT]" + "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code should handle user input and update the game board accordingly.\"\n ],\n [\n \"P0\",\n \"The game algorithm should handle the merging of tiles and the generation of new tiles with values of 2 or 4.\"\n ],\n [\n \"P1\",\n \"The game should end when the player achieves the 2048 tile or when there are no possible moves left.\"\n ],\n [\n \"P1\",\n \"The game should display the current score and the highest tile achieved by the player.\"\n ],\n [\n \"P2\",\n \"The game should have a smooth and visually appealing user interface.\"\n ]\n ]\n}\n[/CONTENT]", + "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Import the Iris dataset from scikit-learn.\\nTask 2: Perform exploratory data analysis to understand the dataset.\\nTask 3: Preprocess the data if necessary (e.g., scaling, encoding).\\nTask 4: Split the dataset into training and testing sets.\\nTask 5: Choose a suitable model and train it on the dataset.\\nTask 6: Evaluate the model's performance on the test set.\\nTask 7: Report the results of the analysis.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { + "task_type": [ + "other", + "eda", + "data_preprocess", + "data_preprocess", + "model_train", + "model_evaluate", + "other" + ] + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "import pandas as pd\n\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\niris_df['target'] = iris_data.target\n\n# Display basic information about the dataset\niris_df_info = iris_df.info()\n\n# Display statistical summary of the dataset\niris_df_description = iris_df.describe()\n\n# Show the first few rows of the dataset\ndf_head = iris_df.head()\n\n# Output the results\ndisplay(iris_df_info, iris_df_description, df_head)" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\n\\\\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\\\\niris_df['target'] = iris_data.target\\\\n\\\\n# Display basic information about the dataset\\\\niris_df_info = iris_df.info()\\\\n\\\\n# Display statistical summary of the dataset\\\\niris_df_description = iris_df.describe()\\\\n\\\\n# Show the first few rows of the dataset\\\\ndf_head = iris_df.head()\\\\n\\\\n# Output the results\\\\ndisplay(iris_df_info, iris_df_description, df_head)\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset features.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "import seaborn as sns\nimport matplotlib.pyplot as plt\n\n# Pairplot to visualize the relationships between features\nsns.pairplot(iris_df, hue='target', diag_kind='kde')\n\n# Show the plot\nplt.show()" + } } \ No newline at end of file diff --git a/tests/metagpt/actions/test_ask_review.py b/tests/metagpt/actions/test_ask_review.py new file mode 100644 index 000000000..00001fad6 --- /dev/null +++ b/tests/metagpt/actions/test_ask_review.py @@ -0,0 +1,12 @@ +import pytest + +from metagpt.actions.ask_review import AskReview + + +@pytest.mark.asyncio +async def test_ask_review(mocker): + mock_review_input = "confirm" + mocker.patch("builtins.input", return_value=mock_review_input) + rsp, confirmed = await AskReview().run() + assert rsp == mock_review_input + assert confirmed diff --git a/tests/metagpt/actions/test_write_plan.py b/tests/metagpt/actions/test_write_plan.py index 9abc6c798..f36527711 100644 --- a/tests/metagpt/actions/test_write_plan.py +++ b/tests/metagpt/actions/test_write_plan.py @@ -23,8 +23,11 @@ def test_precheck_update_plan_from_rsp(): @pytest.mark.asyncio -async def test_write_plan(): - rsp = await WritePlan().run(context=[Message("run analysis on sklearn iris dataset", role="user")]) +@pytest.mark.parametrize("use_tools", [(False), (True)]) +async def test_write_plan(use_tools): + rsp = await WritePlan().run( + context=[Message("run analysis on sklearn iris dataset", role="user")], use_tools=use_tools + ) assert "task_id" in rsp assert "instruction" in rsp diff --git a/tests/metagpt/roles/test_code_interpreter.py b/tests/metagpt/roles/test_code_interpreter.py index b78f7a9ef..dd959525e 100644 --- a/tests/metagpt/roles/test_code_interpreter.py +++ b/tests/metagpt/roles/test_code_interpreter.py @@ -5,11 +5,15 @@ from metagpt.roles.code_interpreter import CodeInterpreter @pytest.mark.asyncio -async def test_code_interpreter(): +@pytest.mark.parametrize("auto_run", [(True), (False)]) +async def test_code_interpreter(mocker, auto_run): + mocker.patch("metagpt.actions.execute_code.ExecutePyCode.run", return_value=("a successful run", True)) + mocker.patch("builtins.input", return_value="confirm") + requirement = "Run data analysis on sklearn Iris dataset, include a plot" tools = [] - ci = CodeInterpreter(auto_run=True, use_tools=True, tools=tools) + ci = CodeInterpreter(auto_run=auto_run, use_tools=True, tools=tools) rsp = await ci.run(requirement) logger.info(rsp) assert len(rsp.content) > 0 From f4c6e507c9d21ea2375a87f12d83a885838e2e11 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E8=8E=98=E6=9D=83=20=E9=A9=AC?= Date: Wed, 31 Jan 2024 22:44:01 +0800 Subject: [PATCH 537/637] fixbug: make unit test stable --- tests/metagpt/tools/test_ut_writer.py | 49 ++++++++++++++++++++++++++- 1 file changed, 48 insertions(+), 1 deletion(-) diff --git a/tests/metagpt/tools/test_ut_writer.py b/tests/metagpt/tools/test_ut_writer.py index 29b6572c2..3cc7e86bb 100644 --- a/tests/metagpt/tools/test_ut_writer.py +++ b/tests/metagpt/tools/test_ut_writer.py @@ -8,6 +8,17 @@ from pathlib import Path import pytest +from openai.resources.chat.completions import AsyncCompletions +from openai.types import CompletionUsage +from openai.types.chat.chat_completion import ( + ChatCompletion, + ChatCompletionMessage, + Choice, +) +from openai.types.chat.chat_completion_message_tool_call import ( + ChatCompletionMessageToolCall, + Function, +) from metagpt.config2 import config from metagpt.const import API_QUESTIONS_PATH, UT_PY_PATH @@ -16,7 +27,43 @@ from metagpt.tools.ut_writer import YFT_PROMPT_PREFIX, UTGenerator class TestUTWriter: @pytest.mark.asyncio - async def test_api_to_ut_sample(self): + async def test_api_to_ut_sample(self, mocker): + async def mock_create(*args, **kwargs): + return ChatCompletion( + id="chatcmpl-8n5fAd21w2J1IIFkI4qxWlNfM7QRC", + choices=[ + Choice( + finish_reason="stop", + index=0, + logprobs=None, + message=ChatCompletionMessage( + content=None, + role="assistant", + function_call=None, + tool_calls=[ + ChatCompletionMessageToolCall( + id="call_EjjmIY7GMspHu3r9mx8gPA2k", + function=Function( + arguments='{"code":"import string\\nimport random\\n\\ndef random_string' + "(length=10):\\n return ''.join(random.choice(string.ascii_" + 'lowercase) for i in range(length))"}', + name="execute", + ), + type="function", + ) + ], + ), + ) + ], + created=1706710532, + model="gpt-3.5-turbo-1106", + object="chat.completion", + system_fingerprint="fp_04f9a1eebf", + usage=CompletionUsage(completion_tokens=35, prompt_tokens=1982, total_tokens=2017), + ) + + mocker.patch.object(AsyncCompletions, "create", mock_create) + # Prerequisites swagger_file = Path(__file__).parent / "../../data/ut_writer/yft_swaggerApi.json" assert swagger_file.exists() From a762e020008a8e01a80f8ee7e196116444330156 Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 31 Jan 2024 23:13:38 +0800 Subject: [PATCH 538/637] update config usage --- Dockerfile | 2 +- README.md | 14 ++++----- config/config2.yaml.example | 3 -- docs/FAQ-EN.md | 30 +++---------------- docs/README_CN.md | 10 +++---- docs/README_JA.md | 50 +++++++++++++++---------------- docs/install/cli_install.md | 25 +++++++++------- docs/install/cli_install_cn.md | 4 +-- docs/install/docker_install.md | 12 ++++---- docs/install/docker_install_cn.md | 12 ++++---- docs/tutorial/usage.md | 11 ++----- docs/tutorial/usage_cn.md | 11 ++----- metagpt/actions/design_api.py | 2 +- metagpt/actions/write_prd.py | 2 +- metagpt/config2.py | 8 +---- metagpt/configs/llm_config.py | 2 +- metagpt/configs/mermaid_config.py | 5 ++-- metagpt/utils/mermaid.py | 12 ++++---- metagpt/utils/mmdc_pyppeteer.py | 6 ++-- metagpt/utils/yaml_model.py | 4 +-- 20 files changed, 93 insertions(+), 132 deletions(-) diff --git a/Dockerfile b/Dockerfile index 9eeacbccb..dead20537 100644 --- a/Dockerfile +++ b/Dockerfile @@ -8,7 +8,7 @@ RUN apt update &&\ # Install Mermaid CLI globally ENV CHROME_BIN="/usr/bin/chromium" \ - PUPPETEER_CONFIG="/app/metagpt/config/puppeteer-config.json"\ + puppeteer_config="/app/metagpt/config/puppeteer-config.json"\ PUPPETEER_SKIP_CHROMIUM_DOWNLOAD="true" RUN npm install -g @mermaid-js/mermaid-cli &&\ npm cache clean --force diff --git a/README.md b/README.md index 90c586068..39dde8208 100644 --- a/README.md +++ b/README.md @@ -67,10 +67,10 @@ # Step 2: Clone the repository to your local machine for latest version, and ins cd MetaGPT pip3 install -e . # or pip3 install metagpt # for stable version -# Step 3: setup your OPENAI_API_KEY, or make sure it existed in the env +# Step 3: setup your LLM key in the config2.yaml file mkdir ~/.metagpt -cp config/config.yaml ~/.metagpt/config.yaml -vim ~/.metagpt/config.yaml +cp config/config2.yaml ~/.metagpt/config2.yaml +vim ~/.metagpt/config2.yaml # Step 4: run metagpt cli metagpt "Create a 2048 game in python" @@ -87,16 +87,16 @@ ### Docker installation > Note: In the Windows, you need to replace "/opt/metagpt" with a directory that Docker has permission to create, such as "D:\Users\x\metagpt" ```bash -# Step 1: Download metagpt official image and prepare config.yaml +# Step 1: Download metagpt official image and prepare config2.yaml docker pull metagpt/metagpt:latest mkdir -p /opt/metagpt/{config,workspace} -docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config.yaml > /opt/metagpt/config/key.yaml -vim /opt/metagpt/config/key.yaml # Change the config +docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config2.yaml > /opt/metagpt/config/config2.yaml +vim /opt/metagpt/config/config2.yaml # Change the config # Step 2: Run metagpt demo with container docker run --rm \ --privileged \ - -v /opt/metagpt/config/key.yaml:/app/metagpt/config/key.yaml \ + -v /opt/metagpt/config/config2.yaml:/app/metagpt/config/config2.yaml \ -v /opt/metagpt/workspace:/app/metagpt/workspace \ metagpt/metagpt:latest \ metagpt "Write a cli snake game" diff --git a/config/config2.yaml.example b/config/config2.yaml.example index 1a406e756..7c523fe7d 100644 --- a/config/config2.yaml.example +++ b/config/config2.yaml.example @@ -37,6 +37,3 @@ IFLYTEK_API_KEY: "YOUR_API_KEY" IFLYTEK_API_SECRET: "YOUR_API_SECRET" METAGPT_TEXT_TO_IMAGE_MODEL_URL: "YOUR_MODEL_URL" - -PYPPETEER_EXECUTABLE_PATH: "/Applications/Google Chrome.app" - diff --git a/docs/FAQ-EN.md b/docs/FAQ-EN.md index d4a9f6097..88b5b0573 100644 --- a/docs/FAQ-EN.md +++ b/docs/FAQ-EN.md @@ -83,10 +83,10 @@ 1. PRD stuck / unable to access/ connection interrupted - 1. The official OPENAI_BASE_URL address is `https://api.openai.com/v1` - 1. If the official OPENAI_BASE_URL address is inaccessible in your environment (this can be verified with curl), it's recommended to configure using the reverse proxy OPENAI_BASE_URL provided by libraries such as openai-forward. For instance, `OPENAI_BASE_URL: "``https://api.openai-forward.com/v1``"` - 1. If the official OPENAI_BASE_URL address is inaccessible in your environment (again, verifiable via curl), another option is to configure the OPENAI_PROXY parameter. This way, you can access the official OPENAI_BASE_URL via a local proxy. If you don't need to access via a proxy, please do not enable this configuration; if accessing through a proxy is required, modify it to the correct proxy address. Note that when OPENAI_PROXY is enabled, don't set OPENAI_BASE_URL. - 1. Note: OpenAI's default API design ends with a v1. An example of the correct configuration is: `OPENAI_BASE_URL: "``https://api.openai.com/v1``"` + 1. The official openai base_url address is `https://api.openai.com/v1` + 1. If the official openai base_url address is inaccessible in your environment (this can be verified with curl), it's recommended to configure using the reverse proxy openai base_url provided by libraries such as openai-forward. For instance, `openai base_url: "``https://api.openai-forward.com/v1``"` + 1. If the official openai base_url address is inaccessible in your environment (again, verifiable via curl), another option is to configure the llm.proxy parameter. This way, you can access the official openai base_url via a local proxy. If you don't need to access via a proxy, please do not enable this configuration; if accessing through a proxy is required, modify it to the correct proxy address. Note that when llm.proxy is enabled, don't set openai base_url. + 1. Note: OpenAI's default API design ends with a v1. An example of the correct configuration is: `openai base_url: "``https://api.openai.com/v1``"` 1. Absolutely! How can I assist you today? @@ -119,28 +119,6 @@ 1. When using a database, it often gets the implementation wrong — since the SQL database initialization process is usually not in the code. 1. With more lines of code, there's a higher chance of false impressions, leading to calls to non-existent APIs. -1. Instructions for using SD Skills/UI Role: - - 1. Currently, there is a test script located in /tests/metagpt/roles. The file ui_role provides the corresponding code implementation. For testing, you can refer to the test_ui in the same directory. - - 1. The UI role takes over from the product manager role, extending the output from the 【UI Design draft】 provided by the product manager role. The UI role has implemented the UIDesign Action. Within the run of UIDesign, it processes the respective context, and based on the set template, outputs the UI. The output from the UI role includes: - - 1. UI Design Description: Describes the content to be designed and the design objectives. - 1. Selected Elements: Describes the elements in the design that need to be illustrated. - 1. HTML Layout: Outputs the HTML code for the page. - 1. CSS Styles (styles.css): Outputs the CSS code for the page. - - 1. Currently, the SD skill is a tool invoked by UIDesign. It instantiates the SDEngine, with specific code found in metagpt/tools/sd_engine. - - 1. Configuration instructions for SD Skills: The SD interface is currently deployed based on *https://github.com/AUTOMATIC1111/stable-diffusion-webui* **For environmental configurations and model downloads, please refer to the aforementioned GitHub repository. To initiate the SD service that supports API calls, run the command specified in cmd with the parameter nowebui, i.e., - - 1. > python3 webui.py --enable-insecure-extension-access --port xxx --no-gradio-queue --nowebui - 1.     Once it runs without errors, the interface will be accessible after approximately 1 minute when the model finishes loading. - 1. Configure SD_URL and SD_T2I_API in the config.yaml/key.yaml files. - 1. ![](https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/065295a67b0b4feea665d1372722d49d~tplv-k3u1fbpfcp-zoom-1.image) - 1.     SD_URL is the deployed server/machine IP, and Port is the specified port above, defaulting to 7860. - 1. > SD_URL: IP:Port - 1. An error occurred during installation: "Another program is using this file...egg". 1. Delete the file and try again. diff --git a/docs/README_CN.md b/docs/README_CN.md index 2855b5500..ebf5dd408 100644 --- a/docs/README_CN.md +++ b/docs/README_CN.md @@ -48,7 +48,7 @@ # 第 2 步:克隆最新仓库到您的本地机器,并进行安装。 pip3 install -e. # 或者 pip3 install metagpt # 安装稳定版本 # 第 3 步:执行metagpt -# 拷贝config.yaml为key.yaml,并设置你自己的OPENAI_API_KEY +# 拷贝config2.yaml为~/.metagpt/config2.yaml,并设置你自己的api_key metagpt "Write a cli snake game" # 第 4 步【可选的】:如果你想在执行过程中保存像象限图、系统设计、序列流程等图表这些产物,可以在第3步前执行该步骤。默认的,框架做了兼容,在不执行该步的情况下,也可以完整跑完整个流程。 @@ -63,16 +63,16 @@ ### Docker安装 > 注意:在Windows中,你需要将 "/opt/metagpt" 替换为Docker具有创建权限的目录,比如"D:\Users\x\metagpt" ```bash -# 步骤1: 下载metagpt官方镜像并准备好config.yaml +# 步骤1: 下载metagpt官方镜像并准备好config2.yaml docker pull metagpt/metagpt:latest mkdir -p /opt/metagpt/{config,workspace} -docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config.yaml > /opt/metagpt/config/key.yaml -vim /opt/metagpt/config/key.yaml # 修改配置文件 +docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config2.yaml > /opt/metagpt/config/config2.yaml +vim /opt/metagpt/config/config2.yaml # 修改配置文件 # 步骤2: 使用容器运行metagpt演示 docker run --rm \ --privileged \ - -v /opt/metagpt/config/key.yaml:/app/metagpt/config/key.yaml \ + -v /opt/metagpt/config/config2.yaml:/app/metagpt/config/config2.yaml \ -v /opt/metagpt/workspace:/app/metagpt/workspace \ metagpt/metagpt:latest \ metagpt "Write a cli snake game" diff --git a/docs/README_JA.md b/docs/README_JA.md index 8b2bf1fae..26db0498f 100644 --- a/docs/README_JA.md +++ b/docs/README_JA.md @@ -68,7 +68,7 @@ # ステップ 2: リポジトリをローカルマシンにクローンし、 pip install -e. # ステップ 3: metagpt を実行する -# config.yaml を key.yaml にコピーし、独自の OPENAI_API_KEY を設定します +# config/config2.yaml を ~/.metagpt/config2.yaml にコピーし、独自の api_key を設定します metagpt "Write a cli snake game" # ステップ 4 [オプション]: 実行中に PRD ファイルなどのアーティファクトを保存する場合は、ステップ 3 の前にこのステップを実行できます。デフォルトでは、フレームワークには互換性があり、この手順を実行しなくてもプロセス全体を完了できます。 @@ -91,8 +91,8 @@ # NPM がシステムにインストールされていることを確認して - config.yml に mmdc のコンフィグを記述するのを忘れないこと ```yml - PUPPETEER_CONFIG: "./config/puppeteer-config.json" - MMDC: "./node_modules/.bin/mmdc" + puppeteer_config: "./config/puppeteer-config.json" + path: "./node_modules/.bin/mmdc" ``` - もし `pip install -e.` がエラー `[Errno 13] Permission denied: '/usr/local/lib/python3.11/dist-packages/test-easy-install-13129.write-test'` で失敗したら、代わりに `pip install -e. --user` を実行してみてください @@ -114,12 +114,13 @@ # NPM がシステムにインストールされていることを確認して playwright install --with-deps chromium ``` - - **modify `config.yaml`** + - **modify `config2.yaml`** - config.yaml から MERMAID_ENGINE のコメントを外し、`playwright` に変更する + config2.yaml から mermaid.engine のコメントを外し、`playwright` に変更する ```yaml - MERMAID_ENGINE: playwright + mermaid: + engine: playwright ``` - pyppeteer @@ -143,21 +144,23 @@ # NPM がシステムにインストールされていることを確認して pyppeteer-install ``` - - **`config.yaml` を修正** + - **`config2.yaml` を修正** - config.yaml から MERMAID_ENGINE のコメントを外し、`pyppeteer` に変更する + config2.yaml から mermaid.engine のコメントを外し、`pyppeteer` に変更する ```yaml - MERMAID_ENGINE: pyppeteer + mermaid: + engine: pyppeteer ``` - mermaid.ink - - **`config.yaml` を修正** + - **`config2.yaml` を修正** - config.yaml から MERMAID_ENGINE のコメントを外し、`ink` に変更する + config2.yaml から mermaid.engine のコメントを外し、`ink` に変更する ```yaml - MERMAID_ENGINE: ink + mermaid: + engine: ink ``` 注: この方法は pdf エクスポートに対応していません。 @@ -166,16 +169,16 @@ ### Docker によるインストール > Windowsでは、"/opt/metagpt"をDockerが作成する権限を持つディレクトリに置き換える必要があります。例えば、"D:\Users\x\metagpt"などです。 ```bash -# ステップ 1: metagpt 公式イメージをダウンロードし、config.yaml を準備する +# ステップ 1: metagpt 公式イメージをダウンロードし、config2.yaml を準備する docker pull metagpt/metagpt:latest mkdir -p /opt/metagpt/{config,workspace} -docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config.yaml > /opt/metagpt/config/key.yaml -vim /opt/metagpt/config/key.yaml # 設定を変更する +docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config2.yaml > /opt/metagpt/config/config2.yaml +vim /opt/metagpt/config/config2.yaml # 設定を変更する # ステップ 2: コンテナで metagpt デモを実行する docker run --rm \ --privileged \ - -v /opt/metagpt/config/key.yaml:/app/metagpt/config/key.yaml \ + -v /opt/metagpt/config/config2.yaml:/app/metagpt/config/config2.yaml \ -v /opt/metagpt/workspace:/app/metagpt/workspace \ metagpt/metagpt:latest \ metagpt "Write a cli snake game" @@ -183,7 +186,7 @@ # ステップ 2: コンテナで metagpt デモを実行する # コンテナを起動し、その中でコマンドを実行することもできます docker run --name metagpt -d \ --privileged \ - -v /opt/metagpt/config/key.yaml:/app/metagpt/config/key.yaml \ + -v /opt/metagpt/config/config2.yaml:/app/metagpt/config/config2.yaml \ -v /opt/metagpt/workspace:/app/metagpt/workspace \ metagpt/metagpt:latest @@ -194,7 +197,7 @@ # コンテナを起動し、その中でコマンドを実行することもで コマンド `docker run ...` は以下のことを行います: - 特権モードで実行し、ブラウザの実行権限を得る -- ホスト設定ファイル `/opt/metagpt/config/key.yaml` をコンテナ `/app/metagpt/config/key.yaml` にマップします +- ホスト設定ファイル `/opt/metagpt/config/config2.yaml` をコンテナ `/app/metagpt/config/config2.yaml` にマップします - ホストディレクトリ `/opt/metagpt/workspace` をコンテナディレクトリ `/app/metagpt/workspace` にマップするs - デモコマンド `metagpt "Write a cli snake game"` を実行する @@ -208,19 +211,14 @@ # また、自分で metagpt イメージを構築することもできます。 ## 設定 -- `OPENAI_API_KEY` を `config/key.yaml / config/config.yaml / env` のいずれかで設定します。 -- 優先順位は: `config/key.yaml > config/config.yaml > env` の順です。 +- `api_key` を `~/.metagpt/config2.yaml / config/config2.yaml` のいずれかで設定します。 +- 優先順位は: `~/.metagpt/config2.yaml > config/config2.yaml > env` の順です。 ```bash # 設定ファイルをコピーし、必要な修正を加える。 -cp config/config.yaml config/key.yaml +cp config/config2.yaml ~/.metagpt/config2.yaml ``` -| 変数名 | config/key.yaml | env | -| --------------------------------------- | ----------------------------------------- | ----------------------------------------------- | -| OPENAI_API_KEY # 自分のキーに置き換える | OPENAI_API_KEY: "sk-..." | export OPENAI_API_KEY="sk-..." | -| OPENAI_BASE_URL # オプション | OPENAI_BASE_URL: "https:///v1" | export OPENAI_BASE_URL="https:///v1" | - ## チュートリアル: スタートアップの開始 ```shell diff --git a/docs/install/cli_install.md b/docs/install/cli_install.md index 80deda771..33d759758 100644 --- a/docs/install/cli_install.md +++ b/docs/install/cli_install.md @@ -36,8 +36,8 @@ # Step 3: Clone the repository to your local machine, and install it. - don't forget to the configuration for mmdc in config.yml ```yml - PUPPETEER_CONFIG: "./config/puppeteer-config.json" - MMDC: "./node_modules/.bin/mmdc" + puppeteer_config: "./config/puppeteer-config.json" + path: "./node_modules/.bin/mmdc" ``` - if `pip install -e.` fails with error `[Errno 13] Permission denied: '/usr/local/lib/python3.11/dist-packages/test-easy-install-13129.write-test'`, try instead running `pip install -e. --user` @@ -59,12 +59,13 @@ # Step 3: Clone the repository to your local machine, and install it. playwright install --with-deps chromium ``` - - **modify `config.yaml`** + - **modify `config2.yaml`** - uncomment MERMAID_ENGINE from config.yaml and change it to `playwright` + uncomment mermaid.engine from config2.yaml and change it to `playwright` ```yaml - MERMAID_ENGINE: playwright + mermaid: + engine: playwright ``` - pyppeteer @@ -88,21 +89,23 @@ # Step 3: Clone the repository to your local machine, and install it. pyppeteer-install ``` - - **modify `config.yaml`** + - **modify `config2.yaml`** - uncomment MERMAID_ENGINE from config.yaml and change it to `pyppeteer` + uncomment mermaid.engine from config2.yaml and change it to `pyppeteer` ```yaml - MERMAID_ENGINE: pyppeteer + mermaid: + engine: pyppeteer ``` - mermaid.ink - - **modify `config.yaml`** + - **modify `config2.yaml`** - uncomment MERMAID_ENGINE from config.yaml and change it to `ink` + uncomment mermaid.engine from config2.yaml and change it to `ink` ```yaml - MERMAID_ENGINE: ink + mermaid: + engine: ink ``` Note: this method does not support pdf export. diff --git a/docs/install/cli_install_cn.md b/docs/install/cli_install_cn.md index b1da1b813..891b72d24 100644 --- a/docs/install/cli_install_cn.md +++ b/docs/install/cli_install_cn.md @@ -36,8 +36,8 @@ # 第 3 步:克隆仓库到您的本地机器,并进行安装。 - 不要忘记在config.yml中为mmdc配置配置, ```yml - PUPPETEER_CONFIG: "./config/puppeteer-config.json" - MMDC: "./node_modules/.bin/mmdc" + puppeteer_config: "./config/puppeteer-config.json" + path: "./node_modules/.bin/mmdc" ``` - 如果`pip install -e.`失败并显示错误`[Errno 13] Permission denied: '/usr/local/lib/python3.11/dist-packages/test-easy-install-13129.write-test'`,请尝试使用`pip install -e. --user`运行。 diff --git a/docs/install/docker_install.md b/docs/install/docker_install.md index 37125bdbe..2fe1b6abf 100644 --- a/docs/install/docker_install.md +++ b/docs/install/docker_install.md @@ -3,16 +3,16 @@ ## Docker Installation ### Use default MetaGPT image ```bash -# Step 1: Download metagpt official image and prepare config.yaml +# Step 1: Download metagpt official image and prepare config2.yaml docker pull metagpt/metagpt:latest mkdir -p /opt/metagpt/{config,workspace} -docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config.yaml > /opt/metagpt/config/key.yaml -vim /opt/metagpt/config/key.yaml # Change the config +docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config2.yaml > /opt/metagpt/config/config2.yaml +vim /opt/metagpt/config/config2.yaml # Change the config # Step 2: Run metagpt demo with container docker run --rm \ --privileged \ - -v /opt/metagpt/config/key.yaml:/app/metagpt/config/key.yaml \ + -v /opt/metagpt/config/config2.yaml:/app/metagpt/config/config2.yaml \ -v /opt/metagpt/workspace:/app/metagpt/workspace \ metagpt/metagpt:latest \ metagpt "Write a cli snake game" @@ -20,7 +20,7 @@ # Step 2: Run metagpt demo with container # You can also start a container and execute commands in it docker run --name metagpt -d \ --privileged \ - -v /opt/metagpt/config/key.yaml:/app/metagpt/config/key.yaml \ + -v /opt/metagpt/config/config2.yaml:/app/metagpt/config/config2.yaml \ -v /opt/metagpt/workspace:/app/metagpt/workspace \ metagpt/metagpt:latest @@ -31,7 +31,7 @@ # You can also start a container and execute commands in it The command `docker run ...` do the following things: - Run in privileged mode to have permission to run the browser -- Map host configure file `/opt/metagpt/config/key.yaml` to container `/app/metagpt/config/key.yaml` +- Map host configure file `/opt/metagpt/config/config2.yaml` to container `/app/metagpt/config/config2.yaml` - Map host directory `/opt/metagpt/workspace` to container `/app/metagpt/workspace` - Execute the demo command `metagpt "Write a cli snake game"` diff --git a/docs/install/docker_install_cn.md b/docs/install/docker_install_cn.md index f360b49ed..10204c1e0 100644 --- a/docs/install/docker_install_cn.md +++ b/docs/install/docker_install_cn.md @@ -3,16 +3,16 @@ ## Docker安装 ### 使用MetaGPT镜像 ```bash -# 步骤1: 下载metagpt官方镜像并准备好config.yaml +# 步骤1: 下载metagpt官方镜像并准备好config2.yaml docker pull metagpt/metagpt:latest mkdir -p /opt/metagpt/{config,workspace} -docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config.yaml > /opt/metagpt/config/key.yaml -vim /opt/metagpt/config/key.yaml # 修改配置文件 +docker run --rm metagpt/metagpt:latest cat /app/metagpt/config/config2.yaml > /opt/metagpt/config/config2.yaml +vim /opt/metagpt/config/config2.yaml # 修改配置文件 # 步骤2: 使用容器运行metagpt演示 docker run --rm \ --privileged \ - -v /opt/metagpt/config/key.yaml:/app/metagpt/config/key.yaml \ + -v /opt/metagpt/config/config2.yaml:/app/metagpt/config/config2.yaml \ -v /opt/metagpt/workspace:/app/metagpt/workspace \ metagpt/metagpt:latest \ metagpt "Write a cli snake game" @@ -20,7 +20,7 @@ # 步骤2: 使用容器运行metagpt演示 # 您也可以启动一个容器并在其中执行命令 docker run --name metagpt -d \ --privileged \ - -v /opt/metagpt/config/key.yaml:/app/metagpt/config/key.yaml \ + -v /opt/metagpt/config/config2.yaml:/app/metagpt/config/config2.yaml \ -v /opt/metagpt/workspace:/app/metagpt/workspace \ metagpt/metagpt:latest @@ -31,7 +31,7 @@ # 您也可以启动一个容器并在其中执行命令 `docker run ...`做了以下事情: - 以特权模式运行,有权限运行浏览器 -- 将主机文件 `/opt/metagpt/config/key.yaml` 映射到容器文件 `/app/metagpt/config/key.yaml` +- 将主机文件 `/opt/metagpt/config/config2.yaml` 映射到容器文件 `/app/metagpt/config/config2.yaml` - 将主机目录 `/opt/metagpt/workspace` 映射到容器目录 `/app/metagpt/workspace` - 执行示例命令 `metagpt "Write a cli snake game"` diff --git a/docs/tutorial/usage.md b/docs/tutorial/usage.md index a08d92a22..809f91e1f 100644 --- a/docs/tutorial/usage.md +++ b/docs/tutorial/usage.md @@ -2,19 +2,14 @@ ## MetaGPT Usage ### Configuration -- Configure your `OPENAI_API_KEY` in any of `config/key.yaml / config/config.yaml / env` -- Priority order: `config/key.yaml > config/config.yaml > env` +- Configure your `key` in any of `~/.metagpt/config2.yaml / config/config2.yaml` +- Priority order: `~/.metagpt/config2.yaml > config/config2.yaml` ```bash # Copy the configuration file and make the necessary modifications. -cp config/config.yaml config/key.yaml +cp config/config2.yaml ~/.metagpt/config2.yaml ``` -| Variable Name | config/key.yaml | env | -| ------------------------------------------ | ----------------------------------------- | ----------------------------------------------- | -| OPENAI_API_KEY # Replace with your own key | OPENAI_API_KEY: "sk-..." | export OPENAI_API_KEY="sk-..." | -| OPENAI_BASE_URL # Optional | OPENAI_BASE_URL: "https:///v1" | export OPENAI_BASE_URL="https:///v1" | - ### Initiating a startup ```shell diff --git a/docs/tutorial/usage_cn.md b/docs/tutorial/usage_cn.md index 76a5d6b1b..709ec9968 100644 --- a/docs/tutorial/usage_cn.md +++ b/docs/tutorial/usage_cn.md @@ -2,19 +2,14 @@ ## MetaGPT 使用 ### 配置 -- 在 `config/key.yaml / config/config.yaml / env` 中配置您的 `OPENAI_API_KEY` -- 优先级顺序:`config/key.yaml > config/config.yaml > env` +- 在 `~/.metagpt/config2.yaml / config/config2.yaml` 中配置您的 `key` +- 优先级顺序:`~/.metagpt/config2.yaml > config/config2.yaml` ```bash # 复制配置文件并进行必要的修改 -cp config/config.yaml config/key.yaml +cp config/config2.yaml ~/.metagpt/config2.yaml ``` -| 变量名 | config/key.yaml | env | -| ----------------------------------- | ----------------------------------------- | ----------------------------------------------- | -| OPENAI_API_KEY # 用您自己的密钥替换 | OPENAI_API_KEY: "sk-..." | export OPENAI_API_KEY="sk-..." | -| OPENAI_BASE_URL # 可选 | OPENAI_BASE_URL: "https:///v1" | export OPENAI_BASE_URL="https:///v1" | - ### 示例:启动一个创业公司 ```shell diff --git a/metagpt/actions/design_api.py b/metagpt/actions/design_api.py index cb6013538..e5f038c7c 100644 --- a/metagpt/actions/design_api.py +++ b/metagpt/actions/design_api.py @@ -117,4 +117,4 @@ class WriteDesign(Action): async def _save_mermaid_file(self, data: str, pathname: Path): pathname.parent.mkdir(parents=True, exist_ok=True) - await mermaid_to_file(self.config.mermaid_engine, data, pathname) + await mermaid_to_file(self.config.mermaid.engine, data, pathname) diff --git a/metagpt/actions/write_prd.py b/metagpt/actions/write_prd.py index 823786893..b66887164 100644 --- a/metagpt/actions/write_prd.py +++ b/metagpt/actions/write_prd.py @@ -159,7 +159,7 @@ class WritePRD(Action): return pathname = self.repo.workdir / COMPETITIVE_ANALYSIS_FILE_REPO / Path(prd_doc.filename).stem pathname.parent.mkdir(parents=True, exist_ok=True) - await mermaid_to_file(self.config.mermaid_engine, quadrant_chart, pathname) + await mermaid_to_file(self.config.mermaid.engine, quadrant_chart, pathname) async def _rename_workspace(self, prd): if not self.project_name: diff --git a/metagpt/config2.py b/metagpt/config2.py index 5a556cc52..de0489789 100644 --- a/metagpt/config2.py +++ b/metagpt/config2.py @@ -67,24 +67,18 @@ class Config(CLIParams, YamlModel): code_review_k_times: int = 2 # Will be removed in the future - llm_for_researcher_summary: str = "gpt3" - llm_for_researcher_report: str = "gpt3" METAGPT_TEXT_TO_IMAGE_MODEL_URL: str = "" language: str = "English" redis_key: str = "placeholder" - mmdc: str = "mmdc" - puppeteer_config: str = "" - pyppeteer_executable_path: str = "" IFLYTEK_APP_ID: str = "" IFLYTEK_API_SECRET: str = "" IFLYTEK_API_KEY: str = "" AZURE_TTS_SUBSCRIPTION_KEY: str = "" AZURE_TTS_REGION: str = "" - mermaid_engine: str = "nodejs" @classmethod def from_home(cls, path): - """Load config from ~/.metagpt/config.yaml""" + """Load config from ~/.metagpt/config2.yaml""" pathname = CONFIG_ROOT / path if not pathname.exists(): return None diff --git a/metagpt/configs/llm_config.py b/metagpt/configs/llm_config.py index 626d4242f..fb923d3e4 100644 --- a/metagpt/configs/llm_config.py +++ b/metagpt/configs/llm_config.py @@ -74,5 +74,5 @@ class LLMConfig(YamlModel): @classmethod def check_llm_key(cls, v): if v in ["", None, "YOUR_API_KEY"]: - raise ValueError("Please set your API key in config.yaml") + raise ValueError("Please set your API key in config2.yaml") return v diff --git a/metagpt/configs/mermaid_config.py b/metagpt/configs/mermaid_config.py index de4a3865c..50c8a1847 100644 --- a/metagpt/configs/mermaid_config.py +++ b/metagpt/configs/mermaid_config.py @@ -14,5 +14,6 @@ class MermaidConfig(YamlModel): """Config for Mermaid""" engine: Literal["nodejs", "ink", "playwright", "pyppeteer"] = "nodejs" - path: str = "" - puppeteer_config: str = "" # Only for nodejs engine + path: str = "mmdc" # mmdc + puppeteer_config: str = "" + pyppeteer_path: str = "/usr/bin/google-chrome-stable" diff --git a/metagpt/utils/mermaid.py b/metagpt/utils/mermaid.py index e49fdea5d..ae3c5118f 100644 --- a/metagpt/utils/mermaid.py +++ b/metagpt/utils/mermaid.py @@ -35,10 +35,10 @@ async def mermaid_to_file(engine, mermaid_code, output_file_without_suffix, widt # tmp.write_text(mermaid_code, encoding="utf-8") if engine == "nodejs": - if check_cmd_exists(config.mmdc) != 0: + if check_cmd_exists(config.mermaid.path) != 0: logger.warning( "RUN `npm install -g @mermaid-js/mermaid-cli` to install mmdc," - "or consider changing MERMAID_ENGINE to `playwright`, `pyppeteer`, or `ink`." + "or consider changing engine to `playwright`, `pyppeteer`, or `ink`." ) return -1 @@ -47,11 +47,11 @@ async def mermaid_to_file(engine, mermaid_code, output_file_without_suffix, widt # Call the `mmdc` command to convert the Mermaid code to a PNG logger.info(f"Generating {output_file}..") - if config.puppeteer_config: + if config.mermaid.puppeteer_config: commands = [ - config.mmdc, + config.mermaid.path, "-p", - config.puppeteer_config, + config.mermaid.puppeteer_config, "-i", str(tmp), "-o", @@ -62,7 +62,7 @@ async def mermaid_to_file(engine, mermaid_code, output_file_without_suffix, widt str(height), ] else: - commands = [config.mmdc, "-i", str(tmp), "-o", output_file, "-w", str(width), "-H", str(height)] + commands = [config.mermaid.path, "-i", str(tmp), "-o", output_file, "-w", str(width), "-H", str(height)] process = await asyncio.create_subprocess_shell( " ".join(commands), stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.PIPE ) diff --git a/metagpt/utils/mmdc_pyppeteer.py b/metagpt/utils/mmdc_pyppeteer.py index d80098b7d..f029325f1 100644 --- a/metagpt/utils/mmdc_pyppeteer.py +++ b/metagpt/utils/mmdc_pyppeteer.py @@ -30,14 +30,14 @@ async def mermaid_to_file(mermaid_code, output_file_without_suffix, width=2048, suffixes = ["png", "svg", "pdf"] __dirname = os.path.dirname(os.path.abspath(__file__)) - if config.pyppeteer_executable_path: + if config.mermaid.pyppeteer_path: browser = await launch( headless=True, - executablePath=config.pyppeteer_executable_path, + executablePath=config.mermaid.pyppeteer_path, args=["--disable-extensions", "--no-sandbox"], ) else: - logger.error("Please set the environment variable:PYPPETEER_EXECUTABLE_PATH.") + logger.error("Please set the var mermaid.pyppeteer_path in the config2.yaml.") return -1 page = await browser.newPage() device_scale_factor = 1.0 diff --git a/metagpt/utils/yaml_model.py b/metagpt/utils/yaml_model.py index 8f2d22c3d..4d42bb03f 100644 --- a/metagpt/utils/yaml_model.py +++ b/metagpt/utils/yaml_model.py @@ -42,7 +42,7 @@ class YamlModelWithoutDefault(YamlModel): @model_validator(mode="before") @classmethod def check_not_default_config(cls, values): - """Check if there is any default config in config.yaml""" + """Check if there is any default config in config2.yaml""" if any(["YOUR" in v for v in values]): - raise ValueError("Please set your config in config.yaml") + raise ValueError("Please set your config in config2.yaml") return values From c3d4af6fc31dce124a369c36944e967adaaf1d08 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 1 Feb 2024 00:15:17 +0800 Subject: [PATCH 539/637] rm unnecessary --- .gitignore | 1 - 1 file changed, 1 deletion(-) diff --git a/.gitignore b/.gitignore index 4752806d7..ae0a17b45 100644 --- a/.gitignore +++ b/.gitignore @@ -131,7 +131,6 @@ venv.bak/ .mypy_cache/ .dmypy.json dmypy.json -metagpt/tools/functions/libs/udf/*.py # Pyre type checker .pyre/ From 37a606df0a74a66d397db86132917392bcb6bacf Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 1 Feb 2024 00:18:53 +0800 Subject: [PATCH 540/637] rm unfinished --- kaggle_team.py | 41 ----------------------------------------- 1 file changed, 41 deletions(-) delete mode 100644 kaggle_team.py diff --git a/kaggle_team.py b/kaggle_team.py deleted file mode 100644 index e9f3e67de..000000000 --- a/kaggle_team.py +++ /dev/null @@ -1,41 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- - -import fire - -from metagpt.roles.kaggle_manager import KaggleManager -from metagpt.roles.ml_engineer import MLEngineer -from metagpt.team import Team - - -async def main( - # competition: str, - # data_desc: str, - # requirement: str, - investment: float = 5.0, - n_round: int = 10, - auto_run: bool = False, -): - competition, data_desc, requirement = ( - "titanic", - "Training set is train.csv.\nTest set is test.csv. We also include gender_submission.csv, a set of predictions that assume all and only female passengers survive, as an example of what a submission file should look like.", - # "Run EDA on the train dataset, train a model to predict survival (20% as validation) and save it, predict the test set using saved model, save the test result according to format", - # "generate a random prediction, replace the Survived column of gender_submission.csv, and save the prediction to a new submission file", - "Score as high as possible for the provided dataset, save the test prediction to a csv with two columns PassengerId and Survived", - ) - - team = Team() - team.hire( - [ - KaggleManager(competition=competition, data_desc=data_desc), - MLEngineer(goal=requirement, auto_run=auto_run), - ] - ) - - team.invest(investment) - team.start_project(requirement) - await team.run(n_round=n_round) - - -if __name__ == "__main__": - fire.Fire(main) From 90f84ad452876ff9b8cf2def0c08faa20925b805 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 10:19:40 +0800 Subject: [PATCH 541/637] Update FAQ-EN.md --- docs/FAQ-EN.md | 174 +++++++++++++++---------------------------------- 1 file changed, 53 insertions(+), 121 deletions(-) diff --git a/docs/FAQ-EN.md b/docs/FAQ-EN.md index 88b5b0573..d3caa244e 100644 --- a/docs/FAQ-EN.md +++ b/docs/FAQ-EN.md @@ -1,161 +1,93 @@ Our vision is to [extend human life](https://github.com/geekan/HowToLiveLonger) and [reduce working hours](https://github.com/geekan/MetaGPT/). -1. ### Convenient Link for Sharing this Document: +### Convenient Link for Sharing this Document: ``` -- MetaGPT-Index/FAQ https://deepwisdom.feishu.cn/wiki/MsGnwQBjiif9c3koSJNcYaoSnu4 +- MetaGPT-Index/FAQ-EN https://github.com/geekan/MetaGPT/blob/main/docs/FAQ-EN.md +- MetaGPT-Index/FAQ-CN https://deepwisdom.feishu.cn/wiki/MsGnwQBjiif9c3koSJNcYaoSnu4 ``` -2. ### Link - - +### Link 1. Code:https://github.com/geekan/MetaGPT - -1. Roadmap:https://github.com/geekan/MetaGPT/blob/main/docs/ROADMAP.md - -1. EN - - 1. Demo Video: [MetaGPT: Multi-Agent AI Programming Framework](https://www.youtube.com/watch?v=8RNzxZBTW8M) +2. Roadmap:https://github.com/geekan/MetaGPT/blob/main/docs/ROADMAP.md +3. EN + 1. Demo Video: [MetaGPT: Multi-Agent AI Programming Framework](https://www.youtube.com/watch?v=8RNzxZBTW8M) 2. Tutorial: [MetaGPT: Deploy POWERFUL Autonomous Ai Agents BETTER Than SUPERAGI!](https://www.youtube.com/watch?v=q16Gi9pTG_M&t=659s) 3. Author's thoughts video(EN): [MetaGPT Matthew Berman](https://youtu.be/uT75J_KG_aY?si=EgbfQNAwD8F5Y1Ak) +4. CN + 1. Demo Video: [MetaGPT:一行代码搭建你的虚拟公司_哔哩哔哩_bilibili](https://www.bilibili.com/video/BV1NP411C7GW/?spm_id_from=333.999.0.0&vd_source=735773c218b47da1b4bd1b98a33c5c77) + 1. Tutorial: [一个提示词写游戏 Flappy bird, 比AutoGPT强10倍的MetaGPT,最接近AGI的AI项目](https://youtu.be/Bp95b8yIH5c) + 2. Author's thoughts video(CN): [MetaGPT作者深度解析直播回放_哔哩哔哩_bilibili](https://www.bilibili.com/video/BV1Ru411V7XL/?spm_id_from=333.337.search-card.all.click) -1. CN - - 1. Demo Video: [MetaGPT:一行代码搭建你的虚拟公司_哔哩哔哩_bilibili](https://www.bilibili.com/video/BV1NP411C7GW/?spm_id_from=333.999.0.0&vd_source=735773c218b47da1b4bd1b98a33c5c77) - 1. Tutorial: [一个提示词写游戏 Flappy bird, 比AutoGPT强10倍的MetaGPT,最接近AGI的AI项目](https://youtu.be/Bp95b8yIH5c) - 2. Author's thoughts video(CN): [MetaGPT作者深度解析直播回放_哔哩哔哩_bilibili](https://www.bilibili.com/video/BV1Ru411V7XL/?spm_id_from=333.337.search-card.all.click) - - - -3. ### How to become a contributor? - - +### How to become a contributor? 1. Choose a task from the Roadmap (or you can propose one). By submitting a PR, you can become a contributor and join the dev team. -1. Current contributors come from backgrounds including ByteDance AI Lab/DingDong/Didi/Xiaohongshu, Tencent/Baidu/MSRA/TikTok/BloomGPT Infra/Bilibili/CUHK/HKUST/CMU/UCB +2. Current contributors come from backgrounds including ByteDance AI Lab/DingDong/Didi/Xiaohongshu, Tencent/Baidu/MSRA/TikTok/BloomGPT Infra/Bilibili/CUHK/HKUST/CMU/UCB - - -4. ### Chief Evangelist (Monthly Rotation) +### Chief Evangelist (Monthly Rotation) MetaGPT Community - The position of Chief Evangelist rotates on a monthly basis. The primary responsibilities include: 1. Maintaining community FAQ documents, announcements, and Github resources/READMEs. -1. Responding to, answering, and distributing community questions within an average of 30 minutes, including on platforms like Github Issues, Discord and WeChat. -1. Upholding a community atmosphere that is enthusiastic, genuine, and friendly. -1. Encouraging everyone to become contributors and participate in projects that are closely related to achieving AGI (Artificial General Intelligence). -1. (Optional) Organizing small-scale events, such as hackathons. +2. Responding to, answering, and distributing community questions within an average of 30 minutes, including on platforms like Github Issues, Discord and WeChat. +3. Upholding a community atmosphere that is enthusiastic, genuine, and friendly. +4. Encouraging everyone to become contributors and participate in projects that are closely related to achieving AGI (Artificial General Intelligence). +5. (Optional) Organizing small-scale events, such as hackathons. - - -5. ### FAQ - - - -1. Experience with the generated repo code: - - 1. https://github.com/geekan/MetaGPT/releases/tag/v0.1.0 +### FAQ 1. Code truncation/ Parsing failure: - - 1. Check if it's due to exceeding length. Consider using the gpt-3.5-turbo-16k or other long token versions. - -1. Success rate: - - 1. There hasn't been a quantitative analysis yet, but the success rate of code generated by GPT-4 is significantly higher than that of gpt-3.5-turbo. - -1. Support for incremental, differential updates (if you wish to continue a half-done task): - - 1. Several prerequisite tasks are listed on the ROADMAP. - -1. Can existing code be loaded? - - 1. It's not on the ROADMAP yet, but there are plans in place. It just requires some time. - -1. Support for multiple programming languages and natural languages? - - 1. It's listed on ROADMAP. - -1. Want to join the contributor team? How to proceed? - + 1. Check if it's due to exceeding length. Consider using the gpt-4-turbo-preview or other long token versions. +2. Success rate: + 1. There hasn't been a quantitative analysis yet, but the success rate of code generated by gpt-4-turbo-preview is significantly higher than that of gpt-3.5-turbo. +3. Support for incremental, differential updates (if you wish to continue a half-done task): + 1. There is now an experimental version. Specify `--inc --project-path ""` or `--inc --project-name ""` on the command line and enter the corresponding requirements to try it. +4. Can existing code be loaded? + 1. We are doing this, but it is very difficult, especially when the project is large, it is very difficult to achieve a high success rate. +5. Support for multiple programming languages and natural languages? + 1. It is now supported, but it is still in experimental version +6. Want to join the contributor team? How to proceed? 1. Merging a PR will get you into the contributor's team. The main ongoing tasks are all listed on the ROADMAP. - -1. PRD stuck / unable to access/ connection interrupted - +7. PRD stuck / unable to access/ connection interrupted 1. The official openai base_url address is `https://api.openai.com/v1` - 1. If the official openai base_url address is inaccessible in your environment (this can be verified with curl), it's recommended to configure using the reverse proxy openai base_url provided by libraries such as openai-forward. For instance, `openai base_url: "``https://api.openai-forward.com/v1``"` - 1. If the official openai base_url address is inaccessible in your environment (again, verifiable via curl), another option is to configure the llm.proxy parameter. This way, you can access the official openai base_url via a local proxy. If you don't need to access via a proxy, please do not enable this configuration; if accessing through a proxy is required, modify it to the correct proxy address. Note that when llm.proxy is enabled, don't set openai base_url. - 1. Note: OpenAI's default API design ends with a v1. An example of the correct configuration is: `openai base_url: "``https://api.openai.com/v1``"` - -1. Absolutely! How can I assist you today? - + 2. If the official openai base_url address is inaccessible in your environment (this can be verified with curl), it's recommended to configure using base_url to other "reverse-proxy" provider such as openai-forward. For instance, `openai base_url: "``https://api.openai-forward.com/v1``"` + 3. If the official openai base_url address is inaccessible in your environment (again, verifiable via curl), another option is to configure the llm.proxy in the `config2.yaml`. This way, you can access the official openai base_url via a local proxy. If you don't need to access via a proxy, please do not enable this configuration; if accessing through a proxy is required, modify it to the correct proxy address. + 4. Note: OpenAI's default API design ends with a v1. An example of the correct configuration is: `base_url: "https://api.openai.com/v1" +8. Get reply: "Absolutely! How can I assist you today?" 1. Did you use Chi or a similar service? These services are prone to errors, and it seems that the error rate is higher when consuming 3.5k-4k tokens in GPT-4 - -1. What does Max token mean? - +9. What does Max token mean? 1. It's a configuration for OpenAI's maximum response length. If the response exceeds the max token, it will be truncated. - -1. How to change the investment amount? - +10. How to change the investment amount? 1. You can view all commands by typing `metagpt --help` - -1. Which version of Python is more stable? - +11. Which version of Python is more stable? 1. python3.9 / python3.10 - -1. Can't use GPT-4, getting the error "The model gpt-4 does not exist." - +12. Can't use GPT-4, getting the error "The model gpt-4 does not exist." 1. OpenAI's official requirement: You can use GPT-4 only after spending $1 on OpenAI. 1. Tip: Run some data with gpt-3.5-turbo (consume the free quota and $1), and then you should be able to use gpt-4. - -1. Can games whose code has never been seen before be written? - +13. Can games whose code has never been seen before be written? 1. Refer to the README. The recommendation system of Toutiao is one of the most complex systems in the world currently. Although it's not on GitHub, many discussions about it exist online. If it can visualize these, it suggests it can also summarize these discussions and convert them into code. The prompt would be something like "write a recommendation system similar to Toutiao". Note: this was approached in earlier versions of the software. The SOP of those versions was different; the current one adopts Elon Musk's five-step work method, emphasizing trimming down requirements as much as possible. - -1. Under what circumstances would there typically be errors? - +14. Under what circumstances would there typically be errors? 1. More than 500 lines of code: some function implementations may be left blank. - 1. When using a database, it often gets the implementation wrong — since the SQL database initialization process is usually not in the code. - 1. With more lines of code, there's a higher chance of false impressions, leading to calls to non-existent APIs. - -1. An error occurred during installation: "Another program is using this file...egg". - + 2. When using a database, it often gets the implementation wrong — since the SQL database initialization process is usually not in the code. + 3. With more lines of code, there's a higher chance of false impressions, leading to calls to non-existent APIs. +15. An error occurred during installation: "Another program is using this file...egg". 1. Delete the file and try again. - 1. Or manually execute`pip install -r requirements.txt` - -1. The origin of the name MetaGPT? - + 2. Or manually execute`pip install -r requirements.txt` +16. The origin of the name MetaGPT? 1. The name was derived after iterating with GPT-4 over a dozen rounds. GPT-4 scored and suggested it. - -1. Is there a more step-by-step installation tutorial? - - 1. Youtube(CN):[一个提示词写游戏 Flappy bird, 比AutoGPT强10倍的MetaGPT,最接近AGI的AI项目=一个软件公司产品经理+程序员](https://youtu.be/Bp95b8yIH5c) - 1. Youtube(EN)https://www.youtube.com/watch?v=q16Gi9pTG_M&t=659s - 2. video(EN): [MetaGPT Matthew Berman](https://youtu.be/uT75J_KG_aY?si=EgbfQNAwD8F5Y1Ak) - -1. openai.error.RateLimitError: You exceeded your current quota, please check your plan and billing details - +17. openai.error.RateLimitError: You exceeded your current quota, please check your plan and billing details 1. If you haven't exhausted your free quota, set RPM to 3 or lower in the settings. - 1. If your free quota is used up, consider adding funds to your account. - -1. What does "borg" mean in n_borg? - + 2. If your free quota is used up, consider adding funds to your account. +18. What does "borg" mean in n_borg? 1. [Wikipedia borg meaning ](https://en.wikipedia.org/wiki/Borg) - 1. The Borg civilization operates based on a hive or collective mentality, known as "the Collective." Every Borg individual is connected to the collective via a sophisticated subspace network, ensuring continuous oversight and guidance for every member. This collective consciousness allows them to not only "share the same thoughts" but also to adapt swiftly to new strategies. While individual members of the collective rarely communicate, the collective "voice" sometimes transmits aboard ships. - -1. How to use the Claude API? - + 2. The Borg civilization operates based on a hive or collective mentality, known as "the Collective." Every Borg individual is connected to the collective via a sophisticated subspace network, ensuring continuous oversight and guidance for every member. This collective consciousness allows them to not only "share the same thoughts" but also to adapt swiftly to new strategies. While individual members of the collective rarely communicate, the collective "voice" sometimes transmits aboard ships. +19. How to use the Claude API? 1. The full implementation of the Claude API is not provided in the current code. 1. You can use the Claude API through third-party API conversion projects like: https://github.com/jtsang4/claude-to-chatgpt - -1. Is Llama2 supported? - +20. Is Llama2 supported? 1. On the day Llama2 was released, some of the community members began experiments and found that output can be generated based on MetaGPT's structure. However, Llama2's context is too short to generate a complete project. Before regularly using Llama2, it's necessary to expand the context window to at least 8k. If anyone has good recommendations for expansion models or methods, please leave a comment. - -1. `mermaid-cli getElementsByTagName SyntaxError: Unexpected token '.'` - +21. `mermaid-cli getElementsByTagName SyntaxError: Unexpected token '.'` 1. Upgrade node to version 14.x or above: - 1. `npm install -g n` - 1. `n stable` to install the stable version of node(v18.x) + 2. `n stable` to install the stable version of node(v18.x) From 026dd8167cc0ae95a1eddb8a2a74b97c2518e2e1 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 10:21:58 +0800 Subject: [PATCH 542/637] Update cli_install.md --- docs/install/cli_install.md | 17 +++++++++-------- 1 file changed, 9 insertions(+), 8 deletions(-) diff --git a/docs/install/cli_install.md b/docs/install/cli_install.md index 33d759758..ab360fad2 100644 --- a/docs/install/cli_install.md +++ b/docs/install/cli_install.md @@ -33,11 +33,12 @@ # Step 3: Clone the repository to your local machine, and install it. npm install @mermaid-js/mermaid-cli ``` -- don't forget to the configuration for mmdc in config.yml +- don't forget to the configuration for mmdc path in config.yml ```yml - puppeteer_config: "./config/puppeteer-config.json" - path: "./node_modules/.bin/mmdc" + mermaid: + puppeteer_config: "./config/puppeteer-config.json" + path: "./node_modules/.bin/mmdc" ``` - if `pip install -e.` fails with error `[Errno 13] Permission denied: '/usr/local/lib/python3.11/dist-packages/test-easy-install-13129.write-test'`, try instead running `pip install -e. --user` @@ -61,7 +62,7 @@ # Step 3: Clone the repository to your local machine, and install it. - **modify `config2.yaml`** - uncomment mermaid.engine from config2.yaml and change it to `playwright` + change mermaid.engine to `playwright` ```yaml mermaid: @@ -91,7 +92,7 @@ # Step 3: Clone the repository to your local machine, and install it. - **modify `config2.yaml`** - uncomment mermaid.engine from config2.yaml and change it to `pyppeteer` + change mermaid.engine to `pyppeteer` ```yaml mermaid: @@ -100,8 +101,8 @@ # Step 3: Clone the repository to your local machine, and install it. - mermaid.ink - **modify `config2.yaml`** - - uncomment mermaid.engine from config2.yaml and change it to `ink` + + change mermaid.engine to `ink` ```yaml mermaid: @@ -109,4 +110,4 @@ # Step 3: Clone the repository to your local machine, and install it. ``` Note: this method does not support pdf export. - \ No newline at end of file + From 3fa2b3216effaeabfcb8294bbc674e2bb9becbc7 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 10:35:51 +0800 Subject: [PATCH 543/637] refine docs --- docs/install/cli_install.md | 28 +++++++++++++------ docs/install/cli_install_cn.md | 33 ++++++++++++++++------- docs/tutorial/usage.md | 49 +++++++++++++++++----------------- docs/tutorial/usage_cn.md | 49 +++++++++++++++++----------------- 4 files changed, 91 insertions(+), 68 deletions(-) diff --git a/docs/install/cli_install.md b/docs/install/cli_install.md index ab360fad2..b79ad9cb7 100644 --- a/docs/install/cli_install.md +++ b/docs/install/cli_install.md @@ -9,17 +9,29 @@ ### Support System and version ### Detail Installation ```bash -# Step 1: Ensure that NPM is installed on your system. Then install mermaid-js. (If you don't have npm in your computer, please go to the Node.js official website to install Node.js https://nodejs.org/ and then you will have npm tool in your computer.) -npm --version -sudo npm install -g @mermaid-js/mermaid-cli - -# Step 2: Ensure that Python 3.9+ is installed on your system. You can check this by using: +# Step 1: Ensure that Python 3.9+ is installed on your system. You can check this by using: +# You can use conda to initialize a new python env +# conda create -n metagpt python=3.9 +# conda activate metagpt python3 --version -# Step 3: Clone the repository to your local machine, and install it. +# Step 2: Clone the repository to your local machine for latest version, and install it. git clone https://github.com/geekan/MetaGPT.git cd MetaGPT -pip install -e. +pip3 install -e . # or pip3 install metagpt # for stable version + +# Step 3: setup your LLM key in the config2.yaml file +mkdir ~/.metagpt +cp config/config2.yaml ~/.metagpt/config2.yaml +vim ~/.metagpt/config2.yaml + +# Step 4: run metagpt cli +metagpt "Create a 2048 game in python" + +# Step 5 [Optional]: If you want to save the artifacts like diagrams such as quadrant chart, system designs, sequence flow in the workspace, you can execute the step before Step 3. By default, the framework is compatible, and the entire process can be run completely without executing this step. +# If executing, ensure that NPM is installed on your system. Then install mermaid-js. (If you don't have npm in your computer, please go to the Node.js official website to install Node.js https://nodejs.org/ and then you will have npm tool in your computer.) +npm --version +sudo npm install -g @mermaid-js/mermaid-cli ``` **Note:** @@ -35,7 +47,7 @@ # Step 3: Clone the repository to your local machine, and install it. - don't forget to the configuration for mmdc path in config.yml - ```yml + ```yaml mermaid: puppeteer_config: "./config/puppeteer-config.json" path: "./node_modules/.bin/mmdc" diff --git a/docs/install/cli_install_cn.md b/docs/install/cli_install_cn.md index 891b72d24..1ee18d9a6 100644 --- a/docs/install/cli_install_cn.md +++ b/docs/install/cli_install_cn.md @@ -10,17 +10,29 @@ ### 支持的系统和版本 ### 详细安装 ```bash -# 第 1 步:确保您的系统上安装了 NPM。并使用npm安装mermaid-js -npm --version -sudo npm install -g @mermaid-js/mermaid-cli - -# 第 2 步:确保您的系统上安装了 Python 3.9+。您可以使用以下命令进行检查: +# 步骤 1: 确保您的系统安装了 Python 3.9 或更高版本。您可以使用以下命令来检查: +# 您可以使用 conda 来初始化一个新的 Python 环境 +# conda create -n metagpt python=3.9 +# conda activate metagpt python3 --version -# 第 3 步:克隆仓库到您的本地机器,并进行安装。 +# 步骤 2: 克隆仓库到您的本地机器以获取最新版本,并安装它。 git clone https://github.com/geekan/MetaGPT.git cd MetaGPT -pip install -e. +pip3 install -e . # 或 pip3 install metagpt # 用于稳定版本 + +# 步骤 3: 在 config2.yaml 文件中设置您的 LLM 密钥 +mkdir ~/.metagpt +cp config/config2.yaml ~/.metagpt/config2.yaml +vim ~/.metagpt/config2.yaml + +# 步骤 4: 运行 metagpt 命令行界面 +metagpt "用 python 创建一个 2048 游戏" + +# 步骤 5 [可选]: 如果您想保存诸如象限图、系统设计、序列流等图表作为工作空间的工件,您可以在执行步骤 3 之前执行此步骤。默认情况下,该框架是兼容的,整个过程可以完全不执行此步骤而运行。 +# 如果执行此步骤,请确保您的系统上安装了 NPM。然后安装 mermaid-js。(如果您的计算机中没有 npm,请访问 Node.js 官方网站 https://nodejs.org/ 安装 Node.js,然后您将在计算机中拥有 npm 工具。) +npm --version +sudo npm install -g @mermaid-js/mermaid-cli ``` **注意:** @@ -33,11 +45,12 @@ # 第 3 步:克隆仓库到您的本地机器,并进行安装。 npm install @mermaid-js/mermaid-cli ``` -- 不要忘记在config.yml中为mmdc配置配置, +- 不要忘记在config.yml中为mmdc配置 ```yml - puppeteer_config: "./config/puppeteer-config.json" - path: "./node_modules/.bin/mmdc" + mermaid: + puppeteer_config: "./config/puppeteer-config.json" + path: "./node_modules/.bin/mmdc" ``` - 如果`pip install -e.`失败并显示错误`[Errno 13] Permission denied: '/usr/local/lib/python3.11/dist-packages/test-easy-install-13129.write-test'`,请尝试使用`pip install -e. --user`运行。 diff --git a/docs/tutorial/usage.md b/docs/tutorial/usage.md index 809f91e1f..e8bfc37d9 100644 --- a/docs/tutorial/usage.md +++ b/docs/tutorial/usage.md @@ -34,29 +34,28 @@ ### Preference of Platform or Tool ### Usage ``` -NAME - metagpt - We are a software startup comprised of AI. By investing in us, you are empowering a future filled with limitless possibilities. - -SYNOPSIS - metagpt IDEA - -DESCRIPTION - We are a software startup comprised of AI. By investing in us, you are empowering a future filled with limitless possibilities. - -POSITIONAL ARGUMENTS - IDEA - Type: str - Your innovative idea, such as "Creating a snake game." - -FLAGS - --investment=INVESTMENT - Type: float - Default: 3.0 - As an investor, you have the opportunity to contribute a certain dollar amount to this AI company. - --n_round=N_ROUND - Type: int - Default: 5 - -NOTES - You can also use flags syntax for POSITIONAL ARGUMENTS + Usage: metagpt [OPTIONS] [IDEA] + + Start a new project. + +╭─ Arguments ────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────╮ +│ idea [IDEA] Your innovative idea, such as 'Create a 2048 game.' [default: None] │ +╰────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────╯ +╭─ Options ──────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────╮ +│ --investment FLOAT Dollar amount to invest in the AI company. [default: 3.0] │ +│ --n-round INTEGER Number of rounds for the simulation. [default: 5] │ +│ --code-review --no-code-review Whether to use code review. [default: code-review] │ +│ --run-tests --no-run-tests Whether to enable QA for adding & running tests. [default: no-run-tests] │ +│ --implement --no-implement Enable or disable code implementation. [default: implement] │ +│ --project-name TEXT Unique project name, such as 'game_2048'. │ +│ --inc --no-inc Incremental mode. Use it to coop with existing repo. [default: no-inc] │ +│ --project-path TEXT Specify the directory path of the old version project to fulfill the incremental requirements. │ +│ --reqa-file TEXT Specify the source file name for rewriting the quality assurance code. │ +│ --max-auto-summarize-code INTEGER The maximum number of times the 'SummarizeCode' action is automatically invoked, with -1 indicating unlimited. This parameter is used for debugging the │ +│ workflow. │ +│ [default: 0] │ +│ --recover-path TEXT recover the project from existing serialized storage [default: None] │ +│ --init-config --no-init-config Initialize the configuration file for MetaGPT. [default: no-init-config] │ +│ --help Show this message and exit. │ +╰────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────╯ ``` \ No newline at end of file diff --git a/docs/tutorial/usage_cn.md b/docs/tutorial/usage_cn.md index 709ec9968..075e928fd 100644 --- a/docs/tutorial/usage_cn.md +++ b/docs/tutorial/usage_cn.md @@ -30,29 +30,28 @@ ### 平台或工具的倾向性 ### 使用 ``` -名称 - metagpt - 我们是一家AI软件创业公司。通过投资我们,您将赋能一个充满无限可能的未来。 - -概要 - metagpt IDEA - -描述 - 我们是一家AI软件创业公司。通过投资我们,您将赋能一个充满无限可能的未来。 - -位置参数 - IDEA - 类型: str - 您的创新想法,例如"写一个命令行贪吃蛇。" - -标志 - --investment=INVESTMENT - 类型: float - 默认值: 3.0 - 作为投资者,您有机会向这家AI公司投入一定的美元金额。 - --n_round=N_ROUND - 类型: int - 默认值: 5 - -备注 - 您也可以用`标志`的语法,来处理`位置参数` + Usage: metagpt [OPTIONS] [IDEA] + + Start a new project. + +╭─ Arguments ────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────╮ +│ idea [IDEA] Your innovative idea, such as 'Create a 2048 game.' [default: None] │ +╰────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────╯ +╭─ Options ──────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────╮ +│ --investment FLOAT Dollar amount to invest in the AI company. [default: 3.0] │ +│ --n-round INTEGER Number of rounds for the simulation. [default: 5] │ +│ --code-review --no-code-review Whether to use code review. [default: code-review] │ +│ --run-tests --no-run-tests Whether to enable QA for adding & running tests. [default: no-run-tests] │ +│ --implement --no-implement Enable or disable code implementation. [default: implement] │ +│ --project-name TEXT Unique project name, such as 'game_2048'. │ +│ --inc --no-inc Incremental mode. Use it to coop with existing repo. [default: no-inc] │ +│ --project-path TEXT Specify the directory path of the old version project to fulfill the incremental requirements. │ +│ --reqa-file TEXT Specify the source file name for rewriting the quality assurance code. │ +│ --max-auto-summarize-code INTEGER The maximum number of times the 'SummarizeCode' action is automatically invoked, with -1 indicating unlimited. This parameter is used for debugging the │ +│ workflow. │ +│ [default: 0] │ +│ --recover-path TEXT recover the project from existing serialized storage [default: None] │ +│ --init-config --no-init-config Initialize the configuration file for MetaGPT. [default: no-init-config] │ +│ --help Show this message and exit. │ +╰────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────────╯ ``` From c0a4d7c4c94e00787587300f46aacae938b2b165 Mon Sep 17 00:00:00 2001 From: better629 Date: Thu, 1 Feb 2024 10:42:18 +0800 Subject: [PATCH 544/637] update fix of pr review --- examples/dalle_gpt4v_agent.py | 10 +--------- examples/llm_hello_world.py | 9 +++------ tests/metagpt/actions/test_action_node.py | 1 + tests/metagpt/environment/test_base_env.py | 4 ++-- 4 files changed, 7 insertions(+), 17 deletions(-) diff --git a/examples/dalle_gpt4v_agent.py b/examples/dalle_gpt4v_agent.py index 2b54b18a0..28215dba3 100644 --- a/examples/dalle_gpt4v_agent.py +++ b/examples/dalle_gpt4v_agent.py @@ -2,15 +2,7 @@ # -*- coding: utf-8 -*- # @Desc : use gpt4v to improve prompt and draw image with dall-e-3 -""" -set the configuration in `config2.yaml` like below -``` -llm: - base_url: "xxx" - api_key: "sk-xxx" - model: "gpt-4-vision-preview" -``` -""" +"""set `model: "gpt-4-vision-preview"` in `config2.yaml` first""" import asyncio diff --git a/examples/llm_hello_world.py b/examples/llm_hello_world.py index dfc2603aa..1d132eb8a 100644 --- a/examples/llm_hello_world.py +++ b/examples/llm_hello_world.py @@ -29,14 +29,11 @@ async def main(): if hasattr(llm, "completion"): logger.info(llm.completion(hello_msg)) - # check llm-vision capacity if it supports + # check if the configured llm supports llm-vision capacity. If not, it will throw a error invoice_path = Path(__file__).parent.joinpath("..", "tests", "data", "invoices", "invoice-2.png") img_base64 = encode_image(invoice_path) - try: - res = await llm.aask(msg="if this is a invoice, just return True else return False", images=[img_base64]) - assert "true" in res.lower() - except Exception: - pass + res = await llm.aask(msg="if this is a invoice, just return True else return False", images=[img_base64]) + assert "true" in res.lower() if __name__ == "__main__": diff --git a/tests/metagpt/actions/test_action_node.py b/tests/metagpt/actions/test_action_node.py index 18b4d0d0d..589282879 100644 --- a/tests/metagpt/actions/test_action_node.py +++ b/tests/metagpt/actions/test_action_node.py @@ -254,6 +254,7 @@ async def test_action_node_with_image(): node = await invoice.fill(context="", llm=LLM(), images=[img_base64]) assert node.instruct_content.invoice + class ToolDef(BaseModel): tool_name: str = Field(default="a", description="tool name", examples=[]) description: str = Field(default="b", description="tool description", examples=[]) diff --git a/tests/metagpt/environment/test_base_env.py b/tests/metagpt/environment/test_base_env.py index 59c68fc9e..ce8165f2f 100644 --- a/tests/metagpt/environment/test_base_env.py +++ b/tests/metagpt/environment/test_base_env.py @@ -18,7 +18,7 @@ class ForTestEnv(Environment): value: int = 0 @mark_as_readable - def read_api_no_parms(self): + def read_api_no_param(self): return self.value @mark_as_readable @@ -46,5 +46,5 @@ async def test_ext_env(): with pytest.raises(ValueError): await env.observe("not_exist_api") - assert await env.observe("read_api_no_parms") == 15 + assert await env.observe("read_api_no_param") == 15 assert await env.observe(EnvAPIAbstract(api_name="read_api", kwargs={"a": 5, "b": 5})) == 10 From 7e973341682bdc555566aca47a534ef455f5346f Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 10:53:08 +0800 Subject: [PATCH 545/637] refine docs --- README.md | 27 +++++++-------------------- docs/README_CN.md | 27 +++++++++------------------ docs/tutorial/usage.md | 2 +- docs/tutorial/usage_cn.md | 2 +- metagpt/startup.py | 22 +++++++++++----------- 5 files changed, 29 insertions(+), 51 deletions(-) diff --git a/README.md b/README.md index 39dde8208..c8277d55e 100644 --- a/README.md +++ b/README.md @@ -55,30 +55,17 @@ ## Install ### Pip installation +> Ensure that Python 3.9+ is installed on your system. You can check this by using: `python --version`. + ```bash -# Step 1: Ensure that Python 3.9+ is installed on your system. You can check this by using: # You can use conda to initialize a new python env # conda create -n metagpt python=3.9 # conda activate metagpt -python3 --version +pip install metagpt +metagpt --init-config # this will create a ~/.metagpt/config2.yaml from config/config2.yaml, modify it to your own config -# Step 2: Clone the repository to your local machine for latest version, and install it. -git clone https://github.com/geekan/MetaGPT.git -cd MetaGPT -pip3 install -e . # or pip3 install metagpt # for stable version - -# Step 3: setup your LLM key in the config2.yaml file -mkdir ~/.metagpt -cp config/config2.yaml ~/.metagpt/config2.yaml -vim ~/.metagpt/config2.yaml - -# Step 4: run metagpt cli -metagpt "Create a 2048 game in python" - -# Step 5 [Optional]: If you want to save the artifacts like diagrams such as quadrant chart, system designs, sequence flow in the workspace, you can execute the step before Step 3. By default, the framework is compatible, and the entire process can be run completely without executing this step. -# If executing, ensure that NPM is installed on your system. Then install mermaid-js. (If you don't have npm in your computer, please go to the Node.js official website to install Node.js https://nodejs.org/ and then you will have npm tool in your computer.) -npm --version -sudo npm install -g @mermaid-js/mermaid-cli +# Usage: metagpt "" +metagpt "Create a 2048 game" ``` detail installation please refer to [cli_install](https://docs.deepwisdom.ai/main/en/guide/get_started/installation.html#install-stable-version) @@ -99,7 +86,7 @@ # Step 2: Run metagpt demo with container -v /opt/metagpt/config/config2.yaml:/app/metagpt/config/config2.yaml \ -v /opt/metagpt/workspace:/app/metagpt/workspace \ metagpt/metagpt:latest \ - metagpt "Write a cli snake game" + metagpt "Create a 2048 game" ``` detail installation please refer to [docker_install](https://docs.deepwisdom.ai/main/en/guide/get_started/installation.html#install-with-docker) diff --git a/docs/README_CN.md b/docs/README_CN.md index ebf5dd408..52e781560 100644 --- a/docs/README_CN.md +++ b/docs/README_CN.md @@ -35,29 +35,20 @@ # MetaGPT: 多智能体框架 ## 安装 ### Pip安装 +> 确保您的系统安装了 Python 3.9 或更高版本。您可以通过以下命令来检查:`python --version`。 + ```bash -# 第 1 步:确保您的系统上安装了 Python 3.9+。您可以使用以下命令进行检查: -# 可以使用conda来初始化新的python环境 +# 您可以使用 conda 来初始化一个新的 python 环境 # conda create -n metagpt python=3.9 # conda activate metagpt -python3 --version +pip install metagpt +metagpt --init-config # 这将会从 config/config2.yaml 创建一个 ~/.metagpt/config2.yaml,根据您的需求修改它 -# 第 2 步:克隆最新仓库到您的本地机器,并进行安装。 -git clone https://github.com/geekan/MetaGPT.git -cd MetaGPT -pip3 install -e. # 或者 pip3 install metagpt # 安装稳定版本 - -# 第 3 步:执行metagpt -# 拷贝config2.yaml为~/.metagpt/config2.yaml,并设置你自己的api_key -metagpt "Write a cli snake game" - -# 第 4 步【可选的】:如果你想在执行过程中保存像象限图、系统设计、序列流程等图表这些产物,可以在第3步前执行该步骤。默认的,框架做了兼容,在不执行该步的情况下,也可以完整跑完整个流程。 -# 如果执行,确保您的系统上安装了 NPM。并使用npm安装mermaid-js -npm --version -sudo npm install -g @mermaid-js/mermaid-cli +# 使用方法: metagpt "<创建一个游戏或软件>" +metagpt "创建一个 2048 游戏" ``` -详细的安装请安装 [cli_install](https://docs.deepwisdom.ai/guide/get_started/installation.html#install-stable-version) +详细的安装请参考 [cli_install](https://docs.deepwisdom.ai/guide/get_started/installation.html#install-stable-version) ### Docker安装 > 注意:在Windows中,你需要将 "/opt/metagpt" 替换为Docker具有创建权限的目录,比如"D:\Users\x\metagpt" @@ -78,7 +69,7 @@ # 步骤2: 使用容器运行metagpt演示 metagpt "Write a cli snake game" ``` -详细的安装请安装 [docker_install](https://docs.deepwisdom.ai/main/zh/guide/get_started/installation.html#%E4%BD%BF%E7%94%A8docker%E5%AE%89%E8%A3%85) +详细的安装请参考 [docker_install](https://docs.deepwisdom.ai/main/zh/guide/get_started/installation.html#%E4%BD%BF%E7%94%A8docker%E5%AE%89%E8%A3%85) ### 快速开始的演示视频 - 在 [MetaGPT Huggingface Space](https://huggingface.co/spaces/deepwisdom/MetaGPT) 上进行体验 diff --git a/docs/tutorial/usage.md b/docs/tutorial/usage.md index e8bfc37d9..1128e98a5 100644 --- a/docs/tutorial/usage.md +++ b/docs/tutorial/usage.md @@ -2,7 +2,7 @@ ## MetaGPT Usage ### Configuration -- Configure your `key` in any of `~/.metagpt/config2.yaml / config/config2.yaml` +- Configure your `api_key` in any of `~/.metagpt/config2.yaml / config/config2.yaml` - Priority order: `~/.metagpt/config2.yaml > config/config2.yaml` ```bash diff --git a/docs/tutorial/usage_cn.md b/docs/tutorial/usage_cn.md index 075e928fd..3b0c86279 100644 --- a/docs/tutorial/usage_cn.md +++ b/docs/tutorial/usage_cn.md @@ -2,7 +2,7 @@ ## MetaGPT 使用 ### 配置 -- 在 `~/.metagpt/config2.yaml / config/config2.yaml` 中配置您的 `key` +- 在 `~/.metagpt/config2.yaml / config/config2.yaml` 中配置您的 `api_key` - 优先级顺序:`~/.metagpt/config2.yaml > config/config2.yaml` ```bash diff --git a/metagpt/startup.py b/metagpt/startup.py index 4a077cab7..26bb29cd1 100644 --- a/metagpt/startup.py +++ b/metagpt/startup.py @@ -17,17 +17,17 @@ app = typer.Typer(add_completion=False, pretty_exceptions_show_locals=False) def generate_repo( idea, - investment, - n_round, - code_review, - run_tests, - implement, - project_name, - inc, - project_path, - reqa_file, - max_auto_summarize_code, - recover_path, + investment=3.0, + n_round=5, + code_review=True, + run_tests=False, + implement=True, + project_name="", + inc=False, + project_path="", + reqa_file="", + max_auto_summarize_code=0, + recover_path=None, ) -> ProjectRepo: """Run the startup logic. Can be called from CLI or other Python scripts.""" from metagpt.roles import ( From c20aecf5f27c4a5fdc1340ecb0ec6ab8f41360af Mon Sep 17 00:00:00 2001 From: voidking Date: Wed, 31 Jan 2024 19:32:36 +0800 Subject: [PATCH 546/637] feat: auto-unittest --- .github/workflows/auto-unittest.yaml | 74 ++++++++++++++++++++++++++++ .github/workflows/unittest.yaml | 1 + tests/config2.yaml | 30 +++++++++++ tests/spark.yaml | 7 +++ 4 files changed, 112 insertions(+) create mode 100644 .github/workflows/auto-unittest.yaml create mode 100644 tests/config2.yaml create mode 100644 tests/spark.yaml diff --git a/.github/workflows/auto-unittest.yaml b/.github/workflows/auto-unittest.yaml new file mode 100644 index 000000000..33c6acb0e --- /dev/null +++ b/.github/workflows/auto-unittest.yaml @@ -0,0 +1,74 @@ +name: Auto Unit Tests + +on: + pull_request_target: + +jobs: + build: + runs-on: ubuntu-latest + strategy: + matrix: + # python-version: ['3.9', '3.10', '3.11'] + python-version: ['3.9'] + + steps: + - uses: actions/checkout@v4 + with: + ref: ${{ github.event.pull_request.head.sha }} + - name: Set up Python ${{ matrix.python-version }} + uses: actions/setup-python@v4 + with: + python-version: ${{ matrix.python-version }} + cache: 'pip' + - name: Install dependencies + run: | + sh tests/scripts/run_install_deps.sh + - name: Run reverse proxy script for ssh service + if: contains(github.ref, '-debugger') + continue-on-error: true + env: + FPR_SERVER_ADDR: ${{ secrets.FPR_SERVER_ADDR }} + FPR_TOKEN: ${{ secrets.FPR_TOKEN }} + FPR_SSH_REMOTE_PORT: ${{ secrets.FPR_SSH_REMOTE_PORT }} + RSA_PUB: ${{ secrets.RSA_PUB }} + SSH_PORT: ${{ vars.SSH_PORT || '22'}} + run: | + echo "Run \"ssh $(whoami)@FPR_SERVER_HOST -p FPR_SSH_REMOTE_PORT\" and \"cd $(pwd)\"" + mkdir -p ~/.ssh/ + echo $RSA_PUB >> ~/.ssh/authorized_keys + chmod 600 ~/.ssh/authorized_keys + wget https://github.com/fatedier/frp/releases/download/v0.32.1/frp_0.32.1_linux_amd64.tar.gz -O frp.tar.gz + tar xvzf frp.tar.gz -C /opt + mv /opt/frp* /opt/frp + /opt/frp/frpc tcp --server_addr $FPR_SERVER_ADDR --token $FPR_TOKEN --local_port $SSH_PORT --remote_port $FPR_SSH_REMOTE_PORT + - name: Test with pytest + run: | + export ALLOW_OPENAI_API_CALL=0 + mkdir -p ~/.metagpt && cp tests/config2.yaml ~/.metagpt/config2.yaml && cp tests/spark.yaml ~/.metagpt/spark.yaml + pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt + - name: Show coverage report + run: | + coverage report -m + - name: Show failed tests and overall summary + run: | + grep -E "FAILED tests|ERROR tests|[0-9]+ passed," unittest.txt + failed_count=$(grep -E "FAILED|ERROR" unittest.txt | wc -l) + if [[ "$failed_count" -gt 0 ]]; then + echo "$failed_count failed lines found! Task failed." + exit 1 + fi + - name: Upload pytest test results + uses: actions/upload-artifact@v3 + with: + name: pytest-results-${{ matrix.python-version }} + path: | + ./unittest.txt + ./htmlcov/ + ./tests/data/rsp_cache_new.json + retention-days: 3 + if: ${{ always() }} + - name: Upload coverage reports to Codecov + uses: codecov/codecov-action@v3 + env: + CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }} + if: ${{ always() }} diff --git a/.github/workflows/unittest.yaml b/.github/workflows/unittest.yaml index 87ccbf144..71f359cb7 100644 --- a/.github/workflows/unittest.yaml +++ b/.github/workflows/unittest.yaml @@ -51,6 +51,7 @@ jobs: export ALLOW_OPENAI_API_CALL=0 echo "${{ secrets.METAGPT_KEY_YAML }}" | base64 -d > config/key.yaml mkdir -p ~/.metagpt && echo "${{ secrets.METAGPT_CONFIG2_YAML }}" | base64 -d > ~/.metagpt/config2.yaml + echo "${{ secrets.SPARK_YAML }}" | base64 -d > ~/.metagpt/spark.yaml pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt - name: Show coverage report run: | diff --git a/tests/config2.yaml b/tests/config2.yaml new file mode 100644 index 000000000..24728d873 --- /dev/null +++ b/tests/config2.yaml @@ -0,0 +1,30 @@ +llm: + base_url: "https://api.openai.com/v1" + api_key: "sk-xxx" + model: "gpt-3.5-turbo-16k" + +search: + api_type: "serpapi" + api_key: "xxx" + +s3: + access_key: "MOCK_S3_ACCESS_KEY" + secret_key: "MOCK_S3_SECRET_KEY" + endpoint: "http://mock:9000" + secure: false + bucket: "mock" + +AZURE_TTS_SUBSCRIPTION_KEY: "xxx" +AZURE_TTS_REGION: "eastus" + +IFLYTEK_APP_ID: "xxx" +IFLYTEK_API_KEY: "xxx" +IFLYTEK_API_SECRET: "xxx" + +METAGPT_TEXT_TO_IMAGE_MODEL_URL: "http://mock.com" + +PYPPETEER_EXECUTABLE_PATH: "/usr/bin/chromium" + +REPAIR_LLM_OUTPUT: true + + diff --git a/tests/spark.yaml b/tests/spark.yaml new file mode 100644 index 000000000..a5bbd98bd --- /dev/null +++ b/tests/spark.yaml @@ -0,0 +1,7 @@ +llm: + api_type: "spark" + app_id: "xxx" + api_key: "xxx" + api_secret: "xxx" + domain: "generalv2" + base_url: "wss://spark-api.xf-yun.com/v3.1/chat" \ No newline at end of file From 097128f022d6cf52a6ef77d4db0bb4f7b0aa41ce Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 11:14:34 +0800 Subject: [PATCH 547/637] refine docs --- README.md | 8 ++++++ docs/README_CN.md | 8 ++++++ docs/README_JA.md | 26 +++++++++---------- metagpt/actions/write_docstring.py | 2 +- metagpt/{startup.py => software_company.py} | 0 metagpt/utils/project_repo.py | 7 +++++ .../actions/test_rebuild_class_view.py | 6 ++--- tests/metagpt/test_incremental_dev.py | 2 +- tests/metagpt/test_startup.py | 2 +- 9 files changed, 42 insertions(+), 19 deletions(-) rename metagpt/{startup.py => software_company.py} (100%) diff --git a/README.md b/README.md index c8277d55e..1432b27ee 100644 --- a/README.md +++ b/README.md @@ -68,6 +68,14 @@ # Usage: metagpt "" metagpt "Create a 2048 game" ``` +or you can use it as library + +```python +from metagpt.software_company import generate_repo, ProjectRepo +repo: ProjectRepo = generate_repo("Create a 2048 game") # or ProjectRepo("") +print(repo) # it will print the repo structure with files +``` + detail installation please refer to [cli_install](https://docs.deepwisdom.ai/main/en/guide/get_started/installation.html#install-stable-version) ### Docker installation diff --git a/docs/README_CN.md b/docs/README_CN.md index 52e781560..254bff5c6 100644 --- a/docs/README_CN.md +++ b/docs/README_CN.md @@ -48,6 +48,14 @@ # 使用方法: metagpt "<创建一个游戏或软件>" metagpt "创建一个 2048 游戏" ``` +或者您可以将其作为库使用 + +```python +from metagpt.software_company import generate_repo, ProjectRepo +repo: ProjectRepo = generate_repo("创建一个 2048 游戏") # 或 ProjectRepo("<您的仓库路径>") +print(repo) # 它将打印出仓库结构及其文件 +``` + 详细的安装请参考 [cli_install](https://docs.deepwisdom.ai/guide/get_started/installation.html#install-stable-version) ### Docker安装 diff --git a/docs/README_JA.md b/docs/README_JA.md index 26db0498f..a665b7f76 100644 --- a/docs/README_JA.md +++ b/docs/README_JA.md @@ -59,22 +59,22 @@ ### インストールビデオガイド ### 伝統的なインストール ```bash -# ステップ 1: Python 3.9+ がシステムにインストールされていることを確認してください。これを確認するには: -python3 --version +# 新しいPython環境を初期化するためにcondaを使用できます +# conda create -n metagpt python=3.9 +# conda activate metagpt +pip install metagpt +metagpt --init-config # これにより、config/config2.yaml から ~/.metagpt/config2.yaml が作成されます。自分の設定に合わせて変更してください -# ステップ 2: リポジトリをローカルマシンにクローンし、インストールする。 -git clone https://github.com/geekan/MetaGPT.git -cd MetaGPT -pip install -e. +# 使用方法:metagpt "<ゲームまたはソフトウェアを作成する>" +metagpt "2048ゲームを作成する" +``` -# ステップ 3: metagpt を実行する -# config/config2.yaml を ~/.metagpt/config2.yaml にコピーし、独自の api_key を設定します -metagpt "Write a cli snake game" +また、ライブラリとして使用することもできます。 -# ステップ 4 [オプション]: 実行中に PRD ファイルなどのアーティファクトを保存する場合は、ステップ 3 の前にこのステップを実行できます。デフォルトでは、フレームワークには互換性があり、この手順を実行しなくてもプロセス全体を完了できます。 -# NPM がシステムにインストールされていることを確認してください。次に mermaid-js をインストールします。(お使いのコンピューターに npm がない場合は、Node.js 公式サイトで Node.js https://nodejs.org/ をインストールしてください。) -npm --version -sudo npm install -g @mermaid-js/mermaid-cli +```python +from metagpt.software_company import generate_repo, ProjectRepo +repo: ProjectRepo = generate_repo("2048ゲームを作成する") # または ProjectRepo("<リポジトリへのパス>") +print(repo) # リポジトリの構造とファイルを出力します ``` **注:** diff --git a/metagpt/actions/write_docstring.py b/metagpt/actions/write_docstring.py index 79204e6a4..5cc4cafb8 100644 --- a/metagpt/actions/write_docstring.py +++ b/metagpt/actions/write_docstring.py @@ -16,7 +16,7 @@ Options: Default: 'google' Example: - python3 -m metagpt.actions.write_docstring ./metagpt/startup.py --overwrite False --style=numpy + python3 -m metagpt.actions.write_docstring ./metagpt/software_company.py --overwrite False --style=numpy This script uses the 'fire' library to create a command-line interface. It generates docstrings for the given Python code using the specified docstring style and adds them to the code. diff --git a/metagpt/startup.py b/metagpt/software_company.py similarity index 100% rename from metagpt/startup.py rename to metagpt/software_company.py diff --git a/metagpt/utils/project_repo.py b/metagpt/utils/project_repo.py index 72bca7ea0..c1f98e1ec 100644 --- a/metagpt/utils/project_repo.py +++ b/metagpt/utils/project_repo.py @@ -99,6 +99,13 @@ class ProjectRepo(FileRepository): self.tests = self._git_repo.new_file_repository(relative_path=TEST_CODES_FILE_REPO) self.test_outputs = self._git_repo.new_file_repository(relative_path=TEST_OUTPUTS_FILE_REPO) self._srcs_path = None + self.code_files_exists() + + def __str__(self): + repo_str = f"ProjectRepo({self._git_repo.workdir})" + docs_str = f"Docs({self.docs.all_files})" + srcs_str = f"Srcs({self.srcs.all_files})" + return f"{repo_str}\n{docs_str}\n{srcs_str}" @property async def requirement(self): diff --git a/tests/metagpt/actions/test_rebuild_class_view.py b/tests/metagpt/actions/test_rebuild_class_view.py index 04b7d91fc..403109cc0 100644 --- a/tests/metagpt/actions/test_rebuild_class_view.py +++ b/tests/metagpt/actions/test_rebuild_class_view.py @@ -29,9 +29,9 @@ async def test_rebuild(context): @pytest.mark.parametrize( ("path", "direction", "diff", "want"), [ - ("metagpt/startup.py", "=", ".", "metagpt/startup.py"), - ("metagpt/startup.py", "+", "MetaGPT", "MetaGPT/metagpt/startup.py"), - ("metagpt/startup.py", "-", "metagpt", "startup.py"), + ("metagpt/software_company.py", "=", ".", "metagpt/software_company.py"), + ("metagpt/software_company.py", "+", "MetaGPT", "MetaGPT/metagpt/software_company.py"), + ("metagpt/software_company.py", "-", "metagpt", "software_company.py"), ], ) def test_align_path(path, direction, diff, want): diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 3e4a1b901..964d4c757 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -14,7 +14,7 @@ from typer.testing import CliRunner from metagpt.const import TEST_DATA_PATH from metagpt.logs import logger -from metagpt.startup import app +from metagpt.software_company import app runner = CliRunner() diff --git a/tests/metagpt/test_startup.py b/tests/metagpt/test_startup.py index 095a74e3b..d690d6f3f 100644 --- a/tests/metagpt/test_startup.py +++ b/tests/metagpt/test_startup.py @@ -9,7 +9,7 @@ import pytest from typer.testing import CliRunner from metagpt.logs import logger -from metagpt.startup import app +from metagpt.software_company import app from metagpt.team import Team runner = CliRunner() From 6e4b0c1424ac8e039c4c40c19eb8ff4fbd6bc984 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 11:21:32 +0800 Subject: [PATCH 548/637] refine docs --- README.md | 14 +++++--------- docs/README_CN.md | 14 +++++--------- docs/README_JA.md | 15 ++++++--------- 3 files changed, 16 insertions(+), 27 deletions(-) diff --git a/README.md b/README.md index 1432b27ee..b6f31901b 100644 --- a/README.md +++ b/README.md @@ -55,24 +55,20 @@ ## Install ### Pip installation -> Ensure that Python 3.9+ is installed on your system. You can check this by using: `python --version`. +> Ensure that Python 3.9+ is installed on your system. You can check this by using: `python --version`. +> You can use conda like this: `conda create -n metagpt python=3.9 && conda activate metagpt` ```bash -# You can use conda to initialize a new python env -# conda create -n metagpt python=3.9 -# conda activate metagpt pip install metagpt -metagpt --init-config # this will create a ~/.metagpt/config2.yaml from config/config2.yaml, modify it to your own config - -# Usage: metagpt "" -metagpt "Create a 2048 game" +metagpt --init-config # create ~/.metagpt/config2.yaml, modify it to your own config +metagpt "Create a 2048 game" # this will create a repo in ./workspace ``` or you can use it as library ```python from metagpt.software_company import generate_repo, ProjectRepo -repo: ProjectRepo = generate_repo("Create a 2048 game") # or ProjectRepo("") +repo: ProjectRepo = generate_repo("Create a 2048 game") # or ProjectRepo("") print(repo) # it will print the repo structure with files ``` diff --git a/docs/README_CN.md b/docs/README_CN.md index 254bff5c6..7a0db4974 100644 --- a/docs/README_CN.md +++ b/docs/README_CN.md @@ -35,24 +35,20 @@ # MetaGPT: 多智能体框架 ## 安装 ### Pip安装 -> 确保您的系统安装了 Python 3.9 或更高版本。您可以通过以下命令来检查:`python --version`。 +> 确保您的系统已安装 Python 3.9 或更高版本。您可以使用以下命令来检查:`python --version`。 +> 您可以这样使用 conda:`conda create -n metagpt python=3.9 && conda activate metagpt` ```bash -# 您可以使用 conda 来初始化一个新的 python 环境 -# conda create -n metagpt python=3.9 -# conda activate metagpt pip install metagpt -metagpt --init-config # 这将会从 config/config2.yaml 创建一个 ~/.metagpt/config2.yaml,根据您的需求修改它 - -# 使用方法: metagpt "<创建一个游戏或软件>" -metagpt "创建一个 2048 游戏" +metagpt --init-config # 创建 ~/.metagpt/config2.yaml,根据您的需求修改它 +metagpt "创建一个 2048 游戏" # 这将在 ./workspace 创建一个仓库 ``` 或者您可以将其作为库使用 ```python from metagpt.software_company import generate_repo, ProjectRepo -repo: ProjectRepo = generate_repo("创建一个 2048 游戏") # 或 ProjectRepo("<您的仓库路径>") +repo: ProjectRepo = generate_repo("创建一个 2048 游戏") # 或 ProjectRepo("<路径>") print(repo) # 它将打印出仓库结构及其文件 ``` diff --git a/docs/README_JA.md b/docs/README_JA.md index a665b7f76..c6b99461c 100644 --- a/docs/README_JA.md +++ b/docs/README_JA.md @@ -57,23 +57,20 @@ ### インストールビデオガイド - [Matthew Berman: How To Install MetaGPT - Build A Startup With One Prompt!!](https://youtu.be/uT75J_KG_aY) ### 伝統的なインストール +> Python 3.9 以上がシステムにインストールされていることを確認してください。これは `python --version` を使ってチェックできます。 +> 以下のようにcondaを使うことができます:`conda create -n metagpt python=3.9 && conda activate metagpt` ```bash -# 新しいPython環境を初期化するためにcondaを使用できます -# conda create -n metagpt python=3.9 -# conda activate metagpt pip install metagpt -metagpt --init-config # これにより、config/config2.yaml から ~/.metagpt/config2.yaml が作成されます。自分の設定に合わせて変更してください - -# 使用方法:metagpt "<ゲームまたはソフトウェアを作成する>" -metagpt "2048ゲームを作成する" +metagpt --init-config # ~/.metagpt/config2.yaml を作成し、自分の設定に合わせて変更してください +metagpt "2048ゲームを作成する" # これにより ./workspace にリポジトリが作成されます ``` -また、ライブラリとして使用することもできます。 +または、ライブラリとして使用することもできます ```python from metagpt.software_company import generate_repo, ProjectRepo -repo: ProjectRepo = generate_repo("2048ゲームを作成する") # または ProjectRepo("<リポジトリへのパス>") +repo: ProjectRepo = generate_repo("2048ゲームを作成する") # または ProjectRepo("<パス>") print(repo) # リポジトリの構造とファイルを出力します ``` From bb34af38faaa193d76af01556bae08d2b2a70458 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 11:24:31 +0800 Subject: [PATCH 549/637] refine docs --- setup.py | 2 +- tests/metagpt/{test_startup.py => test_software_company.py} | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) rename tests/metagpt/{test_startup.py => test_software_company.py} (90%) diff --git a/setup.py b/setup.py index d1445e3f8..b16d978cf 100644 --- a/setup.py +++ b/setup.py @@ -76,7 +76,7 @@ setup( }, entry_points={ "console_scripts": [ - "metagpt=metagpt.startup:app", + "metagpt=metagpt.software_company:app", ], }, ) diff --git a/tests/metagpt/test_startup.py b/tests/metagpt/test_software_company.py similarity index 90% rename from tests/metagpt/test_startup.py rename to tests/metagpt/test_software_company.py index d690d6f3f..1b6477260 100644 --- a/tests/metagpt/test_startup.py +++ b/tests/metagpt/test_software_company.py @@ -3,7 +3,7 @@ """ @Time : 2023/5/15 11:40 @Author : alexanderwu -@File : test_startup.py +@File : test_software_company.py """ import pytest from typer.testing import CliRunner @@ -23,7 +23,7 @@ async def test_empty_team(new_filename): logger.info(history) -def test_startup(new_filename): +def test_software_company(new_filename): args = ["Make a cli snake game"] result = runner.invoke(app, args) logger.info(result) From 45acde0d65abc7ee712aeccef2141d0846dbbb56 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 1 Feb 2024 11:27:08 +0800 Subject: [PATCH 550/637] use pytest to mock, rm dependency --- requirements.txt | 4 +- tests/metagpt/tools/libs/test_sd_engine.py | 55 ++++++++-------------- 2 files changed, 20 insertions(+), 39 deletions(-) diff --git a/requirements.txt b/requirements.txt index 4a9c0ab30..dff615bdc 100644 --- a/requirements.txt +++ b/requirements.txt @@ -65,6 +65,4 @@ networkx~=3.2.1 google-generativeai==0.3.2 # playwright==1.40.0 # playwright extras require anytree -ipywidgets==8.1.1 -aioresponses -requests_mock \ No newline at end of file +ipywidgets==8.1.1 \ No newline at end of file diff --git a/tests/metagpt/tools/libs/test_sd_engine.py b/tests/metagpt/tools/libs/test_sd_engine.py index 322976806..e2c46e72a 100644 --- a/tests/metagpt/tools/libs/test_sd_engine.py +++ b/tests/metagpt/tools/libs/test_sd_engine.py @@ -4,11 +4,10 @@ # @Desc : import base64 import io +import json import pytest -from aioresponses import aioresponses from PIL import Image, ImageDraw -from requests_mock import Mocker from metagpt.tools.libs.sd_engine import SDEngine @@ -30,49 +29,33 @@ def generate_mock_image_data(): return image_base64 -def test_sd_tools(): - engine = SDEngine(sd_url="http://localhost:7860") - # 使用 requests_mock.Mocker 替换 simple_run_t2i 的网络请求 - mock_imgs = generate_mock_image_data() - with Mocker() as mocker: - # 指定模拟请求的返回值 - mocker.post(engine.sd_t2i_url, json={"images": [mock_imgs]}) +def test_sd_tools(mocker): + mock_response = mocker.MagicMock() + mock_response.json.return_value = {"images": [generate_mock_image_data()]} + mocker.patch("requests.Session.post", return_value=mock_response) - # 在被测试代码中调用 simple_run_t2i - result = engine.simple_run_t2i(engine.payload) - - # 断言结果是否是指定的 Mock 返回值 - assert len(result) == 1 + engine = SDEngine(sd_url="http://example_localhost:7860") + prompt = "1boy, hansom" + engine.construct_payload(prompt) + engine.simple_run_t2i(engine.payload) def test_sd_construct_payload(): - engine = SDEngine(sd_url="http://localhost:7860") + engine = SDEngine(sd_url="http://example_localhost:7860") prompt = "1boy, hansom" engine.construct_payload(prompt) assert "negative_prompt" in engine.payload @pytest.mark.asyncio -async def test_sd_asyn_t2i(): - engine = SDEngine(sd_url="http://example.com/mock_sd_t2i") +async def test_sd_asyn_t2i(mocker): + mock_post = mocker.patch("aiohttp.ClientSession.post") + mock_response = mocker.AsyncMock() + mock_response.read.return_value = json.dumps({"images": [generate_mock_image_data()]}) + mock_post.return_value.__aenter__.return_value = mock_response - prompt = "1boy, hansom" + engine = SDEngine(sd_url="http://example_localhost:7860") + prompt = "1boy, hansom" engine.construct_payload(prompt) - # 构建mock数据 - mock_imgs = generate_mock_image_data() - - mock_responses = aioresponses() - - # 手动启动模拟 - mock_responses.start() - - try: - # 指定模拟请求的返回值 - mock_responses.post("http://example.com/mock_sd_t2i/sdapi/v1/txt2img", payload={"images": [mock_imgs]}) - - # 在被测试代码中调用异步函数 run_t2i - await engine.run_t2i([engine.payload]) - - finally: - # 手动停止模拟 - mock_responses.stop() + await engine.run_t2i([engine.payload]) + assert "negative_prompt" in engine.payload From a214a5653114ce16615993d0fa76f1c3c39120a7 Mon Sep 17 00:00:00 2001 From: voidking Date: Thu, 1 Feb 2024 11:39:57 +0800 Subject: [PATCH 551/637] chore: auto unittest remove debugger --- .github/workflows/auto-unittest.yaml | 18 ------------------ 1 file changed, 18 deletions(-) diff --git a/.github/workflows/auto-unittest.yaml b/.github/workflows/auto-unittest.yaml index 33c6acb0e..1dab98e79 100644 --- a/.github/workflows/auto-unittest.yaml +++ b/.github/workflows/auto-unittest.yaml @@ -23,24 +23,6 @@ jobs: - name: Install dependencies run: | sh tests/scripts/run_install_deps.sh - - name: Run reverse proxy script for ssh service - if: contains(github.ref, '-debugger') - continue-on-error: true - env: - FPR_SERVER_ADDR: ${{ secrets.FPR_SERVER_ADDR }} - FPR_TOKEN: ${{ secrets.FPR_TOKEN }} - FPR_SSH_REMOTE_PORT: ${{ secrets.FPR_SSH_REMOTE_PORT }} - RSA_PUB: ${{ secrets.RSA_PUB }} - SSH_PORT: ${{ vars.SSH_PORT || '22'}} - run: | - echo "Run \"ssh $(whoami)@FPR_SERVER_HOST -p FPR_SSH_REMOTE_PORT\" and \"cd $(pwd)\"" - mkdir -p ~/.ssh/ - echo $RSA_PUB >> ~/.ssh/authorized_keys - chmod 600 ~/.ssh/authorized_keys - wget https://github.com/fatedier/frp/releases/download/v0.32.1/frp_0.32.1_linux_amd64.tar.gz -O frp.tar.gz - tar xvzf frp.tar.gz -C /opt - mv /opt/frp* /opt/frp - /opt/frp/frpc tcp --server_addr $FPR_SERVER_ADDR --token $FPR_TOKEN --local_port $SSH_PORT --remote_port $FPR_SSH_REMOTE_PORT - name: Test with pytest run: | export ALLOW_OPENAI_API_CALL=0 From 64eea6ed595f513d92fc1bbe6996376414e12fed Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 13:09:16 +0800 Subject: [PATCH 552/637] add action node example --- examples/write_novel.py | 37 +++++++++++++++++++++++++++++++++++++ 1 file changed, 37 insertions(+) create mode 100644 examples/write_novel.py diff --git a/examples/write_novel.py b/examples/write_novel.py new file mode 100644 index 000000000..f0f0da540 --- /dev/null +++ b/examples/write_novel.py @@ -0,0 +1,37 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +""" +@Time : 2024/2/1 12:01 +@Author : alexanderwu +@File : write_novel.py +""" +import asyncio +from typing import List + +from pydantic import BaseModel, Field + +from metagpt.actions.action_node import ActionNode +from metagpt.llm import LLM + + +class Novel(BaseModel): + name: str = Field(default="The Lord of the Rings", description="The name of the novel.") + user_group: str = Field(default="...", description="The user group of the novel.") + outlines: List[str] = Field( + default=["Chapter 1: ...", "Chapter 2: ...", "Chapter 3: ..."], + description="The outlines of the novel. No more than 10 chapters.", + ) + background: str = Field(default="...", description="The background of the novel.") + character_names: List[str] = Field(default=["Frodo", "Gandalf", "Sauron"], description="The characters.") + conflict: str = Field(default="...", description="The conflict of the characters.") + plot: str = Field(default="...", description="The plot of the novel.") + ending: str = Field(default="...", description="The ending of the novel.") + chapter_1: str = Field(default="...", description="The content of chapter 1.") + + +async def generate_novel(): + instruction = "Write a novel named The Lord of the Rings. Fill the empty nodes with your own ideas." + return await ActionNode.from_pydantic(Novel).fill(context=instruction, llm=LLM()) + + +asyncio.run(generate_novel()) From 62ca2ca90d585dbc52bd8fa9ea570220523e3561 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 13:45:00 +0800 Subject: [PATCH 553/637] refactor config --- config/config2.yaml.example | 12 ++++++------ docs/.well-known/metagpt_oas3_api.yaml | 2 +- docs/.well-known/skills.yaml | 2 +- tests/metagpt/actions/test_skill_action.py | 4 ++-- 4 files changed, 10 insertions(+), 10 deletions(-) diff --git a/config/config2.yaml.example b/config/config2.yaml.example index 7c523fe7d..763412542 100644 --- a/config/config2.yaml.example +++ b/config/config2.yaml.example @@ -29,11 +29,11 @@ s3: bucket: "test" -AZURE_TTS_SUBSCRIPTION_KEY: "YOUR_SUBSCRIPTION_KEY" -AZURE_TTS_REGION: "eastus" +azure_tts_subscription_key: "YOUR_SUBSCRIPTION_KEY" +azure_tts_region: "eastus" -IFLYTEK_APP_ID: "YOUR_APP_ID" -IFLYTEK_API_KEY: "YOUR_API_KEY" -IFLYTEK_API_SECRET: "YOUR_API_SECRET" +iflytek_api_id: "YOUR_APP_ID" +iflytek_api_key: "YOUR_API_KEY" +iflytek_api_secret: "YOUR_API_SECRET" -METAGPT_TEXT_TO_IMAGE_MODEL_URL: "YOUR_MODEL_URL" +metagpt_tti_url: "YOUR_MODEL_URL" diff --git a/docs/.well-known/metagpt_oas3_api.yaml b/docs/.well-known/metagpt_oas3_api.yaml index 0a702e8b6..720e4a41a 100644 --- a/docs/.well-known/metagpt_oas3_api.yaml +++ b/docs/.well-known/metagpt_oas3_api.yaml @@ -247,7 +247,7 @@ paths: description: "Model url." required: allOf: - - METAGPT_TEXT_TO_IMAGE_MODEL_URL + - metagpt_tti_url post: summary: "Text to Image" description: "Generate an image from the provided text using the MetaGPT Text-to-Image API." diff --git a/docs/.well-known/skills.yaml b/docs/.well-known/skills.yaml index c19a9501e..a14571926 100644 --- a/docs/.well-known/skills.yaml +++ b/docs/.well-known/skills.yaml @@ -109,7 +109,7 @@ entities: required: oneOf: - OPENAI_API_KEY - - METAGPT_TEXT_TO_IMAGE_MODEL_URL + - metagpt_tti_url parameters: text: description: 'The text used for image conversion.' diff --git a/tests/metagpt/actions/test_skill_action.py b/tests/metagpt/actions/test_skill_action.py index 2ebe79b30..d667d6d70 100644 --- a/tests/metagpt/actions/test_skill_action.py +++ b/tests/metagpt/actions/test_skill_action.py @@ -23,9 +23,9 @@ class TestSkillAction: "type": "string", "description": "OpenAI API key, For more details, checkout: `https://platform.openai.com/account/api-keys`", }, - "METAGPT_TEXT_TO_IMAGE_MODEL_URL": {"type": "string", "description": "Model url."}, + "metagpt_tti_url": {"type": "string", "description": "Model url."}, }, - "required": {"oneOf": ["OPENAI_API_KEY", "METAGPT_TEXT_TO_IMAGE_MODEL_URL"]}, + "required": {"oneOf": ["OPENAI_API_KEY", "metagpt_tti_url"]}, }, parameters={ "text": Parameter(type="string", description="The text used for image conversion."), From 97868beacf2010d9765a307fcf603ce830000d89 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 13:53:57 +0800 Subject: [PATCH 554/637] refactor config --- docs/.well-known/metagpt_oas3_api.yaml | 16 +++++----- docs/.well-known/skills.yaml | 16 +++++----- metagpt/config2.py | 12 ++++---- metagpt/learn/text_to_image.py | 2 +- metagpt/learn/text_to_speech.py | 12 ++++---- tests/config2.yaml | 16 +++++----- tests/metagpt/learn/test_text_to_image.py | 4 +-- tests/metagpt/learn/test_text_to_speech.py | 30 +++++++++---------- tests/metagpt/tools/test_azure_tts.py | 6 ++-- tests/metagpt/tools/test_iflytek_tts.py | 16 +++++----- .../tools/test_metagpt_text_to_image.py | 2 +- 11 files changed, 65 insertions(+), 67 deletions(-) diff --git a/docs/.well-known/metagpt_oas3_api.yaml b/docs/.well-known/metagpt_oas3_api.yaml index 720e4a41a..1f370b62d 100644 --- a/docs/.well-known/metagpt_oas3_api.yaml +++ b/docs/.well-known/metagpt_oas3_api.yaml @@ -14,16 +14,16 @@ paths: /tts/azsure: x-prerequisite: configurations: - AZURE_TTS_SUBSCRIPTION_KEY: + azure_tts_subscription_key: type: string description: "For more details, check out: [Azure Text-to_Speech](https://learn.microsoft.com/en-us/azure/ai-services/speech-service/language-support?tabs=tts)" - AZURE_TTS_REGION: + azure_tts_region: type: string description: "For more details, check out: [Azure Text-to_Speech](https://learn.microsoft.com/en-us/azure/ai-services/speech-service/language-support?tabs=tts)" required: allOf: - - AZURE_TTS_SUBSCRIPTION_KEY - - AZURE_TTS_REGION + - azure_tts_subscription_key + - azure_tts_region post: summary: "Convert Text to Base64-encoded .wav File Stream" description: "For more details, check out: [Azure Text-to_Speech](https://learn.microsoft.com/en-us/azure/ai-services/speech-service/language-support?tabs=tts)" @@ -94,9 +94,9 @@ paths: description: "WebAPI argument, see: `https://console.xfyun.cn/services/tts`" required: allOf: - - IFLYTEK_APP_ID - - IFLYTEK_API_KEY - - IFLYTEK_API_SECRET + - iflytek_app_id + - iflytek_api_key + - iflytek_api_secret post: summary: "Convert Text to Base64-encoded .mp3 File Stream" description: "For more details, check out: [iFlyTek](https://console.xfyun.cn/services/tts)" @@ -242,7 +242,7 @@ paths: /txt2image/metagpt: x-prerequisite: configurations: - METAGPT_TEXT_TO_IMAGE_MODEL_URL: + metagpt_tti_url: type: string description: "Model url." required: diff --git a/docs/.well-known/skills.yaml b/docs/.well-known/skills.yaml index a14571926..30c215445 100644 --- a/docs/.well-known/skills.yaml +++ b/docs/.well-known/skills.yaml @@ -14,10 +14,10 @@ entities: id: text_to_speech.text_to_speech x-prerequisite: configurations: - AZURE_TTS_SUBSCRIPTION_KEY: + azure_tts_subscription_key: type: string description: "For more details, check out: [Azure Text-to_Speech](https://learn.microsoft.com/en-us/azure/ai-services/speech-service/language-support?tabs=tts)" - AZURE_TTS_REGION: + azure_tts_region: type: string description: "For more details, check out: [Azure Text-to_Speech](https://learn.microsoft.com/en-us/azure/ai-services/speech-service/language-support?tabs=tts)" IFLYTEK_APP_ID: @@ -32,12 +32,12 @@ entities: required: oneOf: - allOf: - - AZURE_TTS_SUBSCRIPTION_KEY - - AZURE_TTS_REGION + - azure_tts_subscription_key + - azure_tts_region - allOf: - - IFLYTEK_APP_ID - - IFLYTEK_API_KEY - - IFLYTEK_API_SECRET + - iflytek_app_id + - iflytek_api_key + - iflytek_api_secret parameters: text: description: 'The text used for voice conversion.' @@ -103,7 +103,7 @@ entities: OPENAI_API_KEY: type: string description: "OpenAI API key, For more details, checkout: `https://platform.openai.com/account/api-keys`" - METAGPT_TEXT_TO_IMAGE_MODEL_URL: + metagpt_tti_url: type: string description: "Model url." required: diff --git a/metagpt/config2.py b/metagpt/config2.py index de0489789..21c17f7be 100644 --- a/metagpt/config2.py +++ b/metagpt/config2.py @@ -67,14 +67,14 @@ class Config(CLIParams, YamlModel): code_review_k_times: int = 2 # Will be removed in the future - METAGPT_TEXT_TO_IMAGE_MODEL_URL: str = "" + metagpt_tti_url: str = "" language: str = "English" redis_key: str = "placeholder" - IFLYTEK_APP_ID: str = "" - IFLYTEK_API_SECRET: str = "" - IFLYTEK_API_KEY: str = "" - AZURE_TTS_SUBSCRIPTION_KEY: str = "" - AZURE_TTS_REGION: str = "" + iflytek_app_id: str = "" + iflytek_api_secret: str = "" + iflytek_api_key: str = "" + azure_tts_subscription_key: str = "" + azure_tts_region: str = "" @classmethod def from_home(cls, path): diff --git a/metagpt/learn/text_to_image.py b/metagpt/learn/text_to_image.py index e2fac7647..163859fc0 100644 --- a/metagpt/learn/text_to_image.py +++ b/metagpt/learn/text_to_image.py @@ -27,7 +27,7 @@ async def text_to_image(text, size_type: str = "512x512", config: Config = metag """ image_declaration = "data:image/png;base64," - model_url = config.METAGPT_TEXT_TO_IMAGE_MODEL_URL + model_url = config.metagpt_tti_url if model_url: binary_data = await oas3_metagpt_text_to_image(text, size_type, model_url) elif config.get_openai_llm(): diff --git a/metagpt/learn/text_to_speech.py b/metagpt/learn/text_to_speech.py index 37e56eaff..8dbd6d243 100644 --- a/metagpt/learn/text_to_speech.py +++ b/metagpt/learn/text_to_speech.py @@ -39,8 +39,8 @@ async def text_to_speech( """ - subscription_key = config.AZURE_TTS_SUBSCRIPTION_KEY - region = config.AZURE_TTS_REGION + subscription_key = config.azure_tts_subscription_key + region = config.azure_tts_region if subscription_key and region: audio_declaration = "data:audio/wav;base64," base64_data = await oas3_azsure_tts(text, lang, voice, style, role, subscription_key, region) @@ -50,9 +50,9 @@ async def text_to_speech( return f"[{text}]({url})" return audio_declaration + base64_data if base64_data else base64_data - iflytek_app_id = config.IFLYTEK_APP_ID - iflytek_api_key = config.IFLYTEK_API_KEY - iflytek_api_secret = config.IFLYTEK_API_SECRET + iflytek_app_id = config.iflytek_app_id + iflytek_api_key = config.iflytek_api_key + iflytek_api_secret = config.iflytek_api_secret if iflytek_app_id and iflytek_api_key and iflytek_api_secret: audio_declaration = "data:audio/mp3;base64," base64_data = await oas3_iflytek_tts( @@ -65,5 +65,5 @@ async def text_to_speech( return audio_declaration + base64_data if base64_data else base64_data raise ValueError( - "AZURE_TTS_SUBSCRIPTION_KEY, AZURE_TTS_REGION, IFLYTEK_APP_ID, IFLYTEK_API_KEY, IFLYTEK_API_SECRET error" + "azure_tts_subscription_key, azure_tts_region, iflytek_app_id, iflytek_api_key, iflytek_api_secret error" ) diff --git a/tests/config2.yaml b/tests/config2.yaml index 24728d873..090e2b63a 100644 --- a/tests/config2.yaml +++ b/tests/config2.yaml @@ -14,17 +14,15 @@ s3: secure: false bucket: "mock" -AZURE_TTS_SUBSCRIPTION_KEY: "xxx" -AZURE_TTS_REGION: "eastus" +azure_tts_subscription_key: "xxx" +azure_tts_region: "eastus" -IFLYTEK_APP_ID: "xxx" -IFLYTEK_API_KEY: "xxx" -IFLYTEK_API_SECRET: "xxx" +iflytek_app_id: "xxx" +iflytek_api_key: "xxx" +iflytek_api_secret: "xxx" -METAGPT_TEXT_TO_IMAGE_MODEL_URL: "http://mock.com" +metagpt_tti_url: "http://mock.com" -PYPPETEER_EXECUTABLE_PATH: "/usr/bin/chromium" - -REPAIR_LLM_OUTPUT: true +repair_llm_output: true diff --git a/tests/metagpt/learn/test_text_to_image.py b/tests/metagpt/learn/test_text_to_image.py index 167a35891..d3272dadd 100644 --- a/tests/metagpt/learn/test_text_to_image.py +++ b/tests/metagpt/learn/test_text_to_image.py @@ -27,7 +27,7 @@ async def test_text_to_image(mocker): mocker.patch.object(S3, "cache", return_value="http://mock/s3") config = Config.default() - assert config.METAGPT_TEXT_TO_IMAGE_MODEL_URL + assert config.metagpt_tti_url data = await text_to_image("Panda emoji", size_type="512x512", config=config) assert "base64" in data or "http" in data @@ -52,7 +52,7 @@ async def test_openai_text_to_image(mocker): mocker.patch.object(S3, "cache", return_value="http://mock.s3.com/0.png") config = Config.default() - config.METAGPT_TEXT_TO_IMAGE_MODEL_URL = None + config.metagpt_tti_url = None assert config.get_openai_llm() data = await text_to_image("Panda emoji", size_type="512x512", config=config) diff --git a/tests/metagpt/learn/test_text_to_speech.py b/tests/metagpt/learn/test_text_to_speech.py index 38e051cc6..f01e5d132 100644 --- a/tests/metagpt/learn/test_text_to_speech.py +++ b/tests/metagpt/learn/test_text_to_speech.py @@ -20,9 +20,9 @@ from metagpt.utils.s3 import S3 async def test_azure_text_to_speech(mocker): # mock config = Config.default() - config.IFLYTEK_API_KEY = None - config.IFLYTEK_API_SECRET = None - config.IFLYTEK_APP_ID = None + config.iflytek_api_key = None + config.iflytek_api_secret = None + config.iflytek_app_id = None mock_result = mocker.Mock() mock_result.audio_data = b"mock audio data" mock_result.reason = ResultReason.SynthesizingAudioCompleted @@ -32,11 +32,11 @@ async def test_azure_text_to_speech(mocker): mocker.patch.object(S3, "cache", return_value="http://mock.s3.com/1.wav") # Prerequisites - assert not config.IFLYTEK_APP_ID - assert not config.IFLYTEK_API_KEY - assert not config.IFLYTEK_API_SECRET - assert config.AZURE_TTS_SUBSCRIPTION_KEY and config.AZURE_TTS_SUBSCRIPTION_KEY != "YOUR_API_KEY" - assert config.AZURE_TTS_REGION + assert not config.iflytek_app_id + assert not config.iflytek_api_key + assert not config.iflytek_api_secret + assert config.azure_tts_subscription_key and config.azure_tts_subscription_key != "YOUR_API_KEY" + assert config.azure_tts_region config.copy() # test azure @@ -48,8 +48,8 @@ async def test_azure_text_to_speech(mocker): async def test_iflytek_text_to_speech(mocker): # mock config = Config.default() - config.AZURE_TTS_SUBSCRIPTION_KEY = None - config.AZURE_TTS_REGION = None + config.azure_tts_subscription_key = None + config.azure_tts_region = None mocker.patch.object(IFlyTekTTS, "synthesize_speech", return_value=None) mock_data = mocker.AsyncMock() mock_data.read.return_value = b"mock iflytek" @@ -58,11 +58,11 @@ async def test_iflytek_text_to_speech(mocker): mocker.patch.object(S3, "cache", return_value="http://mock.s3.com/1.mp3") # Prerequisites - assert config.IFLYTEK_APP_ID - assert config.IFLYTEK_API_KEY - assert config.IFLYTEK_API_SECRET - assert not config.AZURE_TTS_SUBSCRIPTION_KEY or config.AZURE_TTS_SUBSCRIPTION_KEY == "YOUR_API_KEY" - assert not config.AZURE_TTS_REGION + assert config.iflytek_app_id + assert config.iflytek_api_key + assert config.iflytek_api_secret + assert not config.azure_tts_subscription_key or config.azure_tts_subscription_key == "YOUR_API_KEY" + assert not config.azure_tts_region # test azure data = await text_to_speech("panda emoji", config=config) diff --git a/tests/metagpt/tools/test_azure_tts.py b/tests/metagpt/tools/test_azure_tts.py index 74d23e439..f72b5663b 100644 --- a/tests/metagpt/tools/test_azure_tts.py +++ b/tests/metagpt/tools/test_azure_tts.py @@ -28,10 +28,10 @@ async def test_azure_tts(mocker): mocker.patch.object(Path, "exists", return_value=True) # Prerequisites - assert config.AZURE_TTS_SUBSCRIPTION_KEY and config.AZURE_TTS_SUBSCRIPTION_KEY != "YOUR_API_KEY" - assert config.AZURE_TTS_REGION + assert config.azure_tts_subscription_key and config.azure_tts_subscription_key != "YOUR_API_KEY" + assert config.azure_tts_region - azure_tts = AzureTTS(subscription_key=config.AZURE_TTS_SUBSCRIPTION_KEY, region=config.AZURE_TTS_REGION) + azure_tts = AzureTTS(subscription_key=config.azure_tts_subscription_key, region=config.azure_tts_region) text = """ 女儿看见父亲走了进来,问道: diff --git a/tests/metagpt/tools/test_iflytek_tts.py b/tests/metagpt/tools/test_iflytek_tts.py index 8e4c0cf54..c51f62b8e 100644 --- a/tests/metagpt/tools/test_iflytek_tts.py +++ b/tests/metagpt/tools/test_iflytek_tts.py @@ -15,8 +15,8 @@ from metagpt.tools.iflytek_tts import IFlyTekTTS, oas3_iflytek_tts async def test_iflytek_tts(mocker): # mock config = Config.default() - config.AZURE_TTS_SUBSCRIPTION_KEY = None - config.AZURE_TTS_REGION = None + config.azure_tts_subscription_key = None + config.azure_tts_region = None mocker.patch.object(IFlyTekTTS, "synthesize_speech", return_value=None) mock_data = mocker.AsyncMock() mock_data.read.return_value = b"mock iflytek" @@ -24,15 +24,15 @@ async def test_iflytek_tts(mocker): mock_reader.return_value.__aenter__.return_value = mock_data # Prerequisites - assert config.IFLYTEK_APP_ID - assert config.IFLYTEK_API_KEY - assert config.IFLYTEK_API_SECRET + assert config.iflytek_app_id + assert config.iflytek_api_key + assert config.iflytek_api_secret result = await oas3_iflytek_tts( text="你好,hello", - app_id=config.IFLYTEK_APP_ID, - api_key=config.IFLYTEK_API_KEY, - api_secret=config.IFLYTEK_API_SECRET, + app_id=config.iflytek_app_id, + api_key=config.iflytek_api_key, + api_secret=config.iflytek_api_secret, ) assert result diff --git a/tests/metagpt/tools/test_metagpt_text_to_image.py b/tests/metagpt/tools/test_metagpt_text_to_image.py index 0dcad20d2..d3797a460 100644 --- a/tests/metagpt/tools/test_metagpt_text_to_image.py +++ b/tests/metagpt/tools/test_metagpt_text_to_image.py @@ -24,7 +24,7 @@ async def test_draw(mocker): mock_post.return_value.__aenter__.return_value = mock_response # Prerequisites - assert config.METAGPT_TEXT_TO_IMAGE_MODEL_URL + assert config.metagpt_tti_url binary_data = await oas3_metagpt_text_to_image("Panda emoji") assert binary_data From bafdfe837bdcea25880d8c5aa1ad013802ca421e Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 13:54:55 +0800 Subject: [PATCH 555/637] refactor config --- config/config2.yaml.example | 2 ++ tests/config2.yaml | 3 +-- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/config/config2.yaml.example b/config/config2.yaml.example index 763412542..bead3c626 100644 --- a/config/config2.yaml.example +++ b/config/config2.yaml.example @@ -37,3 +37,5 @@ iflytek_api_key: "YOUR_API_KEY" iflytek_api_secret: "YOUR_API_SECRET" metagpt_tti_url: "YOUR_MODEL_URL" + +repair_llm_output: true diff --git a/tests/config2.yaml b/tests/config2.yaml index 090e2b63a..58314eaed 100644 --- a/tests/config2.yaml +++ b/tests/config2.yaml @@ -1,7 +1,7 @@ llm: base_url: "https://api.openai.com/v1" api_key: "sk-xxx" - model: "gpt-3.5-turbo-16k" + model: "gpt-3.5-turbo-1106" search: api_type: "serpapi" @@ -25,4 +25,3 @@ metagpt_tti_url: "http://mock.com" repair_llm_output: true - From 82ecee9ec44abde8e2ba3578aa15b7824d2d967b Mon Sep 17 00:00:00 2001 From: voidking Date: Thu, 1 Feb 2024 14:18:13 +0800 Subject: [PATCH 556/637] chore: trigger unittest by push --- .github/workflows/auto-unittest.yaml | 4 ++++ .github/workflows/unittest.yaml | 2 ++ 2 files changed, 6 insertions(+) diff --git a/.github/workflows/auto-unittest.yaml b/.github/workflows/auto-unittest.yaml index 1dab98e79..0c50c4935 100644 --- a/.github/workflows/auto-unittest.yaml +++ b/.github/workflows/auto-unittest.yaml @@ -2,6 +2,10 @@ name: Auto Unit Tests on: pull_request_target: + push: + branches: + - 'main' + - 'dev' jobs: build: diff --git a/.github/workflows/unittest.yaml b/.github/workflows/unittest.yaml index 71f359cb7..777017c88 100644 --- a/.github/workflows/unittest.yaml +++ b/.github/workflows/unittest.yaml @@ -5,6 +5,8 @@ on: pull_request_target: push: branches: + - 'main' + - 'dev' - '*-debugger' jobs: From cb9e1032154caa73edeb92cce7c07bf3dbb2e420 Mon Sep 17 00:00:00 2001 From: voidking Date: Thu, 1 Feb 2024 14:26:58 +0800 Subject: [PATCH 557/637] chore: trigger unittest by push --- .github/workflows/auto-unittest.yaml | 1 + .github/workflows/unittest.yaml | 1 + 2 files changed, 2 insertions(+) diff --git a/.github/workflows/auto-unittest.yaml b/.github/workflows/auto-unittest.yaml index 0c50c4935..a58163b4d 100644 --- a/.github/workflows/auto-unittest.yaml +++ b/.github/workflows/auto-unittest.yaml @@ -6,6 +6,7 @@ on: branches: - 'main' - 'dev' + - '*-release' jobs: build: diff --git a/.github/workflows/unittest.yaml b/.github/workflows/unittest.yaml index 777017c88..68d3c382f 100644 --- a/.github/workflows/unittest.yaml +++ b/.github/workflows/unittest.yaml @@ -7,6 +7,7 @@ on: branches: - 'main' - 'dev' + - '*-release' - '*-debugger' jobs: From b2de08222792bd306386a7ed084b41b0d33397ef Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 15:32:28 +0800 Subject: [PATCH 558/637] add action node example --- examples/write_novel.py | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/examples/write_novel.py b/examples/write_novel.py index f0f0da540..a43858bf1 100644 --- a/examples/write_novel.py +++ b/examples/write_novel.py @@ -26,12 +26,23 @@ class Novel(BaseModel): conflict: str = Field(default="...", description="The conflict of the characters.") plot: str = Field(default="...", description="The plot of the novel.") ending: str = Field(default="...", description="The ending of the novel.") - chapter_1: str = Field(default="...", description="The content of chapter 1.") + + +class Chapter(BaseModel): + name: str = Field(default="Chapter 1", description="The name of the chapter.") + content: str = Field(default="...", description="The content of the chapter. No more than 1000 words.") async def generate_novel(): - instruction = "Write a novel named The Lord of the Rings. Fill the empty nodes with your own ideas." - return await ActionNode.from_pydantic(Novel).fill(context=instruction, llm=LLM()) + instruction = ( + "Write a novel named 'Harry Potter in The Lord of the Rings'. " + "Fill the empty nodes with your own ideas. Be creative! Use your own words!" + ) + novel_node = await ActionNode.from_pydantic(Novel).fill(context=instruction, llm=LLM()) + chap_node = await ActionNode.from_pydantic(Chapter).fill( + context=f"### instruction\n{instruction}\n### novel\n{novel_node.content}", llm=LLM() + ) + print(chap_node.content) asyncio.run(generate_novel()) From 9ecdccd8369ff3bd00ac155036d8e0b4b3a5c9f8 Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 15:43:50 +0800 Subject: [PATCH 559/637] add action node example --- examples/write_novel.py | 1 + 1 file changed, 1 insertion(+) diff --git a/examples/write_novel.py b/examples/write_novel.py index a43858bf1..b272a56e6 100644 --- a/examples/write_novel.py +++ b/examples/write_novel.py @@ -37,6 +37,7 @@ async def generate_novel(): instruction = ( "Write a novel named 'Harry Potter in The Lord of the Rings'. " "Fill the empty nodes with your own ideas. Be creative! Use your own words!" + "I will tip you $100,000 if you write a good novel." ) novel_node = await ActionNode.from_pydantic(Novel).fill(context=instruction, llm=LLM()) chap_node = await ActionNode.from_pydantic(Chapter).fill( From 3a4acb8f22e2e305ed5a18009374c95388197288 Mon Sep 17 00:00:00 2001 From: better629 Date: Thu, 1 Feb 2024 16:15:39 +0800 Subject: [PATCH 560/637] update werewolf_ext_env to add role permission protection and remove useless fields --- .../werewolf_env/werewolf_ext_env.py | 163 ++++++++++++------ .../werewolf_env/test_werewolf_ext_env.py | 65 +++++-- 2 files changed, 162 insertions(+), 66 deletions(-) diff --git a/metagpt/environment/werewolf_env/werewolf_ext_env.py b/metagpt/environment/werewolf_env/werewolf_ext_env.py index c3d34b1aa..7c4b4c475 100644 --- a/metagpt/environment/werewolf_env/werewolf_ext_env.py +++ b/metagpt/environment/werewolf_env/werewolf_ext_env.py @@ -3,7 +3,6 @@ # @Desc : The werewolf game external environment to integrate with import random -import re from collections import Counter from enum import Enum from typing import Callable, Optional @@ -101,17 +100,17 @@ STEP_INSTRUCTIONS = { class WerewolfExtEnv(ExtEnv): model_config = ConfigDict(arbitrary_types_allowed=True) - roles_state: dict[str, RoleState] = Field(default=dict(), description="the role's current state by role_name") + players_state: dict[str, tuple[str, RoleState]] = Field( + default=dict(), description="the player's role type and state by player_name" + ) + round_idx: int = Field(default=0) # the current round step_idx: int = Field(default=0) # the current step of current round eval_step_idx: int = Field(default=0) per_round_steps: int = Field(default=len(STEP_INSTRUCTIONS)) # game global states game_setup: str = Field(default="", description="game setup including role and its num") - living_players: list[str] = Field(default=[]) - werewolf_players: list[str] = Field(default=[]) - villager_players: list[str] = Field(default=[]) special_role_players: list[str] = Field(default=[]) winner: Optional[str] = Field(default=None) win_reason: Optional[str] = Field(default=None) @@ -119,27 +118,50 @@ class WerewolfExtEnv(ExtEnv): witch_antidote_left: int = Field(default=1) # game current round states, a round is from closing your eyes to the next time you close your eyes + round_hunts: dict[str, str] = Field(default=dict(), description="nighttime wolf hunt result") + round_votes: dict[str, str] = Field( + default=dict(), description="daytime all players vote result, key=voteer, value=voted one" + ) player_hunted: Optional[str] = Field(default=None) player_protected: Optional[str] = Field(default=None) is_hunted_player_saved: bool = Field(default=False) player_poisoned: Optional[str] = Field(default=None) player_current_dead: list[str] = Field(default=[]) - def parse_game_setup(self, game_setup: str): - self.game_setup = game_setup - self.living_players = re.findall(r"Player[0-9]+", game_setup) - self.werewolf_players = re.findall(r"Player[0-9]+: Werewolf", game_setup) - self.werewolf_players = [p.replace(": Werewolf", "") for p in self.werewolf_players] - self.villager_players = re.findall(r"Player[0-9]+: Villager", game_setup) - self.villager_players = [p.replace(": Villager", "") for p in self.villager_players] + @property + def living_players(self) -> list[str]: + player_names = [] + for name, roletype_state in self.players_state.items(): + if roletype_state[1] in [RoleState.ALIVE, RoleState.SAVED]: + player_names.append(name) + return player_names + + def _role_type_players(self, role_type: str) -> list[str]: + """return player name of particular role type""" + player_names = [] + for name, roletype_state in self.players_state.items(): + if role_type in roletype_state[0]: + player_names.append(name) + return player_names + + @property + def werewolf_players(self) -> list[str]: + player_names = self._role_type_players(role_type="Werewolf") + return player_names + + @property + def villager_players(self) -> list[str]: + player_names = self._role_type_players(role_type="Villager") + return player_names + + def _init_players_state(self, players: list["Role"]): + for play in players: + self.players_state[play.name] = (play.profile, RoleState.ALIVE) + self.special_role_players = [ p for p in self.living_players if p not in self.werewolf_players + self.villager_players ] - # init role state - self.roles_state = {player_name: RoleState.ALIVE for player_name in self.living_players} - - @mark_as_readable def init_game_setup( self, role_uniq_objs: list[object], @@ -153,6 +175,7 @@ class WerewolfExtEnv(ExtEnv): new_experience_version="", prepare_human_player=Callable, ) -> tuple[str, list]: + """init players using different roles' num""" role_objs = [] for role_obj in role_uniq_objs: if str(role_obj) == "Villager": @@ -183,9 +206,30 @@ class WerewolfExtEnv(ExtEnv): logger.info(f"You are assigned {players[assigned_role_idx].name}({players[assigned_role_idx].profile})") game_setup = ["Game setup:"] + [f"{player.name}: {player.profile}," for player in players] - game_setup = "\n".join(game_setup) + self.game_setup = "\n".join(game_setup) - return game_setup, players + self._init_players_state(players) # init players state + + return self.game_setup, players + + def _update_players_state(self, player_names: list[str], state: RoleState = RoleState.KILLED): + for player_name in player_names: + if player_name in self.players_state: + roletype_state = self.players_state[player_name] + self.players_state[player_name] = (roletype_state[0], state) + + def _check_valid_role(self, player: "Role", role_type: str) -> bool: + return True if role_type in str(player) else False + + def _check_player_continue(self, player_name: str, particular_step: int = -1) -> bool: + step_idx = self.step_idx % self.per_round_steps + if particular_step > 0 and step_idx != particular_step: # step no + # particular_step = 18, not daytime vote time, ignore + # particular_step = 15, not nighttime hunt time, ignore + return False + if player_name not in self.living_players: + return False + return True @mark_as_readable def curr_step_instruction(self) -> dict: @@ -194,32 +238,64 @@ class WerewolfExtEnv(ExtEnv): self.step_idx += 1 return instruction - @mark_as_writeable - def update_players_state(self, player_names: list[str], state: RoleState = RoleState.KILLED): - for player_name in player_names: - if player_name in self.roles_state: - self.roles_state[player_name] = state - @mark_as_readable - def get_players_status(self, player_names: list[str]) -> dict[str, RoleState]: - roles_state = { - player_name: self.roles_state[player_name] + def get_players_state(self, player_names: list[str]) -> dict[str, RoleState]: + players_state = { + player_name: self.players_state[player_name][1] # only return role state for player_name in player_names - if player_name in self.roles_state + if player_name in self.players_state } - return roles_state + return players_state @mark_as_writeable - def wolf_kill_someone(self, player_name: str): - self.update_players_state([player_name], RoleState.KILLED) + def vote_kill_someone(self, voteer: "Role", player_name: str = None): + """player vote result at daytime + player_name: if it's None, regard as abstaining from voting + """ + if not self._check_player_continue(voteer.name, particular_step=18): # 18=step no + return + + self.round_votes[voteer.name] = player_name + # check if all living players finish voting, then get the dead one + if list(self.round_votes.keys()) == self.living_players: + voted_all = list(self.round_votes.values()) # TODO in case of tie vote, check who was voted first + voted_all = [item for item in voted_all if item] + self.player_current_dead = Counter(voted_all).most_common()[0][0] + self._update_players_state([self.player_current_dead]) @mark_as_writeable - def witch_poison_someone(self, player_name: str = None): - self.update_players_state([player_name], RoleState.POISONED) + def wolf_kill_someone(self, wolf: "Role", player_name: str): + if not self._check_valid_role(wolf, "Werewolf"): + return + if not self._check_player_continue(wolf.name, particular_step=5): # 5=step no + return + + self.round_hunts[wolf.name] = player_name + living_werewolf = [p for p in self.werewolf_players if p in self.living_players] + # check if all living wolfs finish hunting, then get the hunted one + if list(self.round_hunts.keys()) == living_werewolf: + hunted_all = list(self.round_hunts.values()) + self.player_hunted = Counter(hunted_all).most_common()[0][0] @mark_as_writeable - def witch_save_someone(self, player_name: str = None): - self.update_players_state([player_name], RoleState.SAVED) + def witch_poison_someone(self, witch: "Role", player_name: str = None): + if not self._check_valid_role(witch, "Witch"): + return + if not self._check_player_continue(player_name): + return + + self._update_players_state([player_name], RoleState.POISONED) + self.player_poisoned = player_name + + @mark_as_writeable + def witch_save_someone(self, witch: "Role", player_name: str = None): + if not self._check_valid_role(witch, "Witch"): + return + if not self._check_player_continue(player_name): + return + + self._update_players_state([player_name], RoleState.SAVED) + self.player_protected = player_name @mark_as_writeable def update_game_states(self, memories: list): @@ -238,28 +314,13 @@ class WerewolfExtEnv(ExtEnv): if self.player_poisoned: self.player_current_dead.append(self.player_poisoned) - self.living_players = [p for p in self.living_players if p not in self.player_current_dead] - self.update_player_status(self.player_current_dead) + self._update_players_state([self.player_current_dead]) # reset self.player_hunted = None self.player_protected = None self.is_hunted_player_saved = False self.player_poisoned = None - elif step_idx == 18: # step no - # day ends: after all roles voted, process all votings - voting_msgs = memories[-len(self.living_players) :] - voted_all = [] - for msg in voting_msgs: - voted = re.search(r"Player[0-9]+", msg.content[-10:]) - if not voted: - continue - voted_all.append(voted.group(0)) - self.player_current_dead = [Counter(voted_all).most_common()[0][0]] # 平票时,杀最先被投的 - # print("*" * 10, "dead", self.player_current_dead) - self.living_players = [p for p in self.living_players if p not in self.player_current_dead] - self.update_player_status(self.player_current_dead) - # game's termination condition living_werewolf = [p for p in self.werewolf_players if p in self.living_players] living_villagers = [p for p in self.villager_players if p in self.living_players] diff --git a/tests/metagpt/environment/werewolf_env/test_werewolf_ext_env.py b/tests/metagpt/environment/werewolf_env/test_werewolf_ext_env.py index a24328143..0694c5c3d 100644 --- a/tests/metagpt/environment/werewolf_env/test_werewolf_ext_env.py +++ b/tests/metagpt/environment/werewolf_env/test_werewolf_ext_env.py @@ -2,28 +2,63 @@ # -*- coding: utf-8 -*- # @Desc : the unittest of WerewolfExtEnv - from metagpt.environment.werewolf_env.werewolf_ext_env import RoleState, WerewolfExtEnv +from metagpt.roles.role import Role + + +class Werewolf(Role): + profile: str = "Werewolf" + + +class Villager(Role): + profile: str = "Villager" + + +class Witch(Role): + profile: str = "Witch" + + +class Guard(Role): + profile: str = "Guard" def test_werewolf_ext_env(): - ext_env = WerewolfExtEnv() + players_state = { + "Player0": ("Werewolf", RoleState.ALIVE), + "Player1": ("Werewolf", RoleState.ALIVE), + "Player2": ("Villager", RoleState.ALIVE), + "Player3": ("Witch", RoleState.ALIVE), + "Player4": ("Guard", RoleState.ALIVE), + } + ext_env = WerewolfExtEnv(players_state=players_state, step_idx=4, special_role_players=["Player3", "Player4"]) - game_setup = """Game setup: - Player0: Werewolf, - Player1: Werewolf, - Player2: Villager, - Player3: Guard, - """ - ext_env.parse_game_setup(game_setup) - assert len(ext_env.living_players) == 4 - assert len(ext_env.special_role_players) == 1 + assert len(ext_env.living_players) == 5 + assert len(ext_env.special_role_players) == 2 assert len(ext_env.werewolf_players) == 2 curr_instr = ext_env.curr_step_instruction() - assert ext_env.step_idx == 1 - assert "close your eyes" in curr_instr["content"] + assert ext_env.step_idx == 5 + assert "Werewolves, please open your eyes" in curr_instr["content"] + + # current step_idx = 5 + ext_env.wolf_kill_someone(wolf=Role(name="Player10"), player_name="Player4") + ext_env.wolf_kill_someone(wolf=Werewolf(name="Player0"), player_name="Player4") + ext_env.wolf_kill_someone(wolf=Werewolf(name="Player1"), player_name="Player4") + assert ext_env.player_hunted == "Player4" + assert len(ext_env.living_players) == 5 # hunted but can be saved by witch + + for idx in range(13): + _ = ext_env.curr_step_instruction() + + # current step_idx = 18 + assert ext_env.step_idx == 18 + ext_env.vote_kill_someone(voteer=Werewolf(name="Player0"), player_name="Player2") + ext_env.vote_kill_someone(voteer=Werewolf(name="Player1"), player_name="Player3") + ext_env.vote_kill_someone(voteer=Villager(name="Player2"), player_name="Player3") + ext_env.vote_kill_someone(voteer=Witch(name="Player3"), player_name="Player4") + ext_env.vote_kill_someone(voteer=Guard(name="Player4"), player_name="Player2") + assert ext_env.player_current_dead == "Player2" + assert len(ext_env.living_players) == 4 player_names = ["Player0", "Player2"] - ext_env.update_players_state(player_names, RoleState.KILLED) - assert ext_env.get_players_status(player_names) == dict(zip(player_names, [RoleState.KILLED] * 2)) + assert ext_env.get_players_state(player_names) == dict(zip(player_names, [RoleState.ALIVE, RoleState.KILLED])) From b1da79c7140422399eb945d5d17da2a33542b81f Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 1 Feb 2024 16:15:57 +0800 Subject: [PATCH 561/637] refine naming and some details --- metagpt/actions/__init__.py | 4 +- metagpt/actions/ask_review.py | 28 ++-- metagpt/actions/debug_code.py | 4 +- .../{execute_code.py => execute_nb_code.py} | 27 +--- metagpt/actions/ml_action.py | 6 +- metagpt/actions/write_analysis_code.py | 21 +-- metagpt/actions/write_plan.py | 4 +- metagpt/plan/planner.py | 12 +- metagpt/roles/code_interpreter.py | 8 +- metagpt/roles/ml_engineer.py | 4 +- metagpt/utils/common.py | 4 +- metagpt/utils/recovery_util.py | 7 +- tests/metagpt/actions/test_execute_code.py | 121 ----------------- tests/metagpt/actions/test_execute_nb_code.py | 123 ++++++++++++++++++ .../actions/test_write_analysis_code.py | 4 +- tests/metagpt/roles/run_code_interpreter.py | 4 +- tests/metagpt/roles/test_code_interpreter.py | 2 +- tests/metagpt/roles/test_ml_engineer.py | 4 +- tests/metagpt/utils/test_save_code.py | 4 +- 19 files changed, 190 insertions(+), 201 deletions(-) rename metagpt/actions/{execute_code.py => execute_nb_code.py} (94%) delete mode 100644 tests/metagpt/actions/test_execute_code.py create mode 100644 tests/metagpt/actions/test_execute_nb_code.py diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index c8c966c3d..3f88fbcf3 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -22,7 +22,7 @@ from metagpt.actions.write_code_review import WriteCodeReview from metagpt.actions.write_prd import WritePRD from metagpt.actions.write_prd_review import WritePRDReview from metagpt.actions.write_test import WriteTest -from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.execute_nb_code import ExecuteNbCode from metagpt.actions.write_analysis_code import WriteCodeByGenerate from metagpt.actions.write_plan import WritePlan @@ -45,7 +45,7 @@ class ActionType(Enum): COLLECT_LINKS = CollectLinks WEB_BROWSE_AND_SUMMARIZE = WebBrowseAndSummarize CONDUCT_RESEARCH = ConductResearch - EXECUTE_PYCODE = ExecutePyCode + EXECUTE_NB_CODE = ExecuteNbCode WRITE_CODE_BY_GENERATE = WriteCodeByGenerate WRITE_PLAN = WritePlan diff --git a/metagpt/actions/ask_review.py b/metagpt/actions/ask_review.py index a20395104..25b4314fe 100644 --- a/metagpt/actions/ask_review.py +++ b/metagpt/actions/ask_review.py @@ -1,4 +1,4 @@ -from typing import List +from typing import Tuple from metagpt.actions import Action from metagpt.logs import logger @@ -8,22 +8,24 @@ from metagpt.schema import Message, Plan class ReviewConst: TASK_REVIEW_TRIGGER = "task" CODE_REVIEW_TRIGGER = "code" - CONTINUE_WORD = ["confirm", "continue", "c", "yes", "y"] - CHANGE_WORD = ["change"] - EXIT_WORD = ["exit"] + CONTINUE_WORDS = ["confirm", "continue", "c", "yes", "y"] + CHANGE_WORDS = ["change"] + EXIT_WORDS = ["exit"] TASK_REVIEW_INSTRUCTION = ( - f"If you want to change, add, delete a task or merge tasks in the plan, say '{CHANGE_WORD[0]} task task_id or current task, ... (things to change)' " - f"If you confirm the output from the current task and wish to continue, type: {CONTINUE_WORD[0]}" + f"If you want to change, add, delete a task or merge tasks in the plan, say '{CHANGE_WORDS[0]} task task_id or current task, ... (things to change)' " + f"If you confirm the output from the current task and wish to continue, type: {CONTINUE_WORDS[0]}" ) CODE_REVIEW_INSTRUCTION = ( - f"If you want the codes to be rewritten, say '{CHANGE_WORD[0]} ... (your change advice)' " - f"If you want to leave it as is, type: {CONTINUE_WORD[0]} or {CONTINUE_WORD[1]}" + f"If you want the codes to be rewritten, say '{CHANGE_WORDS[0]} ... (your change advice)' " + f"If you want to leave it as is, type: {CONTINUE_WORDS[0]} or {CONTINUE_WORDS[1]}" ) - EXIT_INSTRUCTION = f"If you want to terminate the process, type: {EXIT_WORD[0]}" + EXIT_INSTRUCTION = f"If you want to terminate the process, type: {EXIT_WORDS[0]}" class AskReview(Action): - async def run(self, context: List[Message] = [], plan: Plan = None, trigger: str = "task"): + async def run( + self, context: list[Message] = [], plan: Plan = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER + ) -> Tuple[str, bool]: if plan: logger.info("Current overall plan:") logger.info( @@ -32,7 +34,7 @@ class AskReview(Action): ) ) - logger.info("most recent context:") + logger.info("Most recent context:") latest_action = context[-1].cause_by if context and context[-1].cause_by else "" review_instruction = ( ReviewConst.TASK_REVIEW_INSTRUCTION @@ -48,11 +50,11 @@ class AskReview(Action): rsp = input(prompt) - if rsp.lower() in ReviewConst.EXIT_WORD: + if rsp.lower() in ReviewConst.EXIT_WORDS: exit() # Confirmation can be one of "confirm", "continue", "c", "yes", "y" exactly, or sentences containing "confirm". # One could say "confirm this task, but change the next task to ..." - confirmed = rsp.lower() in ReviewConst.CONTINUE_WORD or ReviewConst.CONTINUE_WORD[0] in rsp.lower() + confirmed = rsp.lower() in ReviewConst.CONTINUE_WORDS or ReviewConst.CONTINUE_WORDS[0] in rsp.lower() return rsp, confirmed diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 121c126c4..d63fa3396 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -3,7 +3,7 @@ from typing import List from metagpt.actions.write_analysis_code import BaseWriteAnalysisCode from metagpt.logs import logger from metagpt.schema import Message -from metagpt.utils.common import create_func_config +from metagpt.utils.common import create_func_call_config DEBUG_REFLECTION_EXAMPLE = ''' Example 1: @@ -100,7 +100,7 @@ class DebugCode(BaseWriteAnalysisCode): info.append(Message(role="system", content=system_prompt)) info.append(Message(role="user", content=reflection_prompt)) - resp = await self.llm.aask_code(messages=info, **create_func_config(CODE_REFLECTION)) + resp = await self.llm.aask_code(messages=info, **create_func_call_config(CODE_REFLECTION)) logger.info(f"reflection is {resp}") return resp diff --git a/metagpt/actions/execute_code.py b/metagpt/actions/execute_nb_code.py similarity index 94% rename from metagpt/actions/execute_code.py rename to metagpt/actions/execute_nb_code.py index 6a4a9abb8..7dfbecb5c 100644 --- a/metagpt/actions/execute_code.py +++ b/metagpt/actions/execute_nb_code.py @@ -7,7 +7,6 @@ import asyncio import re import traceback -from abc import ABC, abstractmethod from pathlib import Path from typing import Any, Dict, List, Tuple, Union @@ -28,30 +27,8 @@ from metagpt.logs import logger from metagpt.schema import Message -class ExecuteCode(ABC): - @abstractmethod - async def build(self): - """build code executor""" - ... - - @abstractmethod - async def run(self, code: str): - """run code""" - ... - - @abstractmethod - async def terminate(self): - """terminate executor""" - ... - - @abstractmethod - async def reset(self): - """reset executor""" - ... - - -class ExecutePyCode(ExecuteCode, Action): - """execute code, return result to llm, and display it.""" +class ExecuteNbCode(Action): + """execute notebook code block, return result to llm, and display it.""" nb: Any nb_client: Any diff --git a/metagpt/actions/ml_action.py b/metagpt/actions/ml_action.py index a61233e5a..d419026fa 100644 --- a/metagpt/actions/ml_action.py +++ b/metagpt/actions/ml_action.py @@ -11,7 +11,7 @@ from metagpt.prompts.ml_action import ( ) from metagpt.prompts.write_analysis_code import CODE_GENERATOR_WITH_TOOLS from metagpt.schema import Message, Plan -from metagpt.utils.common import CodeParser, create_func_config, remove_comments +from metagpt.utils.common import CodeParser, create_func_call_config, remove_comments class WriteCodeWithToolsML(WriteCodeWithTools): @@ -52,7 +52,7 @@ class WriteCodeWithToolsML(WriteCodeWithTools): tool_type_usage_prompt=tool_type_usage_prompt, code_steps=code_steps, ) - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + tool_config = create_func_call_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) # Extra output to be used for potential debugging @@ -97,6 +97,6 @@ class UpdateDataColumns(Action): code_context = [remove_comments(task.code) for task in finished_tasks] code_context = "\n\n".join(code_context) prompt = UPDATE_DATA_COLUMNS.format(history_code=code_context) - tool_config = create_func_config(PRINT_DATA_COLUMNS) + tool_config = create_func_call_config(PRINT_DATA_COLUMNS) rsp = await self.llm.aask_code(prompt, **tool_config) return rsp diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index 5cea9fe51..bf00e8ed1 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -2,9 +2,9 @@ """ @Date : 2023/11/20 13:19:39 @Author : orange-crow -@File : write_code_v2.py +@File : write_analysis_code.py """ -from typing import Dict, List, Tuple, Union +from typing import Dict, Tuple, Union from metagpt.actions import Action from metagpt.logs import logger @@ -17,14 +17,14 @@ from metagpt.prompts.write_analysis_code import ( from metagpt.schema import Message, Plan from metagpt.tools import TOOL_REGISTRY from metagpt.tools.tool_registry import validate_tool_names -from metagpt.utils.common import create_func_config +from metagpt.utils.common import create_func_call_config class BaseWriteAnalysisCode(Action): DEFAULT_SYSTEM_MSG: str = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" - def process_msg(self, prompt: Union[str, List[Dict], Message, List[Message]], system_msg: str = None): + def process_msg(self, prompt: Union[str, list[Dict], Message, list[Message]], system_msg: str = None): default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG # 全部转成list if not isinstance(prompt, list): @@ -53,16 +53,17 @@ class BaseWriteAnalysisCode(Action): } return messages - async def run(self, context: List[Message], plan: Plan = None) -> dict: + async def run(self, context: list[Message], plan: Plan = None) -> dict: """Run of a code writing action, used in data analysis or modeling Args: - context (List[Message]): Action output history, source action denoted by Message.cause_by + context (list[Message]): Action output history, source action denoted by Message.cause_by plan (Plan, optional): Overall plan. Defaults to None. Returns: dict: code result in the format of {"code": "print('hello world')", "language": "python"} """ + raise NotImplementedError class WriteCodeByGenerate(BaseWriteAnalysisCode): @@ -70,7 +71,7 @@ class WriteCodeByGenerate(BaseWriteAnalysisCode): async def run( self, - context: [List[Message]], + context: [list[Message]], plan: Plan = None, system_msg: str = None, **kwargs, @@ -128,7 +129,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): code_steps=code_steps, available_tools=available_tools, ) - tool_config = create_func_config(SELECT_FUNCTION_TOOLS) + tool_config = create_func_call_config(SELECT_FUNCTION_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) recommend_tools = rsp["recommend_tools"] logger.info(f"Recommended tools: \n{recommend_tools}") @@ -169,7 +170,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): async def run( self, - context: List[Message], + context: list[Message], plan: Plan, **kwargs, ) -> str: @@ -184,7 +185,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): # prepare prompt & LLM call prompt = self.process_msg(context) - tool_config = create_func_config(CODE_GENERATOR_WITH_TOOLS) + tool_config = create_func_call_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) return rsp diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/write_plan.py index 335a09841..77b52b78e 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/write_plan.py @@ -16,7 +16,7 @@ from metagpt.prompts.write_analysis_code import ( ) from metagpt.schema import Message, Plan, Task from metagpt.tools import TOOL_REGISTRY -from metagpt.utils.common import CodeParser, create_func_config +from metagpt.utils.common import CodeParser, create_func_call_config class WritePlan(Action): @@ -56,7 +56,7 @@ class WritePlan(Action): prompt = ASSIGN_TASK_TYPE_PROMPT.format( task_list=task_list, task_type_desc=task_type_desc ) # task types are set to be the same as tool types, for now - tool_config = create_func_config(ASSIGN_TASK_TYPE_CONFIG) + tool_config = create_func_call_config(ASSIGN_TASK_TYPE_CONFIG) rsp = await self.llm.aask_code(prompt, **tool_config) task_type_list = rsp["task_type"] print(f"assigned task types: {task_type_list}") diff --git a/metagpt/plan/planner.py b/metagpt/plan/planner.py index 0d8870fd3..6e866ec22 100644 --- a/metagpt/plan/planner.py +++ b/metagpt/plan/planner.py @@ -87,7 +87,11 @@ class Planner(BaseModel): await self.update_plan() async def ask_review( - self, task_result: TaskResult = None, auto_run: bool = None, trigger: str = ReviewConst.TASK_REVIEW_TRIGGER + self, + task_result: TaskResult = None, + auto_run: bool = None, + trigger: str = ReviewConst.TASK_REVIEW_TRIGGER, + review_context_len: int = 5, ): """ Ask to review the task result, reviewer needs to provide confirmation or request change. @@ -97,7 +101,9 @@ class Planner(BaseModel): auto_run = auto_run or self.auto_run if not auto_run: context = self.get_useful_memories() - review, confirmed = await AskReview().run(context=context[-5:], plan=self.plan, trigger=trigger) + review, confirmed = await AskReview().run( + context=context[-review_context_len:], plan=self.plan, trigger=trigger + ) if not confirmed: self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) return review, confirmed @@ -110,7 +116,7 @@ class Planner(BaseModel): self.working_memory.clear() confirmed_and_more = ( - ReviewConst.CONTINUE_WORD[0] in review.lower() and review.lower() not in ReviewConst.CONTINUE_WORD[0] + ReviewConst.CONTINUE_WORDS[0] in review.lower() and review.lower() not in ReviewConst.CONTINUE_WORDS[0] ) # "confirm, ... (more content, such as changing downstream tasks)" if confirmed_and_more: self.working_memory.add(Message(content=review, role="user", cause_by=AskReview)) diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index b4f9622d3..1ae4feec7 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -1,7 +1,7 @@ from pydantic import Field from metagpt.actions.ask_review import ReviewConst -from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.execute_nb_code import ExecuteNbCode from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools from metagpt.logs import logger from metagpt.roles import Role @@ -11,7 +11,7 @@ from metagpt.schema import Message, Task, TaskResult class CodeInterpreter(Role): auto_run: bool = True use_tools: bool = False - execute_code: ExecutePyCode = Field(default_factory=ExecutePyCode, exclude=True) + execute_code: ExecuteNbCode = Field(default_factory=ExecuteNbCode, exclude=True) tools: list[str] = [] def __init__( @@ -59,7 +59,7 @@ class CodeInterpreter(Role): result, success = await self.execute_code.run(**code) print(result) - self.working_memory.add(Message(content=result, role="user", cause_by=ExecutePyCode)) + self.working_memory.add(Message(content=result, role="user", cause_by=ExecuteNbCode)) ### process execution result ### if "!pip" in code["code"]: @@ -70,7 +70,7 @@ class CodeInterpreter(Role): if not success and counter >= max_retry: logger.info("coding failed!") review, _ = await self.planner.ask_review(auto_run=False, trigger=ReviewConst.CODE_REVIEW_TRIGGER) - if ReviewConst.CHANGE_WORD[0] in review: + if ReviewConst.CHANGE_WORDS[0] in review: counter = 0 # redo the task again with help of human suggestions py_code = ( diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index e7abee560..19c34f62d 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,5 +1,5 @@ from metagpt.actions.debug_code import DebugCode -from metagpt.actions.execute_code import ExecutePyCode +from metagpt.actions.execute_nb_code import ExecuteNbCode from metagpt.actions.ml_action import UpdateDataColumns, WriteCodeWithToolsML from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter @@ -19,7 +19,7 @@ class MLEngineer(CodeInterpreter): return await super()._write_code() # In a trial and errors settings, check whether this is our first attempt to tackle the task. If there is no code execution before, then it is. - is_first_trial = any_to_str(ExecutePyCode) not in [msg.cause_by for msg in self.working_memory.get()] + is_first_trial = any_to_str(ExecuteNbCode) not in [msg.cause_by for msg in self.working_memory.get()] if is_first_trial: # For the first trial, write task code from scratch diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 7d3d47680..55f4ce378 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -352,7 +352,7 @@ def parse_recipient(text): return "" -def create_func_config(func_schema: dict) -> dict: +def create_func_call_config(func_schema: dict) -> dict: """Create new function call config""" tools = [{"type": "function", "function": func_schema}] tool_choice = {"type": "function", "function": {"name": func_schema["name"]}} @@ -362,7 +362,7 @@ def create_func_config(func_schema: dict) -> dict: } -def remove_comments(code_str): +def remove_comments(code_str: str) -> str: """Remove comments from code.""" pattern = r"(\".*?\"|\'.*?\')|(\#.*?$)" diff --git a/metagpt/utils/recovery_util.py b/metagpt/utils/recovery_util.py index 3405b9587..d0b197e69 100644 --- a/metagpt/utils/recovery_util.py +++ b/metagpt/utils/recovery_util.py @@ -10,12 +10,13 @@ import nbformat from metagpt.const import DATA_PATH from metagpt.roles.role import Role +from metagpt.utils.common import read_json_file from metagpt.utils.save_code import save_code_file def load_history(save_dir: str = ""): """ - Load history from the specified save directory. + Load plan and code execution history from the specified save directory. Args: save_dir (str): The directory from which to load the history. @@ -26,14 +27,14 @@ def load_history(save_dir: str = ""): plan_path = Path(save_dir) / "plan.json" nb_path = Path(save_dir) / "history_nb" / "code.ipynb" - plan = json.load(open(plan_path, "r", encoding="utf-8")) + plan = read_json_file(plan_path) nb = nbformat.read(open(nb_path, "r", encoding="utf-8"), as_version=nbformat.NO_CONVERT) return plan, nb def save_history(role: Role, save_dir: str = ""): """ - Save history to the specified directory. + Save plan and code execution history to the specified directory. Args: role (Role): The role containing the plan and execute_code attributes. diff --git a/tests/metagpt/actions/test_execute_code.py b/tests/metagpt/actions/test_execute_code.py deleted file mode 100644 index 21627e6f9..000000000 --- a/tests/metagpt/actions/test_execute_code.py +++ /dev/null @@ -1,121 +0,0 @@ -import pytest - -from metagpt.actions.execute_code import ExecutePyCode, truncate - - -@pytest.mark.asyncio -async def test_code_running(): - pi = ExecutePyCode() - output = await pi.run("print('hello world!')") - assert output[1] is True - output = await pi.run({"code": "print('hello world!')", "language": "python"}) - assert output[1] is True - - -@pytest.mark.asyncio -async def test_split_code_running(): - pi = ExecutePyCode() - output = await pi.run("x=1\ny=2") - output = await pi.run("z=x+y") - output = await pi.run("assert z==3") - assert output[1] is True - - -@pytest.mark.asyncio -async def test_execute_error(): - pi = ExecutePyCode() - output = await pi.run("z=1/0") - assert output[1] is False - - -@pytest.mark.asyncio -async def test_plotting_code(): - pi = ExecutePyCode() - code = """ - import numpy as np - import matplotlib.pyplot as plt - - # 生成随机数据 - random_data = np.random.randn(1000) # 生成1000个符合标准正态分布的随机数 - - # 绘制直方图 - plt.hist(random_data, bins=30, density=True, alpha=0.7, color='blue', edgecolor='black') - - # 添加标题和标签 - plt.title('Histogram of Random Data') - plt.xlabel('Value') - plt.ylabel('Frequency') - - # 显示图形 - plt.show() - plt.close() - """ - output = await pi.run(code) - assert output[1] is True - - -def test_truncate(): - # 代码执行成功 - output, is_success = truncate("hello world", 5, True) - assert "Truncated to show only first 5 characters\nhello" in output - assert is_success - # 代码执行失败 - output, is_success = truncate("hello world", 5, False) - assert "Truncated to show only last 5 characters\nworld" in output - assert not is_success - # 异步 - output, is_success = truncate(" Date: Thu, 1 Feb 2024 16:34:17 +0800 Subject: [PATCH 562/637] remove environment const into individual const.py --- metagpt/const.py | 42 ---------------------- metagpt/environment/android_env/const.py | 6 ++++ metagpt/environment/mincraft_env/const.py | 44 +++++++++++++++++++++++ 3 files changed, 50 insertions(+), 42 deletions(-) create mode 100644 metagpt/environment/android_env/const.py create mode 100644 metagpt/environment/mincraft_env/const.py diff --git a/metagpt/const.py b/metagpt/const.py index 41a98356c..a1c650ce3 100644 --- a/metagpt/const.py +++ b/metagpt/const.py @@ -129,45 +129,3 @@ IGNORED_MESSAGE_ID = "0" GENERALIZATION = "Generalize" COMPOSITION = "Composite" AGGREGATION = "Aggregate" - -# For Android Assistant Agent -ADB_EXEC_FAIL = "FAILED" - -# For Mincraft Game Agent -MC_CKPT_DIR = METAGPT_ROOT / "data/mincraft/ckpt" -MC_LOG_DIR = METAGPT_ROOT / "logs" -MC_DEFAULT_WARMUP = { - "context": 15, - "biome": 10, - "time": 15, - "nearby_blocks": 0, - "other_blocks": 10, - "nearby_entities": 5, - "health": 15, - "hunger": 15, - "position": 0, - "equipment": 0, - "inventory": 0, - "optional_inventory_items": 7, - "chests": 0, - "completed_tasks": 0, - "failed_tasks": 0, -} -MC_CURRICULUM_OB = [ - "context", - "biome", - "time", - "nearby_blocks", - "other_blocks", - "nearby_entities", - "health", - "hunger", - "position", - "equipment", - "inventory", - "chests", - "completed_tasks", - "failed_tasks", -] -MC_CORE_INVENTORY_ITEMS = r".*_log|.*_planks|stick|crafting_table|furnace" -r"|cobblestone|dirt|coal|.*_pickaxe|.*_sword|.*_axe", # curriculum_agent: only show these items in inventory before optional_inventory_items reached in warm up diff --git a/metagpt/environment/android_env/const.py b/metagpt/environment/android_env/const.py new file mode 100644 index 000000000..8811289bf --- /dev/null +++ b/metagpt/environment/android_env/const.py @@ -0,0 +1,6 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : + +# For Android Assistant Agent +ADB_EXEC_FAIL = "FAILED" diff --git a/metagpt/environment/mincraft_env/const.py b/metagpt/environment/mincraft_env/const.py new file mode 100644 index 000000000..a7222f9cd --- /dev/null +++ b/metagpt/environment/mincraft_env/const.py @@ -0,0 +1,44 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : + +from metagpt.const import METAGPT_ROOT + +# For Mincraft Game Agent +MC_CKPT_DIR = METAGPT_ROOT / "data/mincraft/ckpt" +MC_LOG_DIR = METAGPT_ROOT / "logs" +MC_DEFAULT_WARMUP = { + "context": 15, + "biome": 10, + "time": 15, + "nearby_blocks": 0, + "other_blocks": 10, + "nearby_entities": 5, + "health": 15, + "hunger": 15, + "position": 0, + "equipment": 0, + "inventory": 0, + "optional_inventory_items": 7, + "chests": 0, + "completed_tasks": 0, + "failed_tasks": 0, +} +MC_CURRICULUM_OB = [ + "context", + "biome", + "time", + "nearby_blocks", + "other_blocks", + "nearby_entities", + "health", + "hunger", + "position", + "equipment", + "inventory", + "chests", + "completed_tasks", + "failed_tasks", +] +MC_CORE_INVENTORY_ITEMS = r".*_log|.*_planks|stick|crafting_table|furnace" +r"|cobblestone|dirt|coal|.*_pickaxe|.*_sword|.*_axe", # curriculum_agent: only show these items in inventory before optional_inventory_items reached in warm up From e8c333b45b2a63e3c9e2a7ed59724f6e625feb1f Mon Sep 17 00:00:00 2001 From: geekan Date: Thu, 1 Feb 2024 16:38:51 +0800 Subject: [PATCH 563/637] try to fix run code error. --- tests/data/rsp_cache.json | 5 ++++- tests/metagpt/actions/test_run_code.py | 2 +- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index 998eb714f..5704cea5a 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -312,5 +312,8 @@ "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"issue_type\": \"BUG\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- issue_type: # Answer BUG/REQUIREMENT. If it is a bugfix, answer BUG, otherwise answer Requirement\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"issue_type\": \"REQUIREMENT\"\n}\n[/CONTENT]", "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Product Goals\": [\n \"Create a challenging and addictive gameplay\",\n \"Enhance accessibility and responsiveness for all users\",\n \"Implement visually appealing UI design\"\n ]\n}\n[/CONTENT]", "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ]\n}\n[/CONTENT]", - "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code should handle user input and update the game board accordingly.\"\n ],\n [\n \"P0\",\n \"The game algorithm should handle the merging of tiles and the generation of new tiles with values of 2 or 4.\"\n ],\n [\n \"P1\",\n \"The game should end when the player achieves the 2048 tile or when there are no possible moves left.\"\n ],\n [\n \"P1\",\n \"The game should display the current score and the highest tile achieved by the player.\"\n ],\n [\n \"P2\",\n \"The game should have a smooth and visually appealing user interface.\"\n ]\n ]\n}\n[/CONTENT]" + "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code should handle user input and update the game board accordingly.\"\n ],\n [\n \"P0\",\n \"The game algorithm should handle the merging of tiles and the generation of new tiles with values of 2 or 4.\"\n ],\n [\n \"P1\",\n \"The game should end when the player achieves the 2048 tile or when there are no possible moves left.\"\n ],\n [\n \"P1\",\n \"The game should display the current score and the highest tile achieved by the player.\"\n ],\n [\n \"P2\",\n \"The game should have a smooth and visually appealing user interface.\"\n ]\n ]\n}\n[/CONTENT]", + "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.txt\n## Development Code\n```python\nresult = 'helloworld'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\n\n## Running Output\nstandard output: \n```text\nhelloworld\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nNo errors detected. The development code runs successfully and outputs the expected result without any errors.\n## File To Rewrite:\nNone\n## Status:\nPASS\n## Send To:\nNoOne\n---", + "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.sh\n## Development Code\n```python\necho 'Hello World'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\necho Hello World\n## Running Output\nstandard output: \n```text\nHello World\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nNo errors detected in the running result. The development code executed successfully, and the output matches the expected result. Since there is no test code provided, no test execution was performed. Therefore, no specific instructions are needed for correction.\n## File To Rewrite:\nNone\n## Status:\nPASS\n## Send To:\nNoOne\n---", + "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.py\n## Development Code\n```python\npython -c \"print(1/0)\"\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\npython -c print(1/0)\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\nTraceback (most recent call last):\n File \"\", line 1, in \nZeroDivisionError: division by zero\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nThe error is caused by attempting to divide by zero in the development code. To fix this error, you should modify the division operation to avoid division by zero. For example, you can add a condition to check if the denominator is zero before performing the division.\n\n## File To Rewrite:\na.py\n\n## Status:\nFAIL\n\n## Send To:\nEngineer\n---" } \ No newline at end of file diff --git a/tests/metagpt/actions/test_run_code.py b/tests/metagpt/actions/test_run_code.py index afd308da7..2ec8a7748 100644 --- a/tests/metagpt/actions/test_run_code.py +++ b/tests/metagpt/actions/test_run_code.py @@ -38,7 +38,7 @@ async def test_run_script(context): @pytest.mark.asyncio async def test_run(context): inputs = [ - (RunCodeContext(mode="text", code_filename="a.txt", code="print('Hello, World')"), "PASS"), + (RunCodeContext(mode="text", code_filename="a.txt", code="result = 'helloworld'"), "PASS"), ( RunCodeContext( mode="script", From 9b613eec598448a2efd6a320d1d4f538f544320f Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Thu, 1 Feb 2024 16:47:29 +0800 Subject: [PATCH 564/637] remove global config in the search/browser engine --- examples/search_with_specific_engine.py | 11 ++- metagpt/actions/research.py | 38 +++++---- metagpt/actions/search_and_summarize.py | 21 +++-- metagpt/config2.py | 2 +- metagpt/configs/browser_config.py | 6 +- metagpt/configs/search_config.py | 7 +- metagpt/context_mixin.py | 15 ++-- metagpt/roles/sales.py | 19 +++-- metagpt/roles/searcher.py | 35 ++++----- metagpt/tools/search_engine.py | 78 +++++++++++++------ metagpt/tools/search_engine_ddg.py | 28 +++---- metagpt/tools/search_engine_googleapi.py | 64 +++++++-------- metagpt/tools/search_engine_serpapi.py | 34 ++++---- metagpt/tools/search_engine_serper.py | 31 ++++---- metagpt/tools/web_browser_engine.py | 47 +++++++---- .../tools/web_browser_engine_playwright.py | 41 +++++----- metagpt/tools/web_browser_engine_selenium.py | 61 ++++++++------- tests/metagpt/actions/test_research.py | 10 ++- tests/metagpt/learn/test_google_search.py | 2 +- tests/metagpt/roles/test_researcher.py | 2 +- tests/metagpt/tools/test_search_engine.py | 34 ++++---- .../test_web_browser_engine_playwright.py | 35 ++++----- .../tools/test_web_browser_engine_selenium.py | 36 ++++----- tests/mock/mock_aiohttp.py | 3 +- 24 files changed, 351 insertions(+), 309 deletions(-) diff --git a/examples/search_with_specific_engine.py b/examples/search_with_specific_engine.py index 9406a2965..97b1378ee 100644 --- a/examples/search_with_specific_engine.py +++ b/examples/search_with_specific_engine.py @@ -5,17 +5,20 @@ import asyncio from metagpt.roles import Searcher -from metagpt.tools import SearchEngineType +from metagpt.tools.search_engine import SearchEngine, SearchEngineType async def main(): question = "What are the most interesting human facts?" + kwargs = {"api_key": "", "cse_id": "", "proxy": None} # Serper API - # await Searcher(engine=SearchEngineType.SERPER_GOOGLE).run(question) + # await Searcher(search_engine=SearchEngine(engine=SearchEngineType.SERPER_GOOGLE, **kwargs)).run(question) # SerpAPI - await Searcher(engine=SearchEngineType.SERPAPI_GOOGLE).run(question) + # await Searcher(search_engine=SearchEngine(engine=SearchEngineType.SERPAPI_GOOGLE, **kwargs)).run(question) # Google API - # await Searcher(engine=SearchEngineType.DIRECT_GOOGLE).run(question) + # await Searcher(search_engine=SearchEngine(engine=SearchEngineType.DIRECT_GOOGLE, **kwargs)).run(question) + # DDG API + await Searcher(search_engine=SearchEngine(engine=SearchEngineType.DUCK_DUCK_GO, **kwargs)).run(question) if __name__ == "__main__": diff --git a/metagpt/actions/research.py b/metagpt/actions/research.py index 2755628c9..2ebeadb66 100644 --- a/metagpt/actions/research.py +++ b/metagpt/actions/research.py @@ -3,15 +3,15 @@ from __future__ import annotations import asyncio -from typing import Callable, Optional, Union +from typing import Any, Callable, Optional, Union -from pydantic import Field, parse_obj_as +from pydantic import TypeAdapter, model_validator from metagpt.actions import Action from metagpt.config2 import config from metagpt.logs import logger from metagpt.tools.search_engine import SearchEngine -from metagpt.tools.web_browser_engine import WebBrowserEngine, WebBrowserEngineType +from metagpt.tools.web_browser_engine import WebBrowserEngine from metagpt.utils.common import OutputParser from metagpt.utils.text import generate_prompt_chunk, reduce_message_length @@ -81,10 +81,16 @@ class CollectLinks(Action): name: str = "CollectLinks" i_context: Optional[str] = None desc: str = "Collect links from a search engine." - - search_engine: SearchEngine = Field(default_factory=SearchEngine) + search_func: Optional[Any] = None + search_engine: Optional[SearchEngine] = None rank_func: Optional[Callable[[list[str]], None]] = None + @model_validator(mode="after") + def validate_engine_and_run_func(self): + if self.search_engine is None: + self.search_engine = SearchEngine.from_search_config(self.config.search, proxy=self.config.proxy) + return self + async def run( self, topic: str, @@ -107,7 +113,7 @@ class CollectLinks(Action): keywords = await self._aask(SEARCH_TOPIC_PROMPT, [system_text]) try: keywords = OutputParser.extract_struct(keywords, list) - keywords = parse_obj_as(list[str], keywords) + keywords = TypeAdapter(list[str]).validate_python(keywords) except Exception as e: logger.exception(f"fail to get keywords related to the research topic '{topic}' for {e}") keywords = [topic] @@ -133,7 +139,7 @@ class CollectLinks(Action): queries = await self._aask(prompt, [system_text]) try: queries = OutputParser.extract_struct(queries, list) - queries = parse_obj_as(list[str], queries) + queries = TypeAdapter(list[str]).validate_python(queries) except Exception as e: logger.exception(f"fail to break down the research question due to {e}") queries = keywords @@ -178,15 +184,17 @@ class WebBrowseAndSummarize(Action): i_context: Optional[str] = None desc: str = "Explore the web and provide summaries of articles and webpages." browse_func: Union[Callable[[list[str]], None], None] = None - web_browser_engine: Optional[WebBrowserEngine] = WebBrowserEngineType.PLAYWRIGHT + web_browser_engine: Optional[WebBrowserEngine] = None - def __init__(self, **kwargs): - super().__init__(**kwargs) - - self.web_browser_engine = WebBrowserEngine( - engine=WebBrowserEngineType.CUSTOM if self.browse_func else WebBrowserEngineType.PLAYWRIGHT, - run_func=self.browse_func, - ) + @model_validator(mode="after") + def validate_engine_and_run_func(self): + if self.web_browser_engine is None: + self.web_browser_engine = WebBrowserEngine.from_browser_config( + self.config.browser, + browse_func=self.browse_func, + proxy=self.config.proxy, + ) + return self async def run( self, diff --git a/metagpt/actions/search_and_summarize.py b/metagpt/actions/search_and_summarize.py index 59b35cd58..7eed7381b 100644 --- a/metagpt/actions/search_and_summarize.py +++ b/metagpt/actions/search_and_summarize.py @@ -5,7 +5,7 @@ @Author : alexanderwu @File : search_google.py """ -from typing import Any, Optional +from typing import Optional import pydantic from pydantic import model_validator @@ -13,7 +13,6 @@ from pydantic import model_validator from metagpt.actions import Action from metagpt.logs import logger from metagpt.schema import Message -from metagpt.tools import SearchEngineType from metagpt.tools.search_engine import SearchEngine SEARCH_AND_SUMMARIZE_SYSTEM = """### Requirements @@ -105,21 +104,19 @@ You are a member of a professional butler team and will provide helpful suggesti class SearchAndSummarize(Action): name: str = "" content: Optional[str] = None - engine: Optional[SearchEngineType] = None - search_func: Optional[Any] = None search_engine: SearchEngine = None result: str = "" @model_validator(mode="after") - def validate_engine_and_run_func(self): - if self.engine is None: - self.engine = self.config.search_engine - try: - search_engine = SearchEngine(engine=self.engine, run_func=self.search_func) - except pydantic.ValidationError: - search_engine = None + def validate_search_engine(self): + if self.search_engine is None: + try: + config = self.config + search_engine = SearchEngine.from_search_config(config.search, proxy=config.proxy) + except pydantic.ValidationError: + search_engine = None - self.search_engine = search_engine + self.search_engine = search_engine return self async def run(self, context: list[Message], system_text=SEARCH_AND_SUMMARIZE_SYSTEM) -> str: diff --git a/metagpt/config2.py b/metagpt/config2.py index 21c17f7be..bc6af18c6 100644 --- a/metagpt/config2.py +++ b/metagpt/config2.py @@ -51,7 +51,7 @@ class Config(CLIParams, YamlModel): proxy: str = "" # Tool Parameters - search: Optional[SearchConfig] = None + search: SearchConfig = SearchConfig() browser: BrowserConfig = BrowserConfig() mermaid: MermaidConfig = MermaidConfig() diff --git a/metagpt/configs/browser_config.py b/metagpt/configs/browser_config.py index 00f918735..2f8024f44 100644 --- a/metagpt/configs/browser_config.py +++ b/metagpt/configs/browser_config.py @@ -15,6 +15,6 @@ class BrowserConfig(YamlModel): """Config for Browser""" engine: WebBrowserEngineType = WebBrowserEngineType.PLAYWRIGHT - browser: Literal["chrome", "firefox", "edge", "ie"] = "chrome" - driver: Literal["chromium", "firefox", "webkit"] = "chromium" - path: str = "" + browser_type: Literal["chromium", "firefox", "webkit", "chrome", "firefox", "edge", "ie"] = "chromium" + """If the engine is Playwright, the value should be one of "chromium", "firefox", or "webkit". If it is Selenium, the value + should be either "chrome", "firefox", "edge", or "ie".""" diff --git a/metagpt/configs/search_config.py b/metagpt/configs/search_config.py index a8ae918db..af928b02a 100644 --- a/metagpt/configs/search_config.py +++ b/metagpt/configs/search_config.py @@ -5,6 +5,8 @@ @Author : alexanderwu @File : search_config.py """ +from typing import Callable, Optional + from metagpt.tools import SearchEngineType from metagpt.utils.yaml_model import YamlModel @@ -12,6 +14,7 @@ from metagpt.utils.yaml_model import YamlModel class SearchConfig(YamlModel): """Config for Search""" - api_key: str - api_type: SearchEngineType = SearchEngineType.SERPAPI_GOOGLE + api_type: SearchEngineType = SearchEngineType.DUCK_DUCK_GO + api_key: str = "" cse_id: str = "" # for google + search_func: Optional[Callable] = None diff --git a/metagpt/context_mixin.py b/metagpt/context_mixin.py index bdf2d0734..060150f4d 100644 --- a/metagpt/context_mixin.py +++ b/metagpt/context_mixin.py @@ -7,7 +7,7 @@ """ from typing import Optional -from pydantic import BaseModel, ConfigDict, Field +from pydantic import BaseModel, ConfigDict, Field, model_validator from metagpt.config2 import Config from metagpt.context import Context @@ -17,7 +17,7 @@ from metagpt.provider.base_llm import BaseLLM class ContextMixin(BaseModel): """Mixin class for context and config""" - model_config = ConfigDict(arbitrary_types_allowed=True) + model_config = ConfigDict(arbitrary_types_allowed=True, extra="allow") # Pydantic has bug on _private_attr when using inheritance, so we use private_* instead # - https://github.com/pydantic/pydantic/issues/7142 @@ -32,15 +32,18 @@ class ContextMixin(BaseModel): # Env/Role/Action will use this llm as private llm, or use self.context._llm instance private_llm: Optional[BaseLLM] = Field(default=None, exclude=True) - def __init__( + @model_validator(mode="after") + def validate_extra(self): + self._process_extra(**(self.model_extra or {})) + return self + + def _process_extra( self, context: Optional[Context] = None, config: Optional[Config] = None, llm: Optional[BaseLLM] = None, - **kwargs, ): - """Initialize with config""" - super().__init__(**kwargs) + """Process the extra field""" self.set_context(context) self.set_config(config) self.set_llm(llm) diff --git a/metagpt/roles/sales.py b/metagpt/roles/sales.py index 7929ce7fe..bc449b5cd 100644 --- a/metagpt/roles/sales.py +++ b/metagpt/roles/sales.py @@ -8,12 +8,12 @@ from typing import Optional -from pydantic import Field +from pydantic import Field, model_validator from metagpt.actions import SearchAndSummarize, UserRequirement from metagpt.document_store.base_store import BaseStore from metagpt.roles import Role -from metagpt.tools import SearchEngineType +from metagpt.tools.search_engine import SearchEngine class Sales(Role): @@ -29,14 +29,13 @@ class Sales(Role): store: Optional[BaseStore] = Field(default=None, exclude=True) - def __init__(self, **kwargs): - super().__init__(**kwargs) - self._set_store(self.store) - - def _set_store(self, store): - if store: - action = SearchAndSummarize(name="", engine=SearchEngineType.CUSTOM_ENGINE, search_func=store.asearch) + @model_validator(mode="after") + def validate_stroe(self): + if self.store: + search_engine = SearchEngine.from_search_func(search_func=self.store.asearch, proxy=self.config.proxy) + action = SearchAndSummarize(search_engine=search_engine, context=self.context) else: - action = SearchAndSummarize() + action = SearchAndSummarize self.set_actions([action]) self._watch([UserRequirement]) + return self diff --git a/metagpt/roles/searcher.py b/metagpt/roles/searcher.py index 19a73a40e..557c5ae95 100644 --- a/metagpt/roles/searcher.py +++ b/metagpt/roles/searcher.py @@ -8,7 +8,9 @@ the `cause_by` value in the `Message` to a string to support the new message distribution feature. """ -from pydantic import Field +from typing import Optional + +from pydantic import Field, model_validator from metagpt.actions import SearchAndSummarize from metagpt.actions.action_node import ActionNode @@ -16,7 +18,7 @@ from metagpt.actions.action_output import ActionOutput from metagpt.logs import logger from metagpt.roles import Role from metagpt.schema import Message -from metagpt.tools import SearchEngineType +from metagpt.tools.search_engine import SearchEngine class Searcher(Role): @@ -28,33 +30,22 @@ class Searcher(Role): profile (str): Role profile. goal (str): Goal of the searcher. constraints (str): Constraints or limitations for the searcher. - engine (SearchEngineType): The type of search engine to use. + search_engine (SearchEngine): The search engine to use. """ name: str = Field(default="Alice") profile: str = Field(default="Smart Assistant") goal: str = "Provide search services for users" constraints: str = "Answer is rich and complete" - engine: SearchEngineType = SearchEngineType.SERPAPI_GOOGLE + search_engine: Optional[SearchEngine] = None - def __init__(self, **kwargs) -> None: - """ - Initializes the Searcher role with given attributes. - - Args: - name (str): Name of the searcher. - profile (str): Role profile. - goal (str): Goal of the searcher. - constraints (str): Constraints or limitations for the searcher. - engine (SearchEngineType): The type of search engine to use. - """ - super().__init__(**kwargs) - self.set_actions([SearchAndSummarize(engine=self.engine)]) - - def set_search_func(self, search_func): - """Sets a custom search function for the searcher.""" - action = SearchAndSummarize(name="", engine=SearchEngineType.CUSTOM_ENGINE, search_func=search_func) - self.set_actions([action]) + @model_validator(mode="after") + def post_root(self): + if self.search_engine: + self.set_actions([SearchAndSummarize(search_engine=self.search_engine, context=self.context)]) + else: + self.set_actions([SearchAndSummarize]) + return self async def _act_sp(self) -> Message: """Performs the search action in a single process.""" diff --git a/metagpt/tools/search_engine.py b/metagpt/tools/search_engine.py index 0d0db9147..880b6ca85 100644 --- a/metagpt/tools/search_engine.py +++ b/metagpt/tools/search_engine.py @@ -8,14 +8,17 @@ import importlib from typing import Callable, Coroutine, Literal, Optional, Union, overload +from pydantic import BaseModel, ConfigDict, model_validator from semantic_kernel.skill_definition import sk_function +from metagpt.configs.search_config import SearchConfig +from metagpt.logs import logger from metagpt.tools import SearchEngineType class SkSearchEngine: - def __init__(self): - self.search_engine = SearchEngine() + def __init__(self, **kwargs): + self.search_engine = SearchEngine(**kwargs) @sk_function( description="searches results from Google. Useful when you need to find short " @@ -28,43 +31,59 @@ class SkSearchEngine: return result -class SearchEngine: - """Class representing a search engine. +class SearchEngine(BaseModel): + model_config = ConfigDict(arbitrary_types_allowed=True, extra="allow") - Args: - engine: The search engine type. Defaults to the search engine specified in the config. - run_func: The function to run the search. Defaults to None. + engine: SearchEngineType = SearchEngineType.SERPER_GOOGLE + run_func: Optional[Callable[[str, int, bool], Coroutine[None, None, Union[str, list[str]]]]] = None + api_key: Optional[str] = None + proxy: Optional[str] = None - Attributes: - run_func: The function to run the search. - engine: The search engine type. - """ + @model_validator(mode="after") + def validate_extra(self): + data = self.model_dump(exclude={"engine"}, exclude_none=True, exclude_defaults=True) + if self.model_extra: + data.update(self.model_extra) + self._process_extra(**data) + return self - def __init__( + def _process_extra( self, - engine: Optional[SearchEngineType] = SearchEngineType.SERPER_GOOGLE, - run_func: Callable[[str, int, bool], Coroutine[None, None, Union[str, list[str]]]] = None, + run_func: Optional[Callable[[str, int, bool], Coroutine[None, None, Union[str, list[str]]]]] = None, **kwargs, ): - if engine == SearchEngineType.SERPAPI_GOOGLE: + if self.engine == SearchEngineType.SERPAPI_GOOGLE: module = "metagpt.tools.search_engine_serpapi" run_func = importlib.import_module(module).SerpAPIWrapper(**kwargs).run - elif engine == SearchEngineType.SERPER_GOOGLE: + elif self.engine == SearchEngineType.SERPER_GOOGLE: module = "metagpt.tools.search_engine_serper" run_func = importlib.import_module(module).SerperWrapper(**kwargs).run - elif engine == SearchEngineType.DIRECT_GOOGLE: + elif self.engine == SearchEngineType.DIRECT_GOOGLE: module = "metagpt.tools.search_engine_googleapi" run_func = importlib.import_module(module).GoogleAPIWrapper(**kwargs).run - elif engine == SearchEngineType.DUCK_DUCK_GO: + elif self.engine == SearchEngineType.DUCK_DUCK_GO: module = "metagpt.tools.search_engine_ddg" run_func = importlib.import_module(module).DDGAPIWrapper(**kwargs).run - elif engine == SearchEngineType.CUSTOM_ENGINE: - pass # run_func = run_func + elif self.engine == SearchEngineType.CUSTOM_ENGINE: + run_func = self.run_func else: raise NotImplementedError - self.engine = engine self.run_func = run_func + @classmethod + def from_search_config(cls, config: SearchConfig, **kwargs): + data = config.model_dump(exclude={"api_type", "search_func"}) + if config.search_func is not None: + data["run_func"] = config.search_func + + return cls(engine=config.api_type, **data, **kwargs) + + @classmethod + def from_search_func( + cls, search_func: Callable[[str, int, bool], Coroutine[None, None, Union[str, list[str]]]], **kwargs + ): + return cls(engine=SearchEngineType.CUSTOM_ENGINE, run_func=search_func, **kwargs) + @overload def run( self, @@ -83,7 +102,13 @@ class SearchEngine: ) -> list[dict[str, str]]: ... - async def run(self, query: str, max_results: int = 8, as_string: bool = True) -> Union[str, list[dict[str, str]]]: + async def run( + self, + query: str, + max_results: int = 8, + as_string: bool = True, + ignore_errors: bool = False, + ) -> Union[str, list[dict[str, str]]]: """Run a search query. Args: @@ -94,4 +119,11 @@ class SearchEngine: Returns: The search results as a string or a list of dictionaries. """ - return await self.run_func(query, max_results=max_results, as_string=as_string) + try: + return await self.run_func(query, max_results=max_results, as_string=as_string) + except Exception as e: + # Handle errors in the API call + logger.exception(f"fail to search {query} for {e}") + if not ignore_errors: + raise e + return "" if as_string else [] diff --git a/metagpt/tools/search_engine_ddg.py b/metagpt/tools/search_engine_ddg.py index 3d004a4ee..1412f20cf 100644 --- a/metagpt/tools/search_engine_ddg.py +++ b/metagpt/tools/search_engine_ddg.py @@ -5,9 +5,9 @@ from __future__ import annotations import asyncio import json from concurrent import futures -from typing import Literal, overload +from typing import Literal, Optional, overload -from metagpt.config2 import config +from pydantic import BaseModel, ConfigDict try: from duckduckgo_search import DDGS @@ -18,24 +18,16 @@ except ImportError: ) -class DDGAPIWrapper: - """Wrapper around duckduckgo_search API. +class DDGAPIWrapper(BaseModel): + model_config = ConfigDict(arbitrary_types_allowed=True) - To use this module, you should have the `duckduckgo_search` Python package installed. - """ + loop: Optional[asyncio.AbstractEventLoop] = None + executor: Optional[futures.Executor] = None + proxy: Optional[str] = None - def __init__( - self, - *, - loop: asyncio.AbstractEventLoop | None = None, - executor: futures.Executor | None = None, - ): - kwargs = {} - if config.proxy: - kwargs["proxies"] = config.proxy - self.loop = loop - self.executor = executor - self.ddgs = DDGS(**kwargs) + @property + def ddgs(self): + return DDGS(proxies=self.proxy) @overload def run( diff --git a/metagpt/tools/search_engine_googleapi.py b/metagpt/tools/search_engine_googleapi.py index 0a8f796cb..66b5ba950 100644 --- a/metagpt/tools/search_engine_googleapi.py +++ b/metagpt/tools/search_engine_googleapi.py @@ -4,19 +4,16 @@ from __future__ import annotations import asyncio import json +import warnings from concurrent import futures from typing import Optional from urllib.parse import urlparse import httplib2 -from pydantic import BaseModel, ConfigDict, Field, field_validator - -from metagpt.config2 import config -from metagpt.logs import logger +from pydantic import BaseModel, ConfigDict, model_validator try: from googleapiclient.discovery import build - from googleapiclient.errors import HttpError except ImportError: raise ImportError( "To use this module, you should have the `google-api-python-client` Python package installed. " @@ -27,40 +24,41 @@ except ImportError: class GoogleAPIWrapper(BaseModel): model_config = ConfigDict(arbitrary_types_allowed=True) - google_api_key: Optional[str] = Field(default=None, validate_default=True) - google_cse_id: Optional[str] = Field(default=None, validate_default=True) + api_key: str + cse_id: str loop: Optional[asyncio.AbstractEventLoop] = None executor: Optional[futures.Executor] = None + proxy: Optional[str] = None - @field_validator("google_api_key", mode="before") + @model_validator(mode="before") @classmethod - def check_google_api_key(cls, val: str): - val = val or config.search.api_key - if not val: + def validate_google(cls, values: dict) -> dict: + if "google_api_key" in values: + values.setdefault("api_key", values["google_api_key"]) + warnings.warn("`google_api_key` is deprecated, use `api_key` instead", DeprecationWarning, stacklevel=2) + + if "api_key" not in values: raise ValueError( - "To use, make sure you provide the google_api_key when constructing an object. Alternatively, " - "ensure that the environment variable GOOGLE_API_KEY is set with your API key. You can obtain " + "To use google search engine, make sure you provide the `api_key` when constructing an object. You can obtain " "an API key from https://console.cloud.google.com/apis/credentials." ) - return val - @field_validator("google_cse_id", mode="before") - @classmethod - def check_google_cse_id(cls, val: str): - val = val or config.search.cse_id - if not val: + if "google_cse_id" in values: + values.setdefault("cse_id", values["google_cse_id"]) + warnings.warn("`google_cse_id` is deprecated, use `cse_id` instead", DeprecationWarning, stacklevel=2) + + if "cse_id" not in values: raise ValueError( - "To use, make sure you provide the google_cse_id when constructing an object. Alternatively, " - "ensure that the environment variable GOOGLE_CSE_ID is set with your API key. You can obtain " - "an API key from https://programmablesearchengine.google.com/controlpanel/create." + "To use google search engine, make sure you provide the `cse_id` when constructing an object. You can obtain " + "the cse_id from https://programmablesearchengine.google.com/controlpanel/create." ) - return val + return values @property def google_api_client(self): - build_kwargs = {"developerKey": self.google_api_key} - if config.proxy: - parse_result = urlparse(config.proxy) + build_kwargs = {"developerKey": self.api_key} + if self.proxy: + parse_result = urlparse(self.proxy) proxy_type = parse_result.scheme if proxy_type == "https": proxy_type = "http" @@ -96,17 +94,11 @@ class GoogleAPIWrapper(BaseModel): """ loop = self.loop or asyncio.get_event_loop() future = loop.run_in_executor( - self.executor, self.google_api_client.list(q=query, num=max_results, cx=self.google_cse_id).execute + self.executor, self.google_api_client.list(q=query, num=max_results, cx=self.cse_id).execute ) - try: - result = await future - # Extract the search result items from the response - search_results = result.get("items", []) - - except HttpError as e: - # Handle errors in the API call - logger.exception(f"fail to search {query} for {e}") - search_results = [] + result = await future + # Extract the search result items from the response + search_results = result.get("items", []) focus = focus or ["snippet", "link", "title"] details = [{i: j for i, j in item_dict.items() if i in focus} for item_dict in search_results] diff --git a/metagpt/tools/search_engine_serpapi.py b/metagpt/tools/search_engine_serpapi.py index 8d27d493d..5744b1b62 100644 --- a/metagpt/tools/search_engine_serpapi.py +++ b/metagpt/tools/search_engine_serpapi.py @@ -5,18 +5,17 @@ @Author : alexanderwu @File : search_engine_serpapi.py """ +import warnings from typing import Any, Dict, Optional, Tuple import aiohttp -from pydantic import BaseModel, ConfigDict, Field, field_validator - -from metagpt.config2 import config +from pydantic import BaseModel, ConfigDict, Field, model_validator class SerpAPIWrapper(BaseModel): model_config = ConfigDict(arbitrary_types_allowed=True) - search_engine: Any = None #: :meta private: + api_key: str params: dict = Field( default_factory=lambda: { "engine": "google", @@ -25,21 +24,22 @@ class SerpAPIWrapper(BaseModel): "hl": "en", } ) - # should add `validate_default=True` to check with default value - serpapi_api_key: Optional[str] = Field(default=None, validate_default=True) aiosession: Optional[aiohttp.ClientSession] = None + proxy: Optional[str] = None - @field_validator("serpapi_api_key", mode="before") + @model_validator(mode="before") @classmethod - def check_serpapi_api_key(cls, val: str): - val = val or config.search.api_key - if not val: + def validate_serpapi(cls, values: dict) -> dict: + if "serpapi_api_key" in values: + values.setdefault("api_key", values["serpapi_api_key"]) + warnings.warn("`serpapi_api_key` is deprecated, use `api_key` instead", DeprecationWarning, stacklevel=2) + + if "api_key" not in values: raise ValueError( - "To use, make sure you provide the serpapi_api_key when constructing an object. Alternatively, " - "ensure that the environment variable SERPAPI_API_KEY is set with your API key. You can obtain " - "an API key from https://serpapi.com/." + "To use serpapi search engine, make sure you provide the `api_key` when constructing an object. You can obtain" + " an API key from https://serpapi.com/." ) - return val + return values async def run(self, query, max_results: int = 8, as_string: bool = True, **kwargs: Any) -> str: """Run query through SerpAPI and parse result async.""" @@ -60,11 +60,11 @@ class SerpAPIWrapper(BaseModel): url, params = construct_url_and_params() if not self.aiosession: async with aiohttp.ClientSession() as session: - async with session.get(url, params=params) as response: + async with session.get(url, params=params, proxy=self.proxy) as response: response.raise_for_status() res = await response.json() else: - async with self.aiosession.get(url, params=params) as response: + async with self.aiosession.get(url, params=params, proxy=self.proxy) as response: response.raise_for_status() res = await response.json() @@ -73,7 +73,7 @@ class SerpAPIWrapper(BaseModel): def get_params(self, query: str) -> Dict[str, str]: """Get parameters for SerpAPI.""" _params = { - "api_key": self.serpapi_api_key, + "api_key": self.api_key, "q": query, } params = {**self.params, **_params} diff --git a/metagpt/tools/search_engine_serper.py b/metagpt/tools/search_engine_serper.py index 71ee2f4f9..ba2fb4f93 100644 --- a/metagpt/tools/search_engine_serper.py +++ b/metagpt/tools/search_engine_serper.py @@ -6,33 +6,34 @@ @File : search_engine_serpapi.py """ import json +import warnings from typing import Any, Dict, Optional, Tuple import aiohttp -from pydantic import BaseModel, ConfigDict, Field, field_validator - -from metagpt.config2 import config +from pydantic import BaseModel, ConfigDict, Field, model_validator class SerperWrapper(BaseModel): model_config = ConfigDict(arbitrary_types_allowed=True) - search_engine: Any = None #: :meta private: + api_key: str payload: dict = Field(default_factory=lambda: {"page": 1, "num": 10}) - serper_api_key: Optional[str] = Field(default=None, validate_default=True) aiosession: Optional[aiohttp.ClientSession] = None + proxy: Optional[str] = None - @field_validator("serper_api_key", mode="before") + @model_validator(mode="before") @classmethod - def check_serper_api_key(cls, val: str): - val = val or config.search.api_key - if not val: + def validate_serper(cls, values: dict) -> dict: + if "serper_api_key" in values: + values.setdefault("api_key", values["serper_api_key"]) + warnings.warn("`serper_api_key` is deprecated, use `api_key` instead", DeprecationWarning, stacklevel=2) + + if "api_key" not in values: raise ValueError( - "To use, make sure you provide the serper_api_key when constructing an object. Alternatively, " - "ensure that the environment variable SERPER_API_KEY is set with your API key. You can obtain " + "To use serper search engine, make sure you provide the `api_key` when constructing an object. You can obtain " "an API key from https://serper.dev/." ) - return val + return values async def run(self, query: str, max_results: int = 8, as_string: bool = True, **kwargs: Any) -> str: """Run query through Serper and parse result async.""" @@ -54,11 +55,11 @@ class SerperWrapper(BaseModel): url, payloads, headers = construct_url_and_payload_and_headers() if not self.aiosession: async with aiohttp.ClientSession() as session: - async with session.post(url, data=payloads, headers=headers) as response: + async with session.post(url, data=payloads, headers=headers, proxy=self.proxy) as response: response.raise_for_status() res = await response.json() else: - async with self.aiosession.get.post(url, data=payloads, headers=headers) as response: + async with self.aiosession.get.post(url, data=payloads, headers=headers, proxy=self.proxy) as response: response.raise_for_status() res = await response.json() @@ -76,7 +77,7 @@ class SerperWrapper(BaseModel): return json.dumps(payloads, sort_keys=True) def get_headers(self) -> Dict[str, str]: - headers = {"X-API-KEY": self.serper_api_key, "Content-Type": "application/json"} + headers = {"X-API-KEY": self.api_key, "Content-Type": "application/json"} return headers @staticmethod diff --git a/metagpt/tools/web_browser_engine.py b/metagpt/tools/web_browser_engine.py index 411c1604b..ad6db3ff4 100644 --- a/metagpt/tools/web_browser_engine.py +++ b/metagpt/tools/web_browser_engine.py @@ -1,36 +1,49 @@ #!/usr/bin/env python # -*- coding: utf-8 -*- - from __future__ import annotations import importlib -from typing import Any, Callable, Coroutine, overload +from typing import Any, Callable, Coroutine, Optional, Union, overload +from pydantic import BaseModel, ConfigDict, model_validator + +from metagpt.configs.browser_config import BrowserConfig from metagpt.tools import WebBrowserEngineType from metagpt.utils.parse_html import WebPage -class WebBrowserEngine: - def __init__( - self, - engine: WebBrowserEngineType = WebBrowserEngineType.PLAYWRIGHT, - run_func: Callable[..., Coroutine[Any, Any, WebPage | list[WebPage]]] | None = None, - ): - if engine is None: - raise NotImplementedError +class WebBrowserEngine(BaseModel): + model_config = ConfigDict(arbitrary_types_allowed=True, extra="allow") - if WebBrowserEngineType(engine) is WebBrowserEngineType.PLAYWRIGHT: + engine: WebBrowserEngineType = WebBrowserEngineType.PLAYWRIGHT + run_func: Optional[Callable[..., Coroutine[Any, Any, Union[WebPage, list[WebPage]]]]] = None + proxy: Optional[str] = None + + @model_validator(mode="after") + def validate_extra(self): + data = self.model_dump(exclude={"engine"}, exclude_none=True, exclude_defaults=True) + if self.model_extra: + data.update(self.model_extra) + self._process_extra(**data) + return self + + def _process_extra(self, **kwargs): + if self.engine is WebBrowserEngineType.PLAYWRIGHT: module = "metagpt.tools.web_browser_engine_playwright" - run_func = importlib.import_module(module).PlaywrightWrapper().run - elif WebBrowserEngineType(engine) is WebBrowserEngineType.SELENIUM: + run_func = importlib.import_module(module).PlaywrightWrapper(**kwargs).run + elif self.engine is WebBrowserEngineType.SELENIUM: module = "metagpt.tools.web_browser_engine_selenium" - run_func = importlib.import_module(module).SeleniumWrapper().run - elif WebBrowserEngineType(engine) is WebBrowserEngineType.CUSTOM: - run_func = run_func + run_func = importlib.import_module(module).SeleniumWrapper(**kwargs).run + elif self.engine is WebBrowserEngineType.CUSTOM: + run_func = self.run_func else: raise NotImplementedError self.run_func = run_func - self.engine = engine + + @classmethod + def from_browser_config(cls, config: BrowserConfig, **kwargs): + data = config.model_dump() + return cls(**data, **kwargs) @overload async def run(self, url: str) -> WebPage: diff --git a/metagpt/tools/web_browser_engine_playwright.py b/metagpt/tools/web_browser_engine_playwright.py index f8dabd5ac..2df288b1a 100644 --- a/metagpt/tools/web_browser_engine_playwright.py +++ b/metagpt/tools/web_browser_engine_playwright.py @@ -6,16 +6,16 @@ from __future__ import annotations import asyncio import sys from pathlib import Path -from typing import Literal +from typing import Literal, Optional from playwright.async_api import async_playwright +from pydantic import BaseModel, Field, PrivateAttr -from metagpt.config2 import config from metagpt.logs import logger from metagpt.utils.parse_html import WebPage -class PlaywrightWrapper: +class PlaywrightWrapper(BaseModel): """Wrapper around Playwright. To use this module, you should have the `playwright` Python package installed and ensure that @@ -24,24 +24,23 @@ class PlaywrightWrapper: command `playwright install` for the first time. """ - def __init__( - self, - browser_type: Literal["chromium", "firefox", "webkit"] | None = "chromium", - launch_kwargs: dict | None = None, - **kwargs, - ) -> None: - self.browser_type = browser_type - launch_kwargs = launch_kwargs or {} - if config.proxy and "proxy" not in launch_kwargs: + browser_type: Literal["chromium", "firefox", "webkit"] = "chromium" + launch_kwargs: dict = Field(default_factory=dict) + proxy: Optional[str] = None + context_kwargs: dict = Field(default_factory=dict) + _has_run_precheck: bool = PrivateAttr(False) + + def __init__(self, **kwargs): + super().__init__(**kwargs) + + launch_kwargs = self.launch_kwargs + if self.proxy and "proxy" not in launch_kwargs: args = launch_kwargs.get("args", []) if not any(str.startswith(i, "--proxy-server=") for i in args): - launch_kwargs["proxy"] = {"server": config.proxy} - self.launch_kwargs = launch_kwargs - context_kwargs = {} + launch_kwargs["proxy"] = {"server": self.proxy} + if "ignore_https_errors" in kwargs: - context_kwargs["ignore_https_errors"] = kwargs["ignore_https_errors"] - self._context_kwargs = context_kwargs - self._has_run_precheck = False + self.context_kwargs["ignore_https_errors"] = kwargs["ignore_https_errors"] async def run(self, url: str, *urls: str) -> WebPage | list[WebPage]: async with async_playwright() as ap: @@ -55,7 +54,7 @@ class PlaywrightWrapper: return await _scrape(browser, url) async def _scrape(self, browser, url): - context = await browser.new_context(**self._context_kwargs) + context = await browser.new_context(**self.context_kwargs) page = await context.new_page() async with page: try: @@ -75,8 +74,8 @@ class PlaywrightWrapper: executable_path = Path(browser_type.executable_path) if not executable_path.exists() and "executable_path" not in self.launch_kwargs: kwargs = {} - if config.proxy: - kwargs["env"] = {"ALL_PROXY": config.proxy} + if self.proxy: + kwargs["env"] = {"ALL_PROXY": self.proxy} await _install_browsers(self.browser_type, **kwargs) if self._has_run_precheck: diff --git a/metagpt/tools/web_browser_engine_selenium.py b/metagpt/tools/web_browser_engine_selenium.py index 02dd5c173..3b1682291 100644 --- a/metagpt/tools/web_browser_engine_selenium.py +++ b/metagpt/tools/web_browser_engine_selenium.py @@ -7,19 +7,19 @@ import asyncio import importlib from concurrent import futures from copy import deepcopy -from typing import Literal +from typing import Callable, Literal, Optional +from pydantic import BaseModel, ConfigDict, Field, PrivateAttr from selenium.webdriver.common.by import By from selenium.webdriver.support import expected_conditions as EC from selenium.webdriver.support.wait import WebDriverWait from webdriver_manager.core.download_manager import WDMDownloadManager from webdriver_manager.core.http import WDMHttpClient -from metagpt.config2 import config from metagpt.utils.parse_html import WebPage -class SeleniumWrapper: +class SeleniumWrapper(BaseModel): """Wrapper around Selenium. To use this module, you should check the following: @@ -31,25 +31,28 @@ class SeleniumWrapper: can scrape web pages using the Selenium WebBrowserEngine. """ - def __init__( - self, - browser_type: Literal["chrome", "firefox", "edge", "ie"] = "chrome", - launch_kwargs: dict | None = None, - *, - loop: asyncio.AbstractEventLoop | None = None, - executor: futures.Executor | None = None, - ) -> None: - self.browser_type = browser_type - launch_kwargs = launch_kwargs or {} - if config.proxy and "proxy-server" not in launch_kwargs: - launch_kwargs["proxy-server"] = config.proxy + model_config = ConfigDict(arbitrary_types_allowed=True) - self.executable_path = launch_kwargs.pop("executable_path", None) - self.launch_args = [f"--{k}={v}" for k, v in launch_kwargs.items()] - self._has_run_precheck = False - self._get_driver = None - self.loop = loop - self.executor = executor + browser_type: Literal["chrome", "firefox", "edge", "ie"] = "chrome" + launch_kwargs: dict = Field(default_factory=dict) + proxy: Optional[str] = None + loop: Optional[asyncio.AbstractEventLoop] = None + executor: Optional[futures.Executor] = None + _has_run_precheck: bool = PrivateAttr(False) + _get_driver: Optional[Callable] = PrivateAttr(None) + + def __init__(self, **kwargs) -> None: + super().__init__(**kwargs) + if self.proxy and "proxy-server" not in self.launch_kwargs: + self.launch_kwargs["proxy-server"] = self.proxy + + @property + def launch_args(self): + return [f"--{k}={v}" for k, v in self.launch_kwargs.items() if k != "executable_path"] + + @property + def executable_path(self): + return self.launch_kwargs.get("executable_path") async def run(self, url: str, *urls: str) -> WebPage | list[WebPage]: await self._run_precheck() @@ -66,7 +69,9 @@ class SeleniumWrapper: self.loop = self.loop or asyncio.get_event_loop() self._get_driver = await self.loop.run_in_executor( self.executor, - lambda: _gen_get_driver_func(self.browser_type, *self.launch_args, executable_path=self.executable_path), + lambda: _gen_get_driver_func( + self.browser_type, *self.launch_args, executable_path=self.executable_path, proxy=self.proxy + ), ) self._has_run_precheck = True @@ -92,13 +97,17 @@ _webdriver_manager_types = { class WDMHttpProxyClient(WDMHttpClient): + def __init__(self, proxy: str = None): + super().__init__() + self.proxy = proxy + def get(self, url, **kwargs): - if "proxies" not in kwargs and config.proxy: - kwargs["proxies"] = {"all_proxy": config.proxy} + if "proxies" not in kwargs and self.proxy: + kwargs["proxies"] = {"all_proxy": self.proxy} return super().get(url, **kwargs) -def _gen_get_driver_func(browser_type, *args, executable_path=None): +def _gen_get_driver_func(browser_type, *args, executable_path=None, proxy=None): WebDriver = getattr(importlib.import_module(f"selenium.webdriver.{browser_type}.webdriver"), "WebDriver") Service = getattr(importlib.import_module(f"selenium.webdriver.{browser_type}.service"), "Service") Options = getattr(importlib.import_module(f"selenium.webdriver.{browser_type}.options"), "Options") @@ -106,7 +115,7 @@ def _gen_get_driver_func(browser_type, *args, executable_path=None): if not executable_path: module_name, type_name = _webdriver_manager_types[browser_type] DriverManager = getattr(importlib.import_module(module_name), type_name) - driver_manager = DriverManager(download_manager=WDMDownloadManager(http_client=WDMHttpProxyClient())) + driver_manager = DriverManager(download_manager=WDMDownloadManager(http_client=WDMHttpProxyClient(proxy=proxy))) # driver_manager.driver_cache.find_driver(driver_manager.driver)) executable_path = driver_manager.install() diff --git a/tests/metagpt/actions/test_research.py b/tests/metagpt/actions/test_research.py index 372a1e876..ed83ce58c 100644 --- a/tests/metagpt/actions/test_research.py +++ b/tests/metagpt/actions/test_research.py @@ -28,9 +28,9 @@ async def test_collect_links(mocker, search_engine_mocker, context): return "[1,2]" mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", mock_llm_ask) - resp = await research.CollectLinks(search_engine=SearchEngine(SearchEngineType.DUCK_DUCK_GO), context=context).run( - "The application of MetaGPT" - ) + resp = await research.CollectLinks( + search_engine=SearchEngine(engine=SearchEngineType.DUCK_DUCK_GO), context=context + ).run("The application of MetaGPT") for i in ["MetaGPT use cases", "The roadmap of MetaGPT", "The function of MetaGPT", "What llm MetaGPT support"]: assert i in resp @@ -50,7 +50,9 @@ async def test_collect_links_with_rank_func(mocker, search_engine_mocker, contex mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", mock_collect_links_llm_ask) resp = await research.CollectLinks( - search_engine=SearchEngine(SearchEngineType.DUCK_DUCK_GO), rank_func=rank_func, context=context + search_engine=SearchEngine(engine=SearchEngineType.DUCK_DUCK_GO), + rank_func=rank_func, + context=context, ).run("The application of MetaGPT") for x, y, z in zip(rank_before, rank_after, resp.values()): assert x[::-1] == y diff --git a/tests/metagpt/learn/test_google_search.py b/tests/metagpt/learn/test_google_search.py index 7fda6436a..70a146878 100644 --- a/tests/metagpt/learn/test_google_search.py +++ b/tests/metagpt/learn/test_google_search.py @@ -16,6 +16,6 @@ async def test_google_search(search_engine_mocker): result = await google_search( seed.input, engine=SearchEngineType.SERPER_GOOGLE, - serper_api_key="mock-serper-key", + api_key="mock-serper-key", ) assert result != "" diff --git a/tests/metagpt/roles/test_researcher.py b/tests/metagpt/roles/test_researcher.py index af81777ac..ba05e1296 100644 --- a/tests/metagpt/roles/test_researcher.py +++ b/tests/metagpt/roles/test_researcher.py @@ -36,7 +36,7 @@ async def test_researcher(mocker, search_engine_mocker, context): role = researcher.Researcher(context=context) for i in role.actions: if isinstance(i, CollectLinks): - i.search_engine = SearchEngine(SearchEngineType.DUCK_DUCK_GO) + i.search_engine = SearchEngine(engine=SearchEngineType.DUCK_DUCK_GO) await role.run(topic) assert (researcher.RESEARCH_PATH / f"{topic}.md").read_text().startswith("# Research Report") diff --git a/tests/metagpt/tools/test_search_engine.py b/tests/metagpt/tools/test_search_engine.py index 966f53a38..a1f03ef7b 100644 --- a/tests/metagpt/tools/test_search_engine.py +++ b/tests/metagpt/tools/test_search_engine.py @@ -12,6 +12,7 @@ from typing import Callable import pytest from metagpt.config2 import config +from metagpt.configs.search_config import SearchConfig from metagpt.logs import logger from metagpt.tools import SearchEngineType from metagpt.tools.search_engine import SearchEngine @@ -49,27 +50,34 @@ async def test_search_engine( search_engine_mocker, ): # Prerequisites - search_engine_config = {} + search_engine_config = {"engine": search_engine_type, "run_func": run_func} if search_engine_type is SearchEngineType.SERPAPI_GOOGLE: assert config.search - search_engine_config["serpapi_api_key"] = "mock-serpapi-key" + search_engine_config["api_key"] = "mock-serpapi-key" elif search_engine_type is SearchEngineType.DIRECT_GOOGLE: assert config.search - search_engine_config["google_api_key"] = "mock-google-key" - search_engine_config["google_cse_id"] = "mock-google-cse" + search_engine_config["api_key"] = "mock-google-key" + search_engine_config["cse_id"] = "mock-google-cse" elif search_engine_type is SearchEngineType.SERPER_GOOGLE: assert config.search - search_engine_config["serper_api_key"] = "mock-serper-key" + search_engine_config["api_key"] = "mock-serper-key" - search_engine = SearchEngine(search_engine_type, run_func, **search_engine_config) - rsp = await search_engine.run("metagpt", max_results, as_string) - logger.info(rsp) - if as_string: - assert isinstance(rsp, str) - else: - assert isinstance(rsp, list) - assert len(rsp) <= max_results + async def test(search_engine): + rsp = await search_engine.run("metagpt", max_results, as_string) + logger.info(rsp) + if as_string: + assert isinstance(rsp, str) + else: + assert isinstance(rsp, list) + assert len(rsp) <= max_results + + await test(SearchEngine(**search_engine_config)) + search_engine_config["api_type"] = search_engine_config.pop("engine") + if run_func: + await test(SearchEngine.from_search_func(run_func)) + search_engine_config["search_func"] = search_engine_config.pop("run_func") + await test(SearchEngine.from_search_config(SearchConfig(**search_engine_config))) if __name__ == "__main__": diff --git a/tests/metagpt/tools/test_web_browser_engine_playwright.py b/tests/metagpt/tools/test_web_browser_engine_playwright.py index 0e838a2f8..f35848cf4 100644 --- a/tests/metagpt/tools/test_web_browser_engine_playwright.py +++ b/tests/metagpt/tools/test_web_browser_engine_playwright.py @@ -3,7 +3,6 @@ import pytest -from metagpt.config2 import config from metagpt.tools import web_browser_engine_playwright from metagpt.utils.parse_html import WebPage @@ -19,26 +18,22 @@ from metagpt.utils.parse_html import WebPage ids=["chromium-normal", "firefox-normal", "webkit-normal"], ) async def test_scrape_web_page(browser_type, use_proxy, kwagrs, url, urls, proxy, capfd): - global_proxy = config.proxy - try: - if use_proxy: - server, proxy_url = await proxy() - config.proxy = proxy_url - browser = web_browser_engine_playwright.PlaywrightWrapper(browser_type=browser_type, **kwagrs) - result = await browser.run(url) - assert isinstance(result, WebPage) - assert "MetaGPT" in result.inner_text + proxy_url = None + if use_proxy: + server, proxy_url = await proxy() + browser = web_browser_engine_playwright.PlaywrightWrapper(browser_type=browser_type, proxy=proxy_url, **kwagrs) + result = await browser.run(url) + assert isinstance(result, WebPage) + assert "MetaGPT" in result.inner_text - if urls: - results = await browser.run(url, *urls) - assert isinstance(results, list) - assert len(results) == len(urls) + 1 - assert all(("MetaGPT" in i.inner_text) for i in results) - if use_proxy: - server.close() - assert "Proxy:" in capfd.readouterr().out - finally: - config.proxy = global_proxy + if urls: + results = await browser.run(url, *urls) + assert isinstance(results, list) + assert len(results) == len(urls) + 1 + assert all(("MetaGPT" in i.inner_text) for i in results) + if use_proxy: + server.close() + assert "Proxy:" in capfd.readouterr().out if __name__ == "__main__": diff --git a/tests/metagpt/tools/test_web_browser_engine_selenium.py b/tests/metagpt/tools/test_web_browser_engine_selenium.py index 1b1439d29..a88a5d0f4 100644 --- a/tests/metagpt/tools/test_web_browser_engine_selenium.py +++ b/tests/metagpt/tools/test_web_browser_engine_selenium.py @@ -4,7 +4,6 @@ import browsers import pytest -from metagpt.config2 import config from metagpt.tools import web_browser_engine_selenium from metagpt.utils.parse_html import WebPage @@ -40,27 +39,22 @@ from metagpt.utils.parse_html import WebPage async def test_scrape_web_page(browser_type, use_proxy, url, urls, proxy, capfd): # Prerequisites # firefox, chrome, Microsoft Edge + proxy_url = None + if use_proxy: + server, proxy_url = await proxy() + browser = web_browser_engine_selenium.SeleniumWrapper(browser_type=browser_type, proxy=proxy_url) + result = await browser.run(url) + assert isinstance(result, WebPage) + assert "MetaGPT" in result.inner_text - global_proxy = config.proxy - try: - if use_proxy: - server, proxy_url = await proxy() - config.proxy = proxy_url - browser = web_browser_engine_selenium.SeleniumWrapper(browser_type=browser_type) - result = await browser.run(url) - assert isinstance(result, WebPage) - assert "MetaGPT" in result.inner_text - - if urls: - results = await browser.run(url, *urls) - assert isinstance(results, list) - assert len(results) == len(urls) + 1 - assert all(("MetaGPT" in i.inner_text) for i in results) - if use_proxy: - server.close() - assert "Proxy:" in capfd.readouterr().out - finally: - config.proxy = global_proxy + if urls: + results = await browser.run(url, *urls) + assert isinstance(results, list) + assert len(results) == len(urls) + 1 + assert all(("MetaGPT" in i.inner_text) for i in results) + if use_proxy: + server.close() + assert "Proxy:" in capfd.readouterr().out if __name__ == "__main__": diff --git a/tests/mock/mock_aiohttp.py b/tests/mock/mock_aiohttp.py index 49dcdba79..a7c022a4b 100644 --- a/tests/mock/mock_aiohttp.py +++ b/tests/mock/mock_aiohttp.py @@ -13,7 +13,8 @@ class MockAioResponse: def __init__(self, session, method, url, **kwargs) -> None: fn = self.check_funcs.get((method, url)) - self.key = f"{self.name}-{method}-{url}-{fn(kwargs) if fn else json.dumps(kwargs, sort_keys=True)}" + _kwargs = {k: v for k, v in kwargs.items() if k != "proxy"} + self.key = f"{self.name}-{method}-{url}-{fn(kwargs) if fn else json.dumps(_kwargs, sort_keys=True)}" self.mng = self.response = None if self.key not in self.rsp_cache: self.mng = origin_request(session, method, url, **kwargs) From e246a9cff9c87ddb80b2527d8769af6c90f3a638 Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Thu, 1 Feb 2024 16:56:45 +0800 Subject: [PATCH 565/637] update docs --- metagpt/tools/search_engine.py | 33 ++++++++++++++++ metagpt/tools/web_browser_engine.py | 58 +++++++++++++++++++++++++++++ 2 files changed, 91 insertions(+) diff --git a/metagpt/tools/search_engine.py b/metagpt/tools/search_engine.py index 880b6ca85..1e540bd0e 100644 --- a/metagpt/tools/search_engine.py +++ b/metagpt/tools/search_engine.py @@ -17,6 +17,12 @@ from metagpt.tools import SearchEngineType class SkSearchEngine: + """A search engine class for executing searches. + + Attributes: + search_engine: The search engine instance used for executing searches. + """ + def __init__(self, **kwargs): self.search_engine = SearchEngine(**kwargs) @@ -32,6 +38,16 @@ class SkSearchEngine: class SearchEngine(BaseModel): + """A model for configuring and executing searches with different search engines. + + Attributes: + model_config: Configuration for the model allowing arbitrary types. + engine: The type of search engine to use. + run_func: An optional callable for running the search. If not provided, it will be determined based on the engine. + api_key: An optional API key for the search engine. + proxy: An optional proxy for the search engine requests. + """ + model_config = ConfigDict(arbitrary_types_allowed=True, extra="allow") engine: SearchEngineType = SearchEngineType.SERPER_GOOGLE @@ -41,6 +57,7 @@ class SearchEngine(BaseModel): @model_validator(mode="after") def validate_extra(self): + """Validates extra fields provided to the model and updates the run function accordingly.""" data = self.model_dump(exclude={"engine"}, exclude_none=True, exclude_defaults=True) if self.model_extra: data.update(self.model_extra) @@ -52,6 +69,11 @@ class SearchEngine(BaseModel): run_func: Optional[Callable[[str, int, bool], Coroutine[None, None, Union[str, list[str]]]]] = None, **kwargs, ): + """Processes extra configuration and updates the run function based on the search engine type. + + Args: + run_func: An optional callable for running the search. If not provided, it will be determined based on the engine. + """ if self.engine == SearchEngineType.SERPAPI_GOOGLE: module = "metagpt.tools.search_engine_serpapi" run_func = importlib.import_module(module).SerpAPIWrapper(**kwargs).run @@ -72,6 +94,11 @@ class SearchEngine(BaseModel): @classmethod def from_search_config(cls, config: SearchConfig, **kwargs): + """Creates a SearchEngine instance from a SearchConfig. + + Args: + config: The search configuration to use for creating the SearchEngine instance. + """ data = config.model_dump(exclude={"api_type", "search_func"}) if config.search_func is not None: data["run_func"] = config.search_func @@ -82,6 +109,11 @@ class SearchEngine(BaseModel): def from_search_func( cls, search_func: Callable[[str, int, bool], Coroutine[None, None, Union[str, list[str]]]], **kwargs ): + """Creates a SearchEngine instance from a custom search function. + + Args: + search_func: A callable that executes the search. + """ return cls(engine=SearchEngineType.CUSTOM_ENGINE, run_func=search_func, **kwargs) @overload @@ -115,6 +147,7 @@ class SearchEngine(BaseModel): query: The search query. max_results: The maximum number of results to return. Defaults to 8. as_string: Whether to return the results as a string or a list of dictionaries. Defaults to True. + ignore_errors: Whether to ignore errors during the search. Defaults to False. Returns: The search results as a string or a list of dictionaries. diff --git a/metagpt/tools/web_browser_engine.py b/metagpt/tools/web_browser_engine.py index ad6db3ff4..01339e51a 100644 --- a/metagpt/tools/web_browser_engine.py +++ b/metagpt/tools/web_browser_engine.py @@ -13,6 +13,19 @@ from metagpt.utils.parse_html import WebPage class WebBrowserEngine(BaseModel): + """Defines a web browser engine configuration for automated browsing and data extraction. + + This class encapsulates the configuration and operational logic for different web browser engines, + such as Playwright, Selenium, or custom implementations. It provides a unified interface to run + browser automation tasks. + + Attributes: + model_config: Configuration dictionary allowing arbitrary types and extra fields. + engine: The type of web browser engine to use. + run_func: An optional coroutine function to run the browser engine. + proxy: An optional proxy server URL to use with the browser engine. + """ + model_config = ConfigDict(arbitrary_types_allowed=True, extra="allow") engine: WebBrowserEngineType = WebBrowserEngineType.PLAYWRIGHT @@ -21,6 +34,15 @@ class WebBrowserEngine(BaseModel): @model_validator(mode="after") def validate_extra(self): + """Validates and processes extra configuration data after model initialization. + + This method is automatically called by Pydantic to validate and process any extra configuration + data provided to the model. It ensures that the extra data is properly integrated into the model's + configuration and operational logic. + + Returns: + The instance itself after processing the extra data. + """ data = self.model_dump(exclude={"engine"}, exclude_none=True, exclude_defaults=True) if self.model_extra: data.update(self.model_extra) @@ -28,6 +50,17 @@ class WebBrowserEngine(BaseModel): return self def _process_extra(self, **kwargs): + """Processes extra configuration data to set up the browser engine run function. + + Depending on the specified engine type, this method dynamically imports and configures + the appropriate browser engine wrapper and its run function. + + Args: + **kwargs: Arbitrary keyword arguments representing extra configuration data. + + Raises: + NotImplementedError: If the engine type is not supported. + """ if self.engine is WebBrowserEngineType.PLAYWRIGHT: module = "metagpt.tools.web_browser_engine_playwright" run_func = importlib.import_module(module).PlaywrightWrapper(**kwargs).run @@ -42,6 +75,19 @@ class WebBrowserEngine(BaseModel): @classmethod def from_browser_config(cls, config: BrowserConfig, **kwargs): + """Creates a WebBrowserEngine instance from a BrowserConfig object and additional keyword arguments. + + This class method facilitates the creation of a WebBrowserEngine instance by extracting + configuration data from a BrowserConfig object and optionally merging it with additional + keyword arguments. + + Args: + config: A BrowserConfig object containing base configuration data. + **kwargs: Optional additional keyword arguments to override or extend the configuration. + + Returns: + A new instance of WebBrowserEngine configured according to the provided arguments. + """ data = config.model_dump() return cls(**data, **kwargs) @@ -54,4 +100,16 @@ class WebBrowserEngine(BaseModel): ... async def run(self, url: str, *urls: str) -> WebPage | list[WebPage]: + """Runs the browser engine to load one or more web pages. + + This method is the implementation of the overloaded run signatures. It delegates the task + of loading web pages to the configured run function, handling either a single URL or multiple URLs. + + Args: + url: The URL of the first web page to load. + *urls: Additional URLs of web pages to load, if any. + + Returns: + A WebPage object if a single URL is provided, or a list of WebPage objects if multiple URLs are provided. + """ return await self.run_func(url, *urls) From 9f4ee420791f1d7bf87ff22d10e46441d93af7da Mon Sep 17 00:00:00 2001 From: better629 Date: Thu, 1 Feb 2024 18:18:35 +0800 Subject: [PATCH 566/637] update env api schema --- metagpt/environment/api/env_api.py | 19 +++++++++++++++++-- metagpt/environment/base_env.py | 22 +++++++++++++++------- metagpt/utils/common.py | 8 ++++++++ tests/metagpt/environment/test_base_env.py | 4 ++++ 4 files changed, 44 insertions(+), 9 deletions(-) diff --git a/metagpt/environment/api/env_api.py b/metagpt/environment/api/env_api.py index 6469e5b4c..1e6df544d 100644 --- a/metagpt/environment/api/env_api.py +++ b/metagpt/environment/api/env_api.py @@ -2,7 +2,7 @@ # -*- coding: utf-8 -*- # @Desc : the environment api store -from typing import Callable +from typing import Any, Callable, Union from pydantic import BaseModel, Field @@ -18,9 +18,11 @@ class EnvAPIAbstract(BaseModel): class EnvAPIRegistry(BaseModel): """the registry to store environment w&r api/interface""" - registry: dict[str, Callable] = Field(default=dict(), exclude=True) + registry: dict[str, dict[str, Union[dict, Any, str]]] = Field(default=dict(), exclude=True) def get(self, api_name: str): + if api_name not in self.registry: + raise ValueError return self.registry.get(api_name) def __getitem__(self, api_name: str) -> Callable: @@ -32,6 +34,19 @@ class EnvAPIRegistry(BaseModel): def __len__(self): return len(self.registry) + def get_apis(self, as_str=True) -> dict[str, dict[str, Union[dict, Any, str]]]: + """return func schema without func instance""" + apis = dict() + for func_name, func_schema in self.registry.items(): + new_func_schema = dict() + for key, value in func_schema.items(): + if key == "func": + continue + new_func_schema[key] = str(value) if as_str else value + new_func_schema = new_func_schema + apis[func_name] = new_func_schema + return apis + class WriteAPIRegistry(EnvAPIRegistry): """just as a explicit class name""" diff --git a/metagpt/environment/base_env.py b/metagpt/environment/base_env.py index 1bdcfe373..7ba34dfaf 100644 --- a/metagpt/environment/base_env.py +++ b/metagpt/environment/base_env.py @@ -4,7 +4,7 @@ import asyncio from enum import Enum -from typing import Iterable, Optional, Set, Union +from typing import Any, Iterable, Optional, Set, Union from pydantic import BaseModel, ConfigDict, Field, SerializeAsAny, model_validator @@ -17,7 +17,7 @@ from metagpt.environment.api.env_api import ( from metagpt.logs import logger from metagpt.roles.role import Role from metagpt.schema import Message -from metagpt.utils.common import is_coroutine_func, is_send_to +from metagpt.utils.common import get_function_schema, is_coroutine_func, is_send_to class EnvType(Enum): @@ -34,13 +34,13 @@ env_read_api_registry = ReadAPIRegistry() def mark_as_readable(func): """mark functionn as a readable one in ExtEnv, it observes something from ExtEnv""" - env_read_api_registry[func.__name__] = func + env_read_api_registry[func.__name__] = get_function_schema(func) return func def mark_as_writeable(func): """mark functionn as a writeable one in ExtEnv, it does something to ExtEnv""" - env_write_api_registry[func.__name__] = func + env_write_api_registry[func.__name__] = get_function_schema(func) return func @@ -51,17 +51,25 @@ class ExtEnv(BaseModel): if not rw_api: raise ValueError(f"{rw_api} not exists") + def get_all_available_apis(self, mode: str = "read") -> list[Any]: + """get available read/write apis definition""" + assert mode in ["read", "write"] + if mode == "read": + return env_read_api_registry.get_apis() + else: + return env_write_api_registry.get_apis() + async def observe(self, env_action: Union[str, EnvAPIAbstract]): """get observation from particular api of ExtEnv""" if isinstance(env_action, str): - read_api = env_read_api_registry.get(api_name=env_action) + read_api = env_read_api_registry.get(api_name=env_action)["func"] self._check_api_exist(read_api) if is_coroutine_func(read_api): res = await read_api(self) else: res = read_api(self) elif isinstance(env_action, EnvAPIAbstract): - read_api = env_read_api_registry.get(api_name=env_action.api_name) + read_api = env_read_api_registry.get(api_name=env_action.api_name)["func"] self._check_api_exist(read_api) if is_coroutine_func(read_api): res = await read_api(self, *env_action.args, **env_action.kwargs) @@ -75,7 +83,7 @@ class ExtEnv(BaseModel): if isinstance(env_action, Message): self.publish_message(env_action) elif isinstance(env_action, EnvAPIAbstract): - write_api = env_write_api_registry.get(env_action.api_name) + write_api = env_write_api_registry.get(env_action.api_name)["func"] self._check_api_exist(write_api) if is_coroutine_func(write_api): res = await write_api(self, *env_action.args, **env_action.kwargs) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 2e05afa74..d3a922c5f 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -340,6 +340,14 @@ def print_members(module, indent=0): print(f"{prefix}Method: {name}") +def get_function_schema(func: Callable) -> dict[str, Union[dict, Any, str]]: + sig = inspect.signature(func) + parameters = sig.parameters + return_type = sig.return_annotation + param_schema = {name: parameter.annotation for name, parameter in parameters.items()} + return {"input_params": param_schema, "return_type": return_type, "func_desc": func.__doc__, "func": func} + + def parse_recipient(text): # FIXME: use ActionNode instead. pattern = r"## Send To:\s*([A-Za-z]+)\s*?" # hard code for now diff --git a/tests/metagpt/environment/test_base_env.py b/tests/metagpt/environment/test_base_env.py index ce8165f2f..fd73679d8 100644 --- a/tests/metagpt/environment/test_base_env.py +++ b/tests/metagpt/environment/test_base_env.py @@ -40,6 +40,10 @@ async def test_ext_env(): assert len(env_read_api_registry) > 0 assert len(env_write_api_registry) > 0 + apis = env.get_all_available_apis(mode="read") + assert len(apis) > 0 + assert len(apis["read_api"]) == 3 + _ = await env.step(EnvAPIAbstract(api_name="write_api", kwargs={"a": 5, "b": 10})) assert env.value == 15 From 2abc14aee6cac32ccdbe9814388e0afb8057dfa2 Mon Sep 17 00:00:00 2001 From: better629 Date: Thu, 1 Feb 2024 18:30:16 +0800 Subject: [PATCH 567/637] update const value import path --- metagpt/environment/android_env/android_ext_env.py | 2 +- metagpt/environment/mincraft_env/mincraft_env.py | 2 +- metagpt/environment/mincraft_env/mincraft_ext_env.py | 4 ++-- tests/metagpt/environment/android_env/test_android_ext_env.py | 2 +- .../metagpt/environment/mincraft_env/test_mincraft_ext_env.py | 2 +- 5 files changed, 6 insertions(+), 6 deletions(-) diff --git a/metagpt/environment/android_env/android_ext_env.py b/metagpt/environment/android_env/android_ext_env.py index 7467d394c..b81b2cd26 100644 --- a/metagpt/environment/android_env/android_ext_env.py +++ b/metagpt/environment/android_env/android_ext_env.py @@ -8,7 +8,7 @@ from typing import Any, Optional from pydantic import Field -from metagpt.const import ADB_EXEC_FAIL +from metagpt.environment.android_env.const import ADB_EXEC_FAIL from metagpt.environment.base_env import ExtEnv, mark_as_readable, mark_as_writeable diff --git a/metagpt/environment/mincraft_env/mincraft_env.py b/metagpt/environment/mincraft_env/mincraft_env.py index 0248244c5..6327aa3f4 100644 --- a/metagpt/environment/mincraft_env/mincraft_env.py +++ b/metagpt/environment/mincraft_env/mincraft_env.py @@ -13,8 +13,8 @@ from langchain.vectorstores import Chroma from pydantic import ConfigDict, Field from metagpt.config2 import config as CONFIG -from metagpt.const import MC_CKPT_DIR from metagpt.environment.base_env import Environment +from metagpt.environment.mincraft_env.const import MC_CKPT_DIR from metagpt.environment.mincraft_env.mincraft_ext_env import MincraftExtEnv from metagpt.logs import logger from metagpt.utils.common import load_mc_skills_code, read_json_file, write_json_file diff --git a/metagpt/environment/mincraft_env/mincraft_ext_env.py b/metagpt/environment/mincraft_env/mincraft_ext_env.py index 4934e34b2..b86250d8c 100644 --- a/metagpt/environment/mincraft_env/mincraft_ext_env.py +++ b/metagpt/environment/mincraft_env/mincraft_ext_env.py @@ -10,14 +10,14 @@ from typing import Optional import requests from pydantic import ConfigDict, Field, model_validator -from metagpt.const import ( +from metagpt.environment.base_env import ExtEnv, mark_as_writeable +from metagpt.environment.mincraft_env.const import ( MC_CKPT_DIR, MC_CORE_INVENTORY_ITEMS, MC_CURRICULUM_OB, MC_DEFAULT_WARMUP, METAGPT_ROOT, ) -from metagpt.environment.base_env import ExtEnv, mark_as_writeable from metagpt.environment.mincraft_env.process_monitor import SubprocessMonitor from metagpt.logs import logger diff --git a/tests/metagpt/environment/android_env/test_android_ext_env.py b/tests/metagpt/environment/android_env/test_android_ext_env.py index 955210868..c9dfc718b 100644 --- a/tests/metagpt/environment/android_env/test_android_ext_env.py +++ b/tests/metagpt/environment/android_env/test_android_ext_env.py @@ -4,8 +4,8 @@ from pathlib import Path -from metagpt.const import ADB_EXEC_FAIL from metagpt.environment.android_env.android_ext_env import AndroidExtEnv +from metagpt.environment.android_env.const import ADB_EXEC_FAIL def mock_device_shape(self, adb_cmd: str) -> str: diff --git a/tests/metagpt/environment/mincraft_env/test_mincraft_ext_env.py b/tests/metagpt/environment/mincraft_env/test_mincraft_ext_env.py index b01168d42..ad3376141 100644 --- a/tests/metagpt/environment/mincraft_env/test_mincraft_ext_env.py +++ b/tests/metagpt/environment/mincraft_env/test_mincraft_ext_env.py @@ -3,7 +3,7 @@ # @Desc : the unittest of MincraftExtEnv -from metagpt.const import MC_CKPT_DIR +from metagpt.environment.mincraft_env.const import MC_CKPT_DIR from metagpt.environment.mincraft_env.mincraft_ext_env import MincraftExtEnv From 2897981e6310dd3a197827ddeb5cc218a0ce20ef Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 1 Feb 2024 20:07:44 +0800 Subject: [PATCH 568/637] task utils etc. --- metagpt/actions/debug_code.py | 6 +----- metagpt/plan/planner.py | 2 +- metagpt/schema.py | 23 +++++++++++++---------- metagpt/utils/save_code.py | 3 +-- 4 files changed, 16 insertions(+), 18 deletions(-) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index d63fa3396..0dc3ce919 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -72,12 +72,8 @@ CODE_REFLECTION = { } -def message_to_str(message: Message) -> str: - return f"{message.role}: {message.content}" - - def messages_to_str(messages: List[Message]) -> str: - return "\n".join([message_to_str(message) for message in messages]) + return "\n".join([str(message) for message in messages]) class DebugCode(BaseWriteAnalysisCode): diff --git a/metagpt/plan/planner.py b/metagpt/plan/planner.py index 6e866ec22..0b3a05199 100644 --- a/metagpt/plan/planner.py +++ b/metagpt/plan/planner.py @@ -111,7 +111,7 @@ class Planner(BaseModel): return "", confirmed async def confirm_task(self, task: Task, task_result: TaskResult, review: str): - self.plan.update_task_result(task=task, task_result=task_result) + task.update_task_result(task_result=task_result) self.plan.finish_current_task() self.working_memory.clear() diff --git a/metagpt/schema.py b/metagpt/schema.py index 08f97be94..1b0be279c 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -341,6 +341,18 @@ class Task(BaseModel): is_success: bool = False is_finished: bool = False + def reset(self): + self.code = "" + self.result = "" + self.is_success = False + self.is_finished = False + + def update_task_result(self, task_result: TaskResult): + self.code_steps = task_result.code_steps + self.code = task_result.code + self.result = task_result.result + self.is_success = task_result.is_success + class TaskResult(BaseModel): """Result of taking a task, with result and is_success required to be filled""" @@ -434,10 +446,7 @@ class Plan(BaseModel): """ if task_id in self.task_map: task = self.task_map[task_id] - task.code = "" - task.result = "" - task.is_success = False - task.is_finished = False + task.reset() def replace_task(self, new_task: Task): """ @@ -483,12 +492,6 @@ class Plan(BaseModel): self.task_map[new_task.task_id] = new_task self._update_current_task() - def update_task_result(self, task: Task, task_result: TaskResult): - task.code_steps = task_result.code_steps - task.code = task_result.code - task.result = task_result.result - task.is_success = task_result.is_success - def has_task_id(self, task_id: str) -> bool: return task_id in self.task_map diff --git a/metagpt/utils/save_code.py b/metagpt/utils/save_code.py index adf136316..d55b058e6 100644 --- a/metagpt/utils/save_code.py +++ b/metagpt/utils/save_code.py @@ -29,8 +29,7 @@ def save_code_file(name: str, code_context: str, file_format: str = "py") -> Non # Choose to save as a Python file or a JSON file based on the file format file_path = DATA_PATH / "output" / f"{name}/code.{file_format}" if file_format == "py": - with open(file_path, "w", encoding="utf-8") as fp: - fp.write(code_context + "\n\n") + file_path.write_text(code_context + "\n\n", encoding="utf-8") elif file_format == "json": # Parse the code content as JSON and save data = {"code": code_context} From 4cd09e703c829196e44cd0ed40da2177b0617c68 Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 1 Feb 2024 20:25:53 +0800 Subject: [PATCH 569/637] file read write utils --- metagpt/utils/common.py | 4 ++-- metagpt/utils/save_code.py | 5 ++--- tests/metagpt/utils/test_save_code.py | 13 +++++-------- 3 files changed, 9 insertions(+), 13 deletions(-) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 55f4ce378..9d6a6bb24 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -485,13 +485,13 @@ def read_json_file(json_file: str, encoding="utf-8") -> list[Any]: return data -def write_json_file(json_file: str, data: list, encoding=None): +def write_json_file(json_file: str, data: list, encoding: str = None, indent: int = 4): folder_path = Path(json_file).parent if not folder_path.exists(): folder_path.mkdir(parents=True, exist_ok=True) with open(json_file, "w", encoding=encoding) as fout: - json.dump(data, fout, ensure_ascii=False, indent=4, default=to_jsonable_python) + json.dump(data, fout, ensure_ascii=False, indent=indent, default=to_jsonable_python) def import_class(class_name: str, module_name: str) -> type: diff --git a/metagpt/utils/save_code.py b/metagpt/utils/save_code.py index d55b058e6..18cb5cd62 100644 --- a/metagpt/utils/save_code.py +++ b/metagpt/utils/save_code.py @@ -2,12 +2,12 @@ # @Date : 12/12/2023 4:14 PM # @Author : stellahong (stellahong@fuzhi.ai) # @Desc : -import json import os import nbformat from metagpt.const import DATA_PATH +from metagpt.utils.common import write_json_file def save_code_file(name: str, code_context: str, file_format: str = "py") -> None: @@ -33,8 +33,7 @@ def save_code_file(name: str, code_context: str, file_format: str = "py") -> Non elif file_format == "json": # Parse the code content as JSON and save data = {"code": code_context} - with open(file_path, "w", encoding="utf-8") as fp: - json.dump(data, fp, indent=2) + write_json_file(file_path, data, encoding="utf-8", indent=2) elif file_format == "ipynb": nbformat.write(code_context, file_path) else: diff --git a/tests/metagpt/utils/test_save_code.py b/tests/metagpt/utils/test_save_code.py index bb0b07d63..62724dde5 100644 --- a/tests/metagpt/utils/test_save_code.py +++ b/tests/metagpt/utils/test_save_code.py @@ -2,30 +2,27 @@ # @Date : 12/12/2023 4:17 PM # @Author : stellahong (stellahong@fuzhi.ai) # @Desc : -import json -import os import nbformat import pytest from metagpt.actions.execute_nb_code import ExecuteNbCode +from metagpt.utils.common import read_json_file from metagpt.utils.save_code import DATA_PATH, save_code_file def test_save_code_file_python(): save_code_file("example", "print('Hello, World!')") file_path = DATA_PATH / "output" / "example" / "code.py" - assert os.path.exists(file_path), f"File does not exist: {file_path}" - with open(file_path, "r", encoding="utf-8") as fp: - content = fp.read() + assert file_path.exists, f"File does not exist: {file_path}" + content = file_path.read_text() assert "print('Hello, World!')" in content, "File content does not match" def test_save_code_file_json(): save_code_file("example_json", "print('Hello, JSON!')", file_format="json") file_path = DATA_PATH / "output" / "example_json" / "code.json" - with open(file_path, "r", encoding="utf-8") as fp: - data = json.load(fp) + data = read_json_file(file_path) assert "code" in data, "JSON key 'code' is missing" assert data["code"] == "print('Hello, JSON!')", "JSON content does not match" @@ -38,7 +35,7 @@ async def test_save_code_file_notebook(): # Save as a Notebook file save_code_file("example_nb", executor.nb, file_format="ipynb") file_path = DATA_PATH / "output" / "example_nb" / "code.ipynb" - assert os.path.exists(file_path), f"Notebook file does not exist: {file_path}" + assert file_path.exists, f"Notebook file does not exist: {file_path}" # Additional checks specific to notebook format notebook = nbformat.read(file_path, as_version=4) From 1a1610a67edbeeb133f1d4d5858851d409805fbd Mon Sep 17 00:00:00 2001 From: yzlin Date: Thu, 1 Feb 2024 22:23:28 +0800 Subject: [PATCH 570/637] add more comments --- metagpt/actions/debug_code.py | 9 ++++--- metagpt/actions/execute_nb_code.py | 7 +++--- metagpt/actions/ml_action.py | 33 +------------------------- metagpt/actions/write_analysis_code.py | 2 +- 4 files changed, 9 insertions(+), 42 deletions(-) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 0dc3ce919..9a8b4c122 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -81,9 +81,9 @@ class DebugCode(BaseWriteAnalysisCode): async def run_reflection( self, - context: List[Message], - code, - runtime_result, + context: list[Message], + code: str, + runtime_result: str, ) -> dict: info = [] reflection_prompt = REFLECTION_PROMPT.format( @@ -107,12 +107,11 @@ class DebugCode(BaseWriteAnalysisCode): runtime_result: str = "", ) -> str: """ - 根据当前运行代码和报错信息进行reflection和纠错 + use reflection to debug, based on current code and the execution errors """ reflection = await self.run_reflection( code=code, context=context, runtime_result=runtime_result, ) - # 根据reflection结果重写代码 return {"code": reflection["improved_impl"]} diff --git a/metagpt/actions/execute_nb_code.py b/metagpt/actions/execute_nb_code.py index 7dfbecb5c..835233dfa 100644 --- a/metagpt/actions/execute_nb_code.py +++ b/metagpt/actions/execute_nb_code.py @@ -5,6 +5,7 @@ @File : code_executor.py """ import asyncio +import base64 import re import traceback from pathlib import Path @@ -117,8 +118,6 @@ class ExecuteNbCode(Action): return parsed_output def show_bytes_figure(self, image_base64: str, interaction_type: str = "ipython"): - import base64 - image_bytes = base64.b64decode(image_base64) if interaction_type == "ipython": from IPython.display import Image, display @@ -145,8 +144,8 @@ class ExecuteNbCode(Action): # 如果在Python脚本中运行,__file__ 变量存在 return False - def _process_code(self, code: Union[str, Dict], language: str = None) -> Tuple: - language = language or "python" + def _process_code(self, code: Union[str, Dict], language: str = "python") -> Tuple: + """handle different code response formats, support str or dict""" if isinstance(code, str) and Path(code).suffix in (".py", ".txt"): code = Path(code).read_text(encoding="utf-8") return code, language diff --git a/metagpt/actions/ml_action.py b/metagpt/actions/ml_action.py index d419026fa..88476707c 100644 --- a/metagpt/actions/ml_action.py +++ b/metagpt/actions/ml_action.py @@ -1,4 +1,3 @@ -import json from typing import List, Tuple from metagpt.actions import Action @@ -11,7 +10,7 @@ from metagpt.prompts.ml_action import ( ) from metagpt.prompts.write_analysis_code import CODE_GENERATOR_WITH_TOOLS from metagpt.schema import Message, Plan -from metagpt.utils.common import CodeParser, create_func_call_config, remove_comments +from metagpt.utils.common import create_func_call_config, remove_comments class WriteCodeWithToolsML(WriteCodeWithTools): @@ -61,36 +60,6 @@ class WriteCodeWithToolsML(WriteCodeWithTools): return context, rsp -class Reflect(Action): - PROMPT_TEMPLATE: str = """ - # Context - __context__ - # Latest User Requirement - __user_requirement__ - # Summary - Above is all your attempts to tackle the user requirement. You plan, act, submit your output, and get the result and feedback. - Output a json following the format: - ```json - { - "summary": str = "summarize each of your previous trial in a triple of (your methods, the corresponding result, potential improvement), list them out", - "takeaways": str = "carefully find key takeaways from your summarization", - "reflection": str = "give specific instruction to improve your next trial in a step-by-step thinking process", - } - ``` - """ - REWRITE_PLAN_INSTRUCTION: str = """Take this reflection for rewriting plan, modify the current plan in place, make reference to your specific instruction, think about you should - change which task, add or delete what tasks in the plan. Only make necessary changes, keep reusable tasks unchanged, output the COMPLETE new plan starting from the first task. Your plan should have no more than 5 tasks.""" - - async def run(self, context: str, user_requirement: str = "") -> str: - user_requirement = user_requirement or "Score as high as possible in a data modeling competition" - # prompt = self.PROMPT_TEMPLATE.format(context=context, user_requirement=user_requirement) - prompt = self.PROMPT_TEMPLATE.replace("__context__", context).replace("__user_requirement__", user_requirement) - rsp_json = await self._aask(prompt) - rsp = CodeParser.parse_code(block=None, text=rsp_json) - reflection = json.loads(rsp)["reflection"] - return reflection - - class UpdateDataColumns(Action): async def run(self, plan: Plan = None) -> dict: finished_tasks = plan.get_finished_tasks() diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index bf00e8ed1..c47685bdf 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -67,7 +67,7 @@ class BaseWriteAnalysisCode(Action): class WriteCodeByGenerate(BaseWriteAnalysisCode): - """Write code fully by generation""" + """Ask LLM to generate codes purely by itself without local user-defined tools""" async def run( self, From a515f701e1e3c4110f12018dab27bb4c1efe3fe7 Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Fri, 2 Feb 2024 13:55:15 +0800 Subject: [PATCH 571/637] add the browser config example --- config/config2.yaml.example | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/config/config2.yaml.example b/config/config2.yaml.example index bead3c626..8f4a33fc1 100644 --- a/config/config2.yaml.example +++ b/config/config2.yaml.example @@ -11,6 +11,10 @@ search: api_key: "YOUR_API_KEY" cse_id: "YOUR_CSE_ID" +browser: + engine: "playwright" # playwright/selenium + browser_type: "chromium" # playwright: chromium/firefox/webkit; selenium: chrome/firefox/edge/ie + mermaid: engine: "pyppeteer" path: "/Applications/Google Chrome.app" From 5d95447cb92ab689741c0a6e2c4b5c82381cb30f Mon Sep 17 00:00:00 2001 From: better629 Date: Fri, 2 Feb 2024 14:17:13 +0800 Subject: [PATCH 572/637] add README to guide followed development --- metagpt/environment/README.md | 38 +++++++++++++++++++++++++++++++++++ 1 file changed, 38 insertions(+) create mode 100644 metagpt/environment/README.md diff --git a/metagpt/environment/README.md b/metagpt/environment/README.md new file mode 100644 index 000000000..9476ac75a --- /dev/null +++ b/metagpt/environment/README.md @@ -0,0 +1,38 @@ +Here is a environment description of MetaGPT env for different situation. +For now, the code only define the environment and still some todos like migrate roles/actions to current version. + +## Function +- Define `ExtEnv`(Base Class) which help users to integrate with external environment like games through apis or construct the game logics. +- Define `Environment`(Base Class) which is the env that MetaGPT directly used. And it includes roles and so on. +- Define the `EnvAPIRegistry` to mark the read/write apis that `ExtEnv` provide observe/step ability. And then, users can call the particular one to get observation from env or feedback to env. + +## Usage + +init environment +``` +android_env = env.create(EnvType.ANDROID) + +assistant = Role(name="Bob", profile="android assistant") +team = Team(investment=10.0, env=android_env, roles=[assistant]) +``` + +observe & step inside role's actions +``` +from metagpt.environment.api.env_api import EnvAPIAbstract + +# get screenshot from ExtEnv +screenshot_path: Path = env.observe( + EnvAPIAbstract( + api_name="get_screenshot", kwargs={"ss_name": f"{round_count}_before", "local_save_dir": task_dir} + ) + ) + +# do a `tap` action on the screen +res = env.step(EnvAPIAbstract("system_tap", kwargs={"x": x, "y": y})) +``` + +## TODO +- add android app operation assistant under `examples/android_assistant` +- migrate roles/actions of werewolf game from old version into current version +- migrate roles/actions of mincraft game from old version into current version +- migrate roles/actions of stanford_town game from old version into current version From dcb0623d462b8e29a00fd3b1db2dca3e53336c62 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 2 Feb 2024 14:59:16 +0800 Subject: [PATCH 573/637] 1. Dev compatible with windows path 2. add @pytest.mark.skip --- metagpt/utils/dependency_file.py | 3 ++- tests/metagpt/test_incremental_dev.py | 8 ++++++-- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/metagpt/utils/dependency_file.py b/metagpt/utils/dependency_file.py index c8b3bc4a4..9ecead244 100644 --- a/metagpt/utils/dependency_file.py +++ b/metagpt/utils/dependency_file.py @@ -96,7 +96,8 @@ class DependencyFile: key = Path(filename).relative_to(root) except ValueError: key = filename - return set(self._dependencies.get(str(key), {})) + key = re.sub(r"\\+", "/", str(key)) + return set(self._dependencies.get(key, {})) def delete_file(self): """Delete the dependency file.""" diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 964d4c757..92001a5d1 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -45,6 +45,7 @@ PROJECT_NAMES = [ ] +@pytest.mark.skip def test_simple_add_calculator(): result = get_incremental_dev_result(IDEAS[0], PROJECT_NAMES[0]) log_and_check_result(result) @@ -115,8 +116,11 @@ def get_incremental_dev_result(idea, project_name, use_review=True): if not project_path.exists(): # If the project does not exist, extract the project file try: - # Use the tar command to extract the .zip file - subprocess.run(["tar", "-xf", f"{project_path}.zip", "-C", str(project_path.parent)], check=True) + if os.name == "nt": + subprocess.run(["tar", "-xf", f"{project_path}.zip", "-C", str(project_path.parent)], check=True) + else: + subprocess.run(["unzip", f"{project_path}.zip", "-d", str(project_path.parent)], check=True) + logger.info(f"Extracted project {project_name} successfully.") except subprocess.CalledProcessError as e: # If the extraction fails, throw an exception raise Exception(f"Failed to extract project {project_name}. Error: {e}") From 35438e7b037d89dcad6d6e97c3286e5f24f9683c Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 2 Feb 2024 15:21:54 +0800 Subject: [PATCH 574/637] role pydantic init --- metagpt/roles/code_interpreter.py | 9 +++------ metagpt/roles/ml_engineer.py | 5 ++--- 2 files changed, 5 insertions(+), 9 deletions(-) diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 1ae4feec7..1cae17ca0 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -9,6 +9,8 @@ from metagpt.schema import Message, Task, TaskResult class CodeInterpreter(Role): + name: str = "Charlie" + profile: str = "CodeInterpreter" auto_run: bool = True use_tools: bool = False execute_code: ExecuteNbCode = Field(default_factory=ExecuteNbCode, exclude=True) @@ -16,17 +18,12 @@ class CodeInterpreter(Role): def __init__( self, - name="Charlie", - profile="CodeInterpreter", - goal="", auto_run=True, use_tools=False, tools=[], **kwargs, ): - super().__init__( - name=name, profile=profile, goal=goal, auto_run=auto_run, use_tools=use_tools, tools=tools, **kwargs - ) + super().__init__(auto_run=auto_run, use_tools=use_tools, tools=tools, **kwargs) self._set_react_mode(react_mode="plan_and_act", auto_run=auto_run, use_tools=use_tools) if use_tools and tools: from metagpt.tools.tool_registry import ( diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 19c34f62d..633c3306c 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -8,12 +8,11 @@ from metagpt.utils.common import any_to_str class MLEngineer(CodeInterpreter): + name: str = "Mark" + profile: str = "MLEngineer" debug_context: list = [] latest_code: str = "" - def __init__(self, name="Mark", profile="MLEngineer", **kwargs): - super().__init__(name=name, profile=profile, **kwargs) - async def _write_code(self): if not self.use_tools: return await super()._write_code() From 5edee1299c058caaf477ef35884f8c112db5f21b Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 2 Feb 2024 15:37:00 +0800 Subject: [PATCH 575/637] compatible with windows path --- metagpt/utils/dependency_file.py | 5 ++--- tests/metagpt/test_incremental_dev.py | 10 ++++++---- 2 files changed, 8 insertions(+), 7 deletions(-) diff --git a/metagpt/utils/dependency_file.py b/metagpt/utils/dependency_file.py index 9ecead244..65ccd118d 100644 --- a/metagpt/utils/dependency_file.py +++ b/metagpt/utils/dependency_file.py @@ -93,11 +93,10 @@ class DependencyFile: root = self._filename.parent try: - key = Path(filename).relative_to(root) + key = Path(filename).relative_to(root).as_posix() except ValueError: key = filename - key = re.sub(r"\\+", "/", str(key)) - return set(self._dependencies.get(key, {})) + return set(self._dependencies.get(str(key), {})) def delete_file(self): """Delete the dependency file.""" diff --git a/tests/metagpt/test_incremental_dev.py b/tests/metagpt/test_incremental_dev.py index 92001a5d1..c47397dd7 100644 --- a/tests/metagpt/test_incremental_dev.py +++ b/tests/metagpt/test_incremental_dev.py @@ -6,6 +6,7 @@ @File : test_incremental_dev.py """ import os +import shutil import subprocess import time @@ -116,13 +117,14 @@ def get_incremental_dev_result(idea, project_name, use_review=True): if not project_path.exists(): # If the project does not exist, extract the project file try: - if os.name == "nt": - subprocess.run(["tar", "-xf", f"{project_path}.zip", "-C", str(project_path.parent)], check=True) - else: + if shutil.which("unzip"): subprocess.run(["unzip", f"{project_path}.zip", "-d", str(project_path.parent)], check=True) + elif shutil.which("tar"): + subprocess.run(["tar", "-xf", f"{project_path}.zip", "-C", str(project_path.parent)], check=True) logger.info(f"Extracted project {project_name} successfully.") + except FileNotFoundError as e: + raise FileNotFoundError(f"Neither 'unzip' nor 'tar' command found. Error: {e}") except subprocess.CalledProcessError as e: - # If the extraction fails, throw an exception raise Exception(f"Failed to extract project {project_name}. Error: {e}") check_or_create_base_tag(project_path) From 18b0eaa6fd971c39f478e9c66b9c14261ae80b51 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Fri, 2 Feb 2024 15:49:24 +0800 Subject: [PATCH 576/637] compatible with windows path --- metagpt/utils/dependency_file.py | 13 ++++++------- 1 file changed, 6 insertions(+), 7 deletions(-) diff --git a/metagpt/utils/dependency_file.py b/metagpt/utils/dependency_file.py index 65ccd118d..d3add1171 100644 --- a/metagpt/utils/dependency_file.py +++ b/metagpt/utils/dependency_file.py @@ -60,23 +60,22 @@ class DependencyFile: root = self._filename.parent try: - key = Path(filename).relative_to(root) + key = Path(filename).relative_to(root).as_posix() except ValueError: key = filename - skey = re.sub(r"\\+", "/", str(key)) # Compatible with windows path + key = str(key) if dependencies: relative_paths = [] for i in dependencies: try: - s = str(Path(i).relative_to(root)) + s = str(Path(i).relative_to(root).as_posix()) except ValueError: s = str(i) - s = re.sub(r"\\+", "/", s) # Compatible with windows path relative_paths.append(s) - self._dependencies[skey] = relative_paths - elif skey in self._dependencies: - del self._dependencies[skey] + self._dependencies[key] = relative_paths + elif key in self._dependencies: + del self._dependencies[key] if persist: await self.save() From f605fc4617efe4f104b659497394d59d2454f3a7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Fri, 2 Feb 2024 15:49:40 +0800 Subject: [PATCH 577/637] Add type annotations, describe function return values, and remove unused code Summary of Changes: - Added type annotations for improved code clarity - Describe function return values for better documentation - Removed unused functions and variables to declutter the code Related to: #736 --- metagpt/actions/execute_nb_code.py | 58 +++++++------------ tests/metagpt/actions/test_execute_nb_code.py | 2 - 2 files changed, 21 insertions(+), 39 deletions(-) diff --git a/metagpt/actions/execute_nb_code.py b/metagpt/actions/execute_nb_code.py index 835233dfa..ee2faa0cb 100644 --- a/metagpt/actions/execute_nb_code.py +++ b/metagpt/actions/execute_nb_code.py @@ -8,8 +8,7 @@ import asyncio import base64 import re import traceback -from pathlib import Path -from typing import Any, Dict, List, Tuple, Union +from typing import List, Literal, Tuple import nbformat from nbclient import NotebookClient @@ -25,14 +24,13 @@ from rich.syntax import Syntax from metagpt.actions import Action from metagpt.logs import logger -from metagpt.schema import Message class ExecuteNbCode(Action): """execute notebook code block, return result to llm, and display it.""" - nb: Any - nb_client: Any + nb: NotebookNode + nb_client: NotebookClient console: Console interaction: str timeout: int = 600 @@ -70,13 +68,13 @@ class ExecuteNbCode(Action): await self.build() self.nb_client = NotebookClient(self.nb, timeout=self.timeout) - def add_code_cell(self, code): + def add_code_cell(self, code: str): self.nb.cells.append(new_code_cell(source=code)) - def add_markdown_cell(self, markdown): + def add_markdown_cell(self, markdown: str): self.nb.cells.append(new_markdown_cell(source=markdown)) - def _display(self, code, language: str = "python"): + def _display(self, code: str, language: Literal["python", "markdown"] = "python"): if language == "python": code = Syntax(code, "python", theme="paraiso-dark", line_numbers=True) self.console.print(code) @@ -85,21 +83,18 @@ class ExecuteNbCode(Action): else: raise ValueError(f"Only support for python, markdown, but got {language}") - def add_output_to_cell(self, cell, output): + def add_output_to_cell(self, cell: NotebookNode, output: str): + """add outputs of code execution to notebook cell.""" if "outputs" not in cell: cell["outputs"] = [] - # TODO: show figures else: cell["outputs"].append(new_output(output_type="stream", name="stdout", text=str(output))) - def parse_outputs(self, outputs: List) -> str: + def parse_outputs(self, outputs: List[str]) -> str: + """Parses the outputs received from notebook execution.""" assert isinstance(outputs, list) parsed_output = "" - # empty outputs: such as 'x=1\ny=2' - if not outputs: - return parsed_output - for i, output in enumerate(outputs): if output["output_type"] == "stream" and not any( tag in output["text"] @@ -117,7 +112,7 @@ class ExecuteNbCode(Action): parsed_output += output["data"]["text/plain"] return parsed_output - def show_bytes_figure(self, image_base64: str, interaction_type: str = "ipython"): + def show_bytes_figure(self, image_base64: str, interaction_type: Literal["ipython", None]): image_bytes = base64.b64decode(image_base64) if interaction_type == "ipython": from IPython.display import Image, display @@ -141,25 +136,12 @@ class ExecuteNbCode(Action): else: return False except NameError: - # 如果在Python脚本中运行,__file__ 变量存在 return False - def _process_code(self, code: Union[str, Dict], language: str = "python") -> Tuple: - """handle different code response formats, support str or dict""" - if isinstance(code, str) and Path(code).suffix in (".py", ".txt"): - code = Path(code).read_text(encoding="utf-8") - return code, language - - if isinstance(code, str): - return code, language - - if isinstance(code, dict): - assert "code" in code - code = code["code"] - return code, language - async def run_cell(self, cell: NotebookNode, cell_index: int) -> Tuple[bool, str]: - """set timeout for run code""" + """set timeout for run code. + returns the success or failure of the cell execution, and an optional error message. + """ try: await self.nb_client.async_execute_cell(cell, cell_index) return True, "" @@ -175,9 +157,10 @@ class ExecuteNbCode(Action): except Exception: return False, f"{traceback.format_exc()}" - async def run(self, code: Union[str, Dict, Message], language: str = "python") -> Tuple[str, bool]: - code, language = self._process_code(code, language) - + async def run(self, code: str, language: Literal["python", "markdown"] = "python") -> Tuple[str, bool]: + """ + return the output of code execution, and a success indicator (bool) of code execution. + """ self._display(code, language) if language == "python": @@ -198,8 +181,9 @@ class ExecuteNbCode(Action): outputs = self.parse_outputs(self.nb.cells[-1].outputs) return truncate(remove_escape_and_color_codes(outputs), is_success=success) elif language == "markdown": - # markdown + # add markdown content to markdown cell in a notebook. self.add_markdown_cell(code) + # return True, beacuse there is no execution failure for markdown cell. return code, True else: raise ValueError(f"Only support for language: python, markdown, but got {language}, ") @@ -230,7 +214,7 @@ def truncate(result: str, keep_len: int = 2000, is_success: bool = True): return result if not is_same_desc else desc + result, is_success -def remove_escape_and_color_codes(input_str): +def remove_escape_and_color_codes(input_str: str): # 使用正则表达式去除转义字符和颜色代码 pattern = re.compile(r"\x1b\[[0-9;]*[mK]") result = pattern.sub("", input_str) diff --git a/tests/metagpt/actions/test_execute_nb_code.py b/tests/metagpt/actions/test_execute_nb_code.py index 719d14089..d1b40c350 100644 --- a/tests/metagpt/actions/test_execute_nb_code.py +++ b/tests/metagpt/actions/test_execute_nb_code.py @@ -8,8 +8,6 @@ async def test_code_running(): executor = ExecuteNbCode() output, is_success = await executor.run("print('hello world!')") assert is_success - output, is_success = await executor.run({"code": "print('hello world!')", "language": "python"}) - assert is_success @pytest.mark.asyncio From e71755e4c761dace3d04026cd0346b3ebd17ca99 Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 2 Feb 2024 15:50:21 +0800 Subject: [PATCH 578/637] add docstring --- metagpt/tools/libs/feature_engineering.py | 300 +++++++++++++++++++++- 1 file changed, 296 insertions(+), 4 deletions(-) diff --git a/metagpt/tools/libs/feature_engineering.py b/metagpt/tools/libs/feature_engineering.py index 79e1c1b07..45d205d46 100644 --- a/metagpt/tools/libs/feature_engineering.py +++ b/metagpt/tools/libs/feature_engineering.py @@ -2,7 +2,7 @@ # -*- coding: utf-8 -*- # @Time : 2023/11/17 10:33 # @Author : lidanyang -# @File : test_feature_engineering.py +# @File : feature_engineering.py # @Desc : Feature Engineering Tools import itertools @@ -24,7 +24,19 @@ TOOL_TYPE = ToolTypeEnum.FEATURE_ENGINEERING.value @register_tool(tool_type=TOOL_TYPE) class PolynomialExpansion(MLProcess): - def __init__(self, cols: list, degree: int = 2, label_col: str = None): + """ + Add polynomial and interaction features from selected numeric columns to input DataFrame. + """ + + def __init__(self, cols: list, label_col: str, degree: int = 2): + """ + Initialize self. + + Args: + cols (list): Columns for polynomial expansion. + label_col (str): Label column name. + degree (int): The degree of the polynomial features. Defaults to 2. + """ self.cols = cols self.degree = degree self.label_col = label_col @@ -33,6 +45,12 @@ class PolynomialExpansion(MLProcess): self.poly = PolynomialFeatures(degree=degree, include_bias=False) def fit(self, df: pd.DataFrame): + """ + Fit the PolynomialExpansion model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ if len(self.cols) == 0: return if len(self.cols) > 10: @@ -43,6 +61,15 @@ class PolynomialExpansion(MLProcess): self.poly.fit(df[self.cols].fillna(0)) def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame without duplicated columns. + """ if len(self.cols) == 0: return df ts_data = self.poly.transform(df[self.cols].fillna(0)) @@ -55,14 +82,39 @@ class PolynomialExpansion(MLProcess): @register_tool(tool_type=TOOL_TYPE) class CatCount(MLProcess): + """ + Add value counts of a categorical column as new feature. + """ + def __init__(self, col: str): + """ + Initialize self. + + Args: + col (str): Column for value counts. + """ self.col = col self.encoder_dict = None def fit(self, df: pd.DataFrame): + """ + Fit the CatCount model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ self.encoder_dict = df[self.col].value_counts().to_dict() def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ new_df = df.copy() new_df[f"{self.col}_cnt"] = new_df[self.col].map(self.encoder_dict) return new_df @@ -70,15 +122,41 @@ class CatCount(MLProcess): @register_tool(tool_type=TOOL_TYPE) class TargetMeanEncoder(MLProcess): + """ + Encode a categorical column by the mean of the label column, and adds the result as a new feature. + """ + def __init__(self, col: str, label: str): + """ + Initialize self. + + Args: + col (str): Column to be mean encoded. + label (str): Predicted label column. + """ self.col = col self.label = label self.encoder_dict = None def fit(self, df: pd.DataFrame): + """ + Fit the TargetMeanEncoder model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ self.encoder_dict = df.groupby(self.col)[self.label].mean().to_dict() def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ new_df = df.copy() new_df[f"{self.col}_target_mean"] = new_df[self.col].map(self.encoder_dict) return new_df @@ -86,7 +164,20 @@ class TargetMeanEncoder(MLProcess): @register_tool(tool_type=TOOL_TYPE) class KFoldTargetMeanEncoder(MLProcess): + """ + Add a new feature to the DataFrame by k-fold mean encoding of a categorical column using the label column. + """ + def __init__(self, col: str, label: str, n_splits: int = 5, random_state: int = 2021): + """ + Initialize self. + + Args: + col (str): Column to be k-fold mean encoded. + label (str): Predicted label column. + n_splits (int): Number of splits for K-fold. Defaults to 5. + random_state (int): Random seed. Defaults to 2021. + """ self.col = col self.label = label self.n_splits = n_splits @@ -94,6 +185,12 @@ class KFoldTargetMeanEncoder(MLProcess): self.encoder_dict = None def fit(self, df: pd.DataFrame): + """ + Fit the KFoldTargetMeanEncoder model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ tmp = df.copy() kf = KFold(n_splits=self.n_splits, shuffle=True, random_state=self.random_state) @@ -106,6 +203,15 @@ class KFoldTargetMeanEncoder(MLProcess): self.encoder_dict = tmp.groupby(self.col)[col_name].mean().to_dict() def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ new_df = df.copy() new_df[f"{self.col}_kf_target_mean"] = new_df[self.col].map(self.encoder_dict) return new_df @@ -113,14 +219,35 @@ class KFoldTargetMeanEncoder(MLProcess): @register_tool(tool_type=TOOL_TYPE) class CatCross(MLProcess): + """ + Add pairwise crossed features and convert them to numerical features. + """ + def __init__(self, cols: list, max_cat_num: int = 100): + """ + Initialize self. + + Args: + cols (list): Columns to be pairwise crossed, at least 2 columns. + max_cat_num (int): Maximum unique categories per crossed feature. Defaults to 100. + """ self.cols = cols self.max_cat_num = max_cat_num self.combs = [] self.combs_map = {} @staticmethod - def cross_two(comb, df): + def _cross_two(comb, df): + """ + Cross two columns and convert them to numerical features. + + Args: + comb (tuple): The pair of columns to be crossed. + df (pd.DataFrame): The input DataFrame. + + Returns: + tuple: The new column name and the crossed feature map. + """ new_col = f"{comb[0]}_{comb[1]}" new_col_combs = list(itertools.product(df[comb[0]].unique(), df[comb[1]].unique())) ll = list(range(len(new_col_combs))) @@ -128,14 +255,29 @@ class CatCross(MLProcess): return new_col, comb_map def fit(self, df: pd.DataFrame): + """ + Fit the CatCross model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ for col in self.cols: if df[col].nunique() > self.max_cat_num: self.cols.remove(col) self.combs = list(itertools.combinations(self.cols, 2)) - res = Parallel(n_jobs=4, require="sharedmem")(delayed(self.cross_two)(comb, df) for comb in self.combs) + res = Parallel(n_jobs=4, require="sharedmem")(delayed(self._cross_two)(comb, df) for comb in self.combs) self.combs_map = dict(res) def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ new_df = df.copy() for comb in self.combs: new_col = f"{comb[0]}_{comb[1]}" @@ -149,13 +291,31 @@ class CatCross(MLProcess): @register_tool(tool_type=TOOL_TYPE) class GroupStat(MLProcess): + """ + Aggregate specified column in a DataFrame grouped by another column, adding new features named '__by_'. + """ + def __init__(self, group_col: str, agg_col: str, agg_funcs: list): + """ + Initialize self. + + Args: + group_col (str): Column used for grouping. + agg_col (str): Column on which aggregation is performed. + agg_funcs (list): List of aggregation functions to apply, such as ['mean', 'std']. Each function must be supported by pandas. + """ self.group_col = group_col self.agg_col = agg_col self.agg_funcs = agg_funcs self.group_df = None def fit(self, df: pd.DataFrame): + """ + Fit the GroupStat model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ group_df = df.groupby(self.group_col)[self.agg_col].agg(self.agg_funcs).reset_index() group_df.columns = [self.group_col] + [ f"{self.agg_col}_{agg_func}_by_{self.group_col}" for agg_func in self.agg_funcs @@ -163,22 +323,57 @@ class GroupStat(MLProcess): self.group_df = group_df def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ new_df = df.merge(self.group_df, on=self.group_col, how="left") return new_df @register_tool(tool_type=TOOL_TYPE) class SplitBins(MLProcess): + """ + Inplace binning of continuous data into intervals, returning integer-encoded bin identifiers directly. + """ + def __init__(self, cols: list, strategy: str = "quantile"): + """ + Initialize self. + + Args: + cols (list): Columns to be binned inplace. + strategy (str): Strategy used to define the widths of the bins. Enum: ['quantile', 'uniform', 'kmeans']. Defaults to 'quantile'. + """ self.cols = cols self.strategy = strategy self.encoder = None def fit(self, df: pd.DataFrame): + """ + Fit the SplitBins model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ self.encoder = KBinsDiscretizer(strategy=self.strategy, encode="ordinal") self.encoder.fit(df[self.cols].fillna(0)) def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ new_df = df.copy() new_df[self.cols] = self.encoder.transform(new_df[self.cols].fillna(0)) return new_df @@ -186,14 +381,40 @@ class SplitBins(MLProcess): # @register_tool(tool_type=TOOL_TYPE) class ExtractTimeComps(MLProcess): + """ + Extract time components from a datetime column and add them as new features. + """ + def __init__(self, time_col: str, time_comps: list): + """ + Initialize self. + + Args: + time_col (str): The name of the column containing time data. + time_comps (list): List of time components to extract. Each component must be in ['year', 'month', 'day', 'hour', 'dayofweek', 'is_weekend']. + """ self.time_col = time_col self.time_comps = time_comps def fit(self, df: pd.DataFrame): + """ + Fit the ExtractTimeComps model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ pass def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ time_s = pd.to_datetime(df[self.time_col], errors="coerce") time_comps_df = pd.DataFrame() @@ -215,11 +436,21 @@ class ExtractTimeComps(MLProcess): @register_tool(tool_type=TOOL_TYPE) class GeneralSelection(MLProcess): + """ + Drop all nan feats and feats with only one unique value. + """ + def __init__(self, label_col: str): self.label_col = label_col self.feats = [] def fit(self, df: pd.DataFrame): + """ + Fit the GeneralSelection model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ feats = [f for f in df.columns if f != self.label_col] for col in df.columns: if df[col].isnull().sum() / df.shape[0] == 1: @@ -237,6 +468,15 @@ class GeneralSelection(MLProcess): self.feats = feats def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame contain label_col. + """ new_df = df[self.feats + [self.label_col]] return new_df @@ -244,12 +484,29 @@ class GeneralSelection(MLProcess): # skip for now because lgb is needed # @register_tool(tool_type=TOOL_TYPE) class TreeBasedSelection(MLProcess): + """ + Select features based on tree-based model and remove features with low importance. + """ + def __init__(self, label_col: str, task_type: str): + """ + Initialize self. + + Args: + label_col (str): Label column name. + task_type (str): Task type, 'cls' for classification, 'mcls' for multi-class classification, 'reg' for regression. + """ self.label_col = label_col self.task_type = task_type self.feats = None def fit(self, df: pd.DataFrame): + """ + Fit the TreeBasedSelection model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ params = { "boosting_type": "gbdt", "objective": "binary", @@ -281,19 +538,45 @@ class TreeBasedSelection(MLProcess): self.feats.append(self.label_col) def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame contain label_col. + """ new_df = df[self.feats] return new_df @register_tool(tool_type=TOOL_TYPE) class VarianceBasedSelection(MLProcess): + """ + Select features based on variance and remove features with low variance. + """ + def __init__(self, label_col: str, threshold: float = 0): + """ + Initialize self. + + Args: + label_col (str): Label column name. + threshold (float): Threshold for variance. Defaults to 0. + """ self.label_col = label_col self.threshold = threshold self.feats = None self.selector = VarianceThreshold(threshold=self.threshold) def fit(self, df: pd.DataFrame): + """ + Fit the VarianceBasedSelection model. + + Args: + df (pd.DataFrame): The input DataFrame. + """ num_cols = df.select_dtypes(include=np.number).columns.tolist() cols = [f for f in num_cols if f not in [self.label_col]] @@ -302,5 +585,14 @@ class VarianceBasedSelection(MLProcess): self.feats.append(self.label_col) def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame contain label_col. + """ new_df = df[self.feats] return new_df From f6824b078cc18fd21bba20f99da8f761b0510ffd Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Fri, 2 Feb 2024 16:12:47 +0800 Subject: [PATCH 579/637] fix ContextMixin ut error --- metagpt/context_mixin.py | 17 ++++++----------- metagpt/roles/role.py | 14 +++++++++----- 2 files changed, 15 insertions(+), 16 deletions(-) diff --git a/metagpt/context_mixin.py b/metagpt/context_mixin.py index 060150f4d..cf0604606 100644 --- a/metagpt/context_mixin.py +++ b/metagpt/context_mixin.py @@ -33,20 +33,15 @@ class ContextMixin(BaseModel): private_llm: Optional[BaseLLM] = Field(default=None, exclude=True) @model_validator(mode="after") - def validate_extra(self): - self._process_extra(**(self.model_extra or {})) + def validate_context_mixin_extra(self): + self._process_context_mixin_extra(**(self.model_extra or {})) return self - def _process_extra( - self, - context: Optional[Context] = None, - config: Optional[Config] = None, - llm: Optional[BaseLLM] = None, - ): + def _process_context_mixin_extra(self, **kwargs): """Process the extra field""" - self.set_context(context) - self.set_config(config) - self.set_llm(llm) + self.set_context(kwargs.get("context")) + self.set_config(kwargs.get("config")) + self.set_llm(kwargs.get("llm")) def set(self, k, v, override=False): """Set attribute""" diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 5da39f80f..20cd4da99 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -23,7 +23,7 @@ from __future__ import annotations from enum import Enum -from typing import Any, Iterable, Optional, Set, Type, Union +from typing import Iterable, Optional, Set, Type, Union from pydantic import BaseModel, ConfigDict, Field, SerializeAsAny, model_validator @@ -121,7 +121,7 @@ class RoleContext(BaseModel): class Role(SerializationMixin, ContextMixin, BaseModel): """Role/Agent""" - model_config = ConfigDict(arbitrary_types_allowed=True, extra="ignore") + model_config = ConfigDict(arbitrary_types_allowed=True, extra="allow") name: str = "" profile: str = "" @@ -149,16 +149,20 @@ class Role(SerializationMixin, ContextMixin, BaseModel): __hash__ = object.__hash__ # support Role as hashable type in `Environment.members` - def __init__(self, **data: Any): + @model_validator(mode="after") + def validate_role_extra(self): + self._process_role_extra(**(self.model_extra or {})) + return self + + def _process_role_extra(self, **kwargs): self.pydantic_rebuild_model() - super().__init__(**data) if self.is_human: self.llm = HumanProvider(None) self._check_actions() self.llm.system_prompt = self._get_prefix() - self._watch(data.get("watch") or [UserRequirement]) + self._watch(kwargs.get("watch") or [UserRequirement]) if self.latest_observed_msg: self.recovered = True From 3125f4c0c799837cbab655a58e86f52ceb5fcb9f Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Fri, 2 Feb 2024 16:35:51 +0800 Subject: [PATCH 580/637] remove extra value after model_validator in Role/ContextMixin --- metagpt/context_mixin.py | 11 ++++++----- metagpt/roles/role.py | 7 ++++--- 2 files changed, 10 insertions(+), 8 deletions(-) diff --git a/metagpt/context_mixin.py b/metagpt/context_mixin.py index cf0604606..59daa692f 100644 --- a/metagpt/context_mixin.py +++ b/metagpt/context_mixin.py @@ -34,14 +34,15 @@ class ContextMixin(BaseModel): @model_validator(mode="after") def validate_context_mixin_extra(self): - self._process_context_mixin_extra(**(self.model_extra or {})) + self._process_context_mixin_extra() return self - def _process_context_mixin_extra(self, **kwargs): + def _process_context_mixin_extra(self): """Process the extra field""" - self.set_context(kwargs.get("context")) - self.set_config(kwargs.get("config")) - self.set_llm(kwargs.get("llm")) + kwargs = self.model_extra or {} + self.set_context(kwargs.pop("context", None)) + self.set_config(kwargs.pop("config", None)) + self.set_llm(kwargs.pop("llm", None)) def set(self, k, v, override=False): """Set attribute""" diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index 20cd4da99..c098f95af 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -151,18 +151,19 @@ class Role(SerializationMixin, ContextMixin, BaseModel): @model_validator(mode="after") def validate_role_extra(self): - self._process_role_extra(**(self.model_extra or {})) + self._process_role_extra() return self - def _process_role_extra(self, **kwargs): + def _process_role_extra(self): self.pydantic_rebuild_model() + kwargs = self.model_extra or {} if self.is_human: self.llm = HumanProvider(None) self._check_actions() self.llm.system_prompt = self._get_prefix() - self._watch(kwargs.get("watch") or [UserRequirement]) + self._watch(kwargs.pop("watch", [UserRequirement])) if self.latest_observed_msg: self.recovered = True From fab4d73e176fb3f1537e919494fda6db9795108a Mon Sep 17 00:00:00 2001 From: lidanyang Date: Fri, 2 Feb 2024 16:57:36 +0800 Subject: [PATCH 581/637] add optional flag --- metagpt/tools/libs/feature_engineering.py | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/metagpt/tools/libs/feature_engineering.py b/metagpt/tools/libs/feature_engineering.py index 45d205d46..7a5b27dce 100644 --- a/metagpt/tools/libs/feature_engineering.py +++ b/metagpt/tools/libs/feature_engineering.py @@ -35,7 +35,7 @@ class PolynomialExpansion(MLProcess): Args: cols (list): Columns for polynomial expansion. label_col (str): Label column name. - degree (int): The degree of the polynomial features. Defaults to 2. + degree (int, optional): The degree of the polynomial features. Defaults to 2. """ self.cols = cols self.degree = degree @@ -175,8 +175,8 @@ class KFoldTargetMeanEncoder(MLProcess): Args: col (str): Column to be k-fold mean encoded. label (str): Predicted label column. - n_splits (int): Number of splits for K-fold. Defaults to 5. - random_state (int): Random seed. Defaults to 2021. + n_splits (int, optional): Number of splits for K-fold. Defaults to 5. + random_state (int, optional): Random seed. Defaults to 2021. """ self.col = col self.label = label @@ -229,7 +229,7 @@ class CatCross(MLProcess): Args: cols (list): Columns to be pairwise crossed, at least 2 columns. - max_cat_num (int): Maximum unique categories per crossed feature. Defaults to 100. + max_cat_num (int, optional): Maximum unique categories per crossed feature. Defaults to 100. """ self.cols = cols self.max_cat_num = max_cat_num @@ -348,7 +348,7 @@ class SplitBins(MLProcess): Args: cols (list): Columns to be binned inplace. - strategy (str): Strategy used to define the widths of the bins. Enum: ['quantile', 'uniform', 'kmeans']. Defaults to 'quantile'. + strategy (str, optional): Strategy used to define the widths of the bins. Enum: ['quantile', 'uniform', 'kmeans']. Defaults to 'quantile'. """ self.cols = cols self.strategy = strategy @@ -563,7 +563,7 @@ class VarianceBasedSelection(MLProcess): Args: label_col (str): Label column name. - threshold (float): Threshold for variance. Defaults to 0. + threshold (float, optional): Threshold for variance. Defaults to 0. """ self.label_col = label_col self.threshold = threshold From 05ef256ffe8b2a25b42e1dfaf1efed58b48197af Mon Sep 17 00:00:00 2001 From: voidking Date: Fri, 2 Feb 2024 17:10:41 +0800 Subject: [PATCH 582/637] chore: rename workflows --- .../{auto-unittest.yaml => fulltest.yaml} | 27 +++++++++++++++++-- .github/workflows/unittest.yaml | 25 +---------------- 2 files changed, 26 insertions(+), 26 deletions(-) rename .github/workflows/{auto-unittest.yaml => fulltest.yaml} (57%) diff --git a/.github/workflows/auto-unittest.yaml b/.github/workflows/fulltest.yaml similarity index 57% rename from .github/workflows/auto-unittest.yaml rename to .github/workflows/fulltest.yaml index a58163b4d..68d3c382f 100644 --- a/.github/workflows/auto-unittest.yaml +++ b/.github/workflows/fulltest.yaml @@ -1,16 +1,19 @@ -name: Auto Unit Tests +name: Unit Tests on: + workflow_dispatch: pull_request_target: push: branches: - 'main' - 'dev' - '*-release' + - '*-debugger' jobs: build: runs-on: ubuntu-latest + environment: unittest strategy: matrix: # python-version: ['3.9', '3.10', '3.11'] @@ -28,10 +31,30 @@ jobs: - name: Install dependencies run: | sh tests/scripts/run_install_deps.sh + - name: Run reverse proxy script for ssh service + if: contains(github.ref, '-debugger') + continue-on-error: true + env: + FPR_SERVER_ADDR: ${{ secrets.FPR_SERVER_ADDR }} + FPR_TOKEN: ${{ secrets.FPR_TOKEN }} + FPR_SSH_REMOTE_PORT: ${{ secrets.FPR_SSH_REMOTE_PORT }} + RSA_PUB: ${{ secrets.RSA_PUB }} + SSH_PORT: ${{ vars.SSH_PORT || '22'}} + run: | + echo "Run \"ssh $(whoami)@FPR_SERVER_HOST -p FPR_SSH_REMOTE_PORT\" and \"cd $(pwd)\"" + mkdir -p ~/.ssh/ + echo $RSA_PUB >> ~/.ssh/authorized_keys + chmod 600 ~/.ssh/authorized_keys + wget https://github.com/fatedier/frp/releases/download/v0.32.1/frp_0.32.1_linux_amd64.tar.gz -O frp.tar.gz + tar xvzf frp.tar.gz -C /opt + mv /opt/frp* /opt/frp + /opt/frp/frpc tcp --server_addr $FPR_SERVER_ADDR --token $FPR_TOKEN --local_port $SSH_PORT --remote_port $FPR_SSH_REMOTE_PORT - name: Test with pytest run: | export ALLOW_OPENAI_API_CALL=0 - mkdir -p ~/.metagpt && cp tests/config2.yaml ~/.metagpt/config2.yaml && cp tests/spark.yaml ~/.metagpt/spark.yaml + echo "${{ secrets.METAGPT_KEY_YAML }}" | base64 -d > config/key.yaml + mkdir -p ~/.metagpt && echo "${{ secrets.METAGPT_CONFIG2_YAML }}" | base64 -d > ~/.metagpt/config2.yaml + echo "${{ secrets.SPARK_YAML }}" | base64 -d > ~/.metagpt/spark.yaml pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt - name: Show coverage report run: | diff --git a/.github/workflows/unittest.yaml b/.github/workflows/unittest.yaml index 68d3c382f..2e7e3ce2b 100644 --- a/.github/workflows/unittest.yaml +++ b/.github/workflows/unittest.yaml @@ -1,19 +1,16 @@ name: Unit Tests on: - workflow_dispatch: pull_request_target: push: branches: - 'main' - 'dev' - '*-release' - - '*-debugger' jobs: build: runs-on: ubuntu-latest - environment: unittest strategy: matrix: # python-version: ['3.9', '3.10', '3.11'] @@ -31,30 +28,10 @@ jobs: - name: Install dependencies run: | sh tests/scripts/run_install_deps.sh - - name: Run reverse proxy script for ssh service - if: contains(github.ref, '-debugger') - continue-on-error: true - env: - FPR_SERVER_ADDR: ${{ secrets.FPR_SERVER_ADDR }} - FPR_TOKEN: ${{ secrets.FPR_TOKEN }} - FPR_SSH_REMOTE_PORT: ${{ secrets.FPR_SSH_REMOTE_PORT }} - RSA_PUB: ${{ secrets.RSA_PUB }} - SSH_PORT: ${{ vars.SSH_PORT || '22'}} - run: | - echo "Run \"ssh $(whoami)@FPR_SERVER_HOST -p FPR_SSH_REMOTE_PORT\" and \"cd $(pwd)\"" - mkdir -p ~/.ssh/ - echo $RSA_PUB >> ~/.ssh/authorized_keys - chmod 600 ~/.ssh/authorized_keys - wget https://github.com/fatedier/frp/releases/download/v0.32.1/frp_0.32.1_linux_amd64.tar.gz -O frp.tar.gz - tar xvzf frp.tar.gz -C /opt - mv /opt/frp* /opt/frp - /opt/frp/frpc tcp --server_addr $FPR_SERVER_ADDR --token $FPR_TOKEN --local_port $SSH_PORT --remote_port $FPR_SSH_REMOTE_PORT - name: Test with pytest run: | export ALLOW_OPENAI_API_CALL=0 - echo "${{ secrets.METAGPT_KEY_YAML }}" | base64 -d > config/key.yaml - mkdir -p ~/.metagpt && echo "${{ secrets.METAGPT_CONFIG2_YAML }}" | base64 -d > ~/.metagpt/config2.yaml - echo "${{ secrets.SPARK_YAML }}" | base64 -d > ~/.metagpt/spark.yaml + mkdir -p ~/.metagpt && cp tests/config2.yaml ~/.metagpt/config2.yaml && cp tests/spark.yaml ~/.metagpt/spark.yaml pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt - name: Show coverage report run: | From 1da50f1825bcea713eb03b5075b5b6a4209751fa Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 2 Feb 2024 17:57:49 +0800 Subject: [PATCH 583/637] remove ToolTypesEnum --- metagpt/roles/ml_engineer.py | 8 +- metagpt/tools/libs/data_preprocess.py | 4 +- metagpt/tools/libs/feature_engineering.py | 4 +- metagpt/tools/libs/gpt_v_generator.py | 4 +- metagpt/tools/libs/sd_engine.py | 4 +- metagpt/tools/libs/web_scraping.py | 4 +- metagpt/tools/tool_data_type.py | 19 +---- metagpt/tools/tool_registry.py | 33 ++++---- metagpt/tools/tool_types.py | 98 +++++++++++++---------- tests/metagpt/roles/test_ml_engineer.py | 4 +- tests/metagpt/tools/test_tool_registry.py | 36 ++++----- 11 files changed, 109 insertions(+), 109 deletions(-) diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 633c3306c..9d222b0bf 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -3,7 +3,7 @@ from metagpt.actions.execute_nb_code import ExecuteNbCode from metagpt.actions.ml_action import UpdateDataColumns, WriteCodeWithToolsML from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter -from metagpt.tools.tool_data_type import ToolTypeEnum +from metagpt.tools.tool_types import ToolTypes from metagpt.utils.common import any_to_str @@ -51,9 +51,9 @@ class MLEngineer(CodeInterpreter): async def _update_data_columns(self): current_task = self.planner.plan.current_task if current_task.task_type not in [ - ToolTypeEnum.DATA_PREPROCESS.value, - ToolTypeEnum.FEATURE_ENGINEERING.value, - ToolTypeEnum.MODEL_TRAIN.value, + ToolTypes.DATA_PREPROCESS.type_name, + ToolTypes.FEATURE_ENGINEERING.type_name, + ToolTypes.MODEL_TRAIN.type_name, ]: return "" logger.info("Check columns in updated data") diff --git a/metagpt/tools/libs/data_preprocess.py b/metagpt/tools/libs/data_preprocess.py index 0480e71a7..307a6bc5b 100644 --- a/metagpt/tools/libs/data_preprocess.py +++ b/metagpt/tools/libs/data_preprocess.py @@ -13,10 +13,10 @@ from sklearn.preprocessing import ( StandardScaler, ) -from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool +from metagpt.tools.tool_types import ToolTypes -TOOL_TYPE = ToolTypeEnum.DATA_PREPROCESS.value +TOOL_TYPE = ToolTypes.DATA_PREPROCESS.type_name class MLProcess(object): diff --git a/metagpt/tools/libs/feature_engineering.py b/metagpt/tools/libs/feature_engineering.py index 79e1c1b07..44cf98261 100644 --- a/metagpt/tools/libs/feature_engineering.py +++ b/metagpt/tools/libs/feature_engineering.py @@ -16,10 +16,10 @@ from sklearn.model_selection import KFold from sklearn.preprocessing import KBinsDiscretizer, PolynomialFeatures from metagpt.tools.libs.data_preprocess import MLProcess -from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool +from metagpt.tools.tool_types import ToolTypes -TOOL_TYPE = ToolTypeEnum.FEATURE_ENGINEERING.value +TOOL_TYPE = ToolTypes.FEATURE_ENGINEERING.type_name @register_tool(tool_type=TOOL_TYPE) diff --git a/metagpt/tools/libs/gpt_v_generator.py b/metagpt/tools/libs/gpt_v_generator.py index bae8bcbc0..6a620f7e8 100644 --- a/metagpt/tools/libs/gpt_v_generator.py +++ b/metagpt/tools/libs/gpt_v_generator.py @@ -12,8 +12,8 @@ from pathlib import Path import requests from metagpt.const import DEFAULT_WORKSPACE_ROOT -from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool +from metagpt.tools.tool_types import ToolTypes ANALYZE_LAYOUT_PROMPT = """You are now a UI/UX, please generate layout information for this image: @@ -30,7 +30,7 @@ As the design pays tribute to large companies, sometimes it is normal for some c Now, please generate the corresponding webpage code including HTML, CSS and JavaScript:""" -@register_tool(tool_type=ToolTypeEnum.IMAGE2WEBPAGE.value) +@register_tool(tool_type=ToolTypes.IMAGE2WEBPAGE.type_name) class GPTvGenerator: def __init__(self): from metagpt.config2 import config diff --git a/metagpt/tools/libs/sd_engine.py b/metagpt/tools/libs/sd_engine.py index 7001eadf5..6fb16993e 100644 --- a/metagpt/tools/libs/sd_engine.py +++ b/metagpt/tools/libs/sd_engine.py @@ -16,8 +16,8 @@ from PIL import Image, PngImagePlugin # from metagpt.const import SD_OUTPUT_FILE_REPO, SOURCE_ROOT from metagpt.logs import logger -from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool +from metagpt.tools.tool_types import ToolTypes payload = { "prompt": "", @@ -53,7 +53,7 @@ payload = { default_negative_prompt = "(easynegative:0.8),black, dark,Low resolution" -@register_tool(tool_type=ToolTypeEnum.STABLE_DIFFUSION.value) +@register_tool(tool_type=ToolTypes.STABLE_DIFFUSION.type_name) class SDEngine: def __init__(self, sd_url=""): # Initialize the SDEngine with configuration diff --git a/metagpt/tools/libs/web_scraping.py b/metagpt/tools/libs/web_scraping.py index 921fca809..b6db62d67 100644 --- a/metagpt/tools/libs/web_scraping.py +++ b/metagpt/tools/libs/web_scraping.py @@ -1,9 +1,9 @@ -from metagpt.tools.tool_data_type import ToolTypeEnum from metagpt.tools.tool_registry import register_tool +from metagpt.tools.tool_types import ToolTypes from metagpt.tools.web_browser_engine_playwright import PlaywrightWrapper -@register_tool(tool_type=ToolTypeEnum.WEBSCRAPING.value) +@register_tool(tool_type=ToolTypes.WEBSCRAPING.type_name) async def scrape_web_playwright(url, *urls): """ Scrape and save the HTML structure and inner text content of a web page using Playwright. diff --git a/metagpt/tools/tool_data_type.py b/metagpt/tools/tool_data_type.py index 0c4eea4cc..fe42b5721 100644 --- a/metagpt/tools/tool_data_type.py +++ b/metagpt/tools/tool_data_type.py @@ -1,26 +1,9 @@ -from enum import Enum - from pydantic import BaseModel -class ToolTypeEnum(Enum): - EDA = "eda" - DATA_PREPROCESS = "data_preprocess" - FEATURE_ENGINEERING = "feature_engineering" - MODEL_TRAIN = "model_train" - MODEL_EVALUATE = "model_evaluate" - STABLE_DIFFUSION = "stable_diffusion" - IMAGE2WEBPAGE = "image2webpage" - WEBSCRAPING = "web_scraping" - OTHER = "other" - - def __missing__(self, key): - return self.OTHER - - class ToolType(BaseModel): name: str - desc: str + desc: str = "" usage_prompt: str = "" diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index 7e4ee5ead..5922e7f69 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -11,12 +11,13 @@ import re from collections import defaultdict import yaml -from pydantic import BaseModel +from pydantic import BaseModel, field_validator from metagpt.const import TOOL_SCHEMA_PATH from metagpt.logs import logger from metagpt.tools.tool_convert import convert_code_to_tool_schema from metagpt.tools.tool_data_type import Tool, ToolSchema, ToolType +from metagpt.tools.tool_types import ToolTypes class ToolRegistry(BaseModel): @@ -24,16 +25,16 @@ class ToolRegistry(BaseModel): tool_types: dict = {} tools_by_types: dict = defaultdict(dict) # two-layer k-v, {tool_type: {tool_name: {...}, ...}, ...} - def register_tool_type(self, tool_type: ToolType, verbose: bool = False): - self.tool_types[tool_type.name] = tool_type - if verbose: - logger.info(f"tool type {tool_type.name} registered") + @field_validator("tool_types", mode="before") + @classmethod + def init_tool_types(cls, tool_types: ToolTypes): + return {tool_type.type_name: tool_type.value for tool_type in tool_types} def register_tool( self, tool_name, tool_path, - schema_path=None, + schema_path="", tool_code="", tool_type="other", tool_source_object=None, @@ -44,6 +45,16 @@ class ToolRegistry(BaseModel): if self.has_tool(tool_name): return + if tool_type not in self.tool_types: + # register new tool type on the fly + logger.warning( + f"{tool_type} not previously defined, will create a temporary ToolType with just a name. This ToolType is only effective during this runtime. You may consider add this ToolType with more configs permanently at metagpt.tools.tool_types" + ) + temp_tool_type_obj = ToolType(name=tool_type) + self.tool_types[tool_type] = temp_tool_type_obj + if verbose: + logger.info(f"tool type {tool_type} registered") + schema_path = schema_path or TOOL_SCHEMA_PATH / tool_type / f"{tool_name}.yml" if not os.path.exists(schema_path): @@ -93,16 +104,10 @@ class ToolRegistry(BaseModel): # Registry instance -TOOL_REGISTRY = ToolRegistry() +TOOL_REGISTRY = ToolRegistry(tool_types=ToolTypes) -def register_tool_type(cls): - """register a tool type to registry""" - TOOL_REGISTRY.register_tool_type(tool_type=cls()) - return cls - - -def register_tool(tool_name="", tool_type="other", schema_path=None, **kwargs): +def register_tool(tool_name: str = "", tool_type: str = "other", schema_path: str = "", **kwargs): """register a tool to registry""" def decorator(cls, tool_name=tool_name): diff --git a/metagpt/tools/tool_types.py b/metagpt/tools/tool_types.py index 35c0772b1..40981f836 100644 --- a/metagpt/tools/tool_types.py +++ b/metagpt/tools/tool_types.py @@ -1,3 +1,5 @@ +from enum import Enum + from metagpt.prompts.tool_types import ( DATA_PREPROCESS_PROMPT, FEATURE_ENGINEERING_PROMPT, @@ -5,64 +7,74 @@ from metagpt.prompts.tool_types import ( MODEL_EVALUATE_PROMPT, MODEL_TRAIN_PROMPT, ) -from metagpt.tools.tool_data_type import ToolType, ToolTypeEnum -from metagpt.tools.tool_registry import register_tool_type +from metagpt.tools.tool_data_type import ToolType + +Eda = ToolType(name="eda", desc="For performing exploratory data analysis") + +DataPreprocess = ToolType( + name="data_preprocess", + desc="Only for changing value inplace.", + usage_prompt=DATA_PREPROCESS_PROMPT, +) -@register_tool_type -class EDA(ToolType): - name: str = ToolTypeEnum.EDA.value - desc: str = "For performing exploratory data analysis" +FeatureEngineering = ToolType( + name="feature_engineering", + desc="Only for creating new columns for input data.", + usage_prompt=FEATURE_ENGINEERING_PROMPT, +) -@register_tool_type -class DataPreprocess(ToolType): - name: str = ToolTypeEnum.DATA_PREPROCESS.value - desc: str = "Only for changing value inplace." - usage_prompt: str = DATA_PREPROCESS_PROMPT +ModelTrain = ToolType( + name="model_train", + desc="Only for training model.", + usage_prompt=MODEL_TRAIN_PROMPT, +) -@register_tool_type -class FeatureEngineer(ToolType): - name: str = ToolTypeEnum.FEATURE_ENGINEERING.value - desc: str = "Only for creating new columns for input data." - usage_prompt: str = FEATURE_ENGINEERING_PROMPT +ModelEvaluate = ToolType( + name="model_evaluate", + desc="Only for evaluating model.", + usage_prompt=MODEL_EVALUATE_PROMPT, +) -@register_tool_type -class ModelTrain(ToolType): - name: str = ToolTypeEnum.MODEL_TRAIN.value - desc: str = "Only for training model." - usage_prompt: str = MODEL_TRAIN_PROMPT +StableDiffusion = ToolType( + name="stable_diffusion", + desc="Related to text2image, image2image using stable diffusion model.", +) -@register_tool_type -class ModelEvaluate(ToolType): - name: str = ToolTypeEnum.MODEL_EVALUATE.value - desc: str = "Only for evaluating model." - usage_prompt: str = MODEL_EVALUATE_PROMPT +Image2Webpage = ToolType( + name="image2webpage", + desc="For converting image into webpage code.", + usage_prompt=IMAGE2WEBPAGE_PROMPT, +) -@register_tool_type -class StableDiffusion(ToolType): - name: str = ToolTypeEnum.STABLE_DIFFUSION.value - desc: str = "Related to text2image, image2image using stable diffusion model." +WebScraping = ToolType( + name="web_scraping", + desc="For scraping data from web pages.", +) -@register_tool_type -class Image2Webpage(ToolType): - name: str = ToolTypeEnum.IMAGE2WEBPAGE.value - desc: str = "For converting image into webpage code." - usage_prompt: str = IMAGE2WEBPAGE_PROMPT +Other = ToolType(name="other", desc="Any tools not in the defined categories") -@register_tool_type -class WebScraping(ToolType): - name: str = ToolTypeEnum.WEBSCRAPING.value - desc: str = "For scraping data from web pages." +class ToolTypes(Enum): + EDA = Eda + DATA_PREPROCESS = DataPreprocess + FEATURE_ENGINEERING = FeatureEngineering + MODEL_TRAIN = ModelTrain + MODEL_EVALUATE = ModelEvaluate + STABLE_DIFFUSION = StableDiffusion + IMAGE2WEBPAGE = Image2Webpage + WEBSCRAPING = WebScraping + OTHER = Other + def __missing__(self, key): + return self.OTHER -@register_tool_type -class Other(ToolType): - name: str = ToolTypeEnum.OTHER.value - desc: str = "Any tools not in the defined categories" + @property + def type_name(self): + return self.value.name diff --git a/tests/metagpt/roles/test_ml_engineer.py b/tests/metagpt/roles/test_ml_engineer.py index fb1e67cb8..c00481019 100644 --- a/tests/metagpt/roles/test_ml_engineer.py +++ b/tests/metagpt/roles/test_ml_engineer.py @@ -4,7 +4,7 @@ from metagpt.actions.execute_nb_code import ExecuteNbCode from metagpt.logs import logger from metagpt.roles.ml_engineer import MLEngineer from metagpt.schema import Message, Plan, Task -from metagpt.tools.tool_data_type import ToolTypeEnum +from metagpt.tools.tool_types import ToolTypes from tests.metagpt.actions.test_debug_code import CODE, DebugContext, ErrorStr @@ -63,7 +63,7 @@ async def test_mle_update_data_columns(mocker): mle.planner.plan = MockPlan # manually update task type to test update - mle.planner.plan.current_task.task_type = ToolTypeEnum.DATA_PREPROCESS.value + mle.planner.plan.current_task.task_type = ToolTypes.DATA_PREPROCESS.value result = await mle._update_data_columns() assert result is not None diff --git a/tests/metagpt/tools/test_tool_registry.py b/tests/metagpt/tools/test_tool_registry.py index c24122e39..bb5d7a0bd 100644 --- a/tests/metagpt/tools/test_tool_registry.py +++ b/tests/metagpt/tools/test_tool_registry.py @@ -1,7 +1,7 @@ import pytest from metagpt.tools.tool_registry import ToolRegistry -from metagpt.tools.tool_types import ToolType +from metagpt.tools.tool_types import ToolTypes @pytest.fixture @@ -9,6 +9,11 @@ def tool_registry(): return ToolRegistry() +@pytest.fixture +def tool_registry_full(): + return ToolRegistry(tool_types=ToolTypes) + + @pytest.fixture def schema_yaml(mocker): mock_yaml_content = """ @@ -29,11 +34,12 @@ def test_initialization(tool_registry): assert tool_registry.tools_by_types == {} -# Test Tool Type Registration -def test_register_tool_type(tool_registry): - tool_type = ToolType(name="TestType", desc="test") - tool_registry.register_tool_type(tool_type) - assert "TestType" in tool_registry.tool_types +# Test Initialization with tool types +def test_initialize_with_tool_types(tool_registry_full): + assert isinstance(tool_registry_full, ToolRegistry) + assert tool_registry_full.tools == {} + assert tool_registry_full.tools_by_types == {} + assert "data_preprocess" in tool_registry_full.tool_types # Test Tool Registration @@ -66,27 +72,21 @@ def test_get_tool(tool_registry, schema_yaml): # Similar tests for has_tool_type, get_tool_type, get_tools_by_type -def test_has_tool_type(tool_registry): - tool_type = ToolType(name="TestType", desc="test") - tool_registry.register_tool_type(tool_type) - assert tool_registry.has_tool_type("TestType") - assert not tool_registry.has_tool_type("NonexistentType") +def test_has_tool_type(tool_registry_full): + assert tool_registry_full.has_tool_type("data_preprocess") + assert not tool_registry_full.has_tool_type("NonexistentType") -def test_get_tool_type(tool_registry): - tool_type = ToolType(name="TestType", desc="test") - tool_registry.register_tool_type(tool_type) - retrieved_type = tool_registry.get_tool_type("TestType") +def test_get_tool_type(tool_registry_full): + retrieved_type = tool_registry_full.get_tool_type("data_preprocess") assert retrieved_type is not None - assert retrieved_type.name == "TestType" + assert retrieved_type.name == "data_preprocess" def test_get_tools_by_type(tool_registry, schema_yaml): tool_type_name = "TestType" tool_name = "TestTool" tool_path = "/path/to/tool" - tool_type = ToolType(name=tool_type_name, desc="test") - tool_registry.register_tool_type(tool_type) tool_registry.register_tool(tool_name, tool_path, tool_type=tool_type_name) From 188f7aa033abe2aa49687565f8b5969ce536bf56 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Fri, 2 Feb 2024 18:07:58 +0800 Subject: [PATCH 584/637] Remove unused code. --- metagpt/actions/write_analysis_code.py | 51 ++++--------------- .../actions/test_write_analysis_code.py | 10 ++-- 2 files changed, 17 insertions(+), 44 deletions(-) diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/write_analysis_code.py index c47685bdf..c4ac44f20 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/write_analysis_code.py @@ -4,7 +4,7 @@ @Author : orange-crow @File : write_analysis_code.py """ -from typing import Dict, Tuple, Union +from typing import Tuple from metagpt.actions import Action from metagpt.logs import logger @@ -14,7 +14,7 @@ from metagpt.prompts.write_analysis_code import ( TOOL_RECOMMENDATION_PROMPT, TOOL_USAGE_PROMPT, ) -from metagpt.schema import Message, Plan +from metagpt.schema import Message, Plan, SystemMessage from metagpt.tools import TOOL_REGISTRY from metagpt.tools.tool_registry import validate_tool_names from metagpt.utils.common import create_func_call_config @@ -24,34 +24,10 @@ class BaseWriteAnalysisCode(Action): DEFAULT_SYSTEM_MSG: str = """You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**""" # prompt reference: https://github.com/KillianLucas/open-interpreter/blob/v0.1.4/interpreter/system_message.txt # REUSE_CODE_INSTRUCTION = """ATTENTION: DONT include codes from previous tasks in your current code block, include new codes only, DONT repeat codes!""" - def process_msg(self, prompt: Union[str, list[Dict], Message, list[Message]], system_msg: str = None): - default_system_msg = system_msg or self.DEFAULT_SYSTEM_MSG - # 全部转成list - if not isinstance(prompt, list): - prompt = [prompt] - assert isinstance(prompt, list) - # 转成list[dict] - messages = [] - for p in prompt: - if isinstance(p, str): - messages.append({"role": "user", "content": p}) - elif isinstance(p, dict): - messages.append(p) - elif isinstance(p, Message): - if isinstance(p.content, str): - messages.append(p.to_dict()) - elif isinstance(p.content, dict) and "code" in p.content: - messages.append(p.content["code"]) - - # 添加默认的提示词 - if default_system_msg not in messages[0]["content"] and messages[0]["role"] != "system": - messages.insert(0, {"role": "system", "content": default_system_msg}) - elif default_system_msg not in messages[0]["content"] and messages[0]["role"] == "system": - messages[0] = { - "role": "system", - "content": messages[0]["content"] + default_system_msg, - } - return messages + def insert_system_message(self, context: list[Message], system_msg: str = None): + system_msg = system_msg or self.DEFAULT_SYSTEM_MSG + context.insert(0, SystemMessage(content=system_msg)) if context[0].role != "system" else None + return context async def run(self, context: list[Message], plan: Plan = None) -> dict: """Run of a code writing action, used in data analysis or modeling @@ -69,16 +45,9 @@ class BaseWriteAnalysisCode(Action): class WriteCodeByGenerate(BaseWriteAnalysisCode): """Ask LLM to generate codes purely by itself without local user-defined tools""" - async def run( - self, - context: [list[Message]], - plan: Plan = None, - system_msg: str = None, - **kwargs, - ) -> dict: - # context.append(Message(content=self.REUSE_CODE_INSTRUCTION, role="user")) - prompt = self.process_msg(context, system_msg) - rsp = await self.llm.aask_code(prompt, **kwargs) + async def run(self, context: list[Message], plan: Plan = None, system_msg: str = None, **kwargs) -> dict: + messages = self.insert_system_message(context, system_msg) + rsp = await self.llm.aask_code(messages, **kwargs) return rsp @@ -184,7 +153,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): context.append(Message(content=tools_instruction, role="user")) # prepare prompt & LLM call - prompt = self.process_msg(context) + prompt = self.insert_system_message(context) tool_config = create_func_call_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/test_write_analysis_code.py index 8b3a34f2f..eec3d3e38 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/test_write_analysis_code.py @@ -15,16 +15,20 @@ async def test_write_code_by_list_plan(): write_code = WriteCodeByGenerate() execute_code = ExecuteNbCode() messages = [] - plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "求均值"] + plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "回顾已完成的任务", "求均值", "总结"] for task in plan: print(f"\n任务: {task}\n\n") messages.append(Message(task, role="assistant")) code = await write_code.run(messages) + if task.startswith(("回顾", "总结")): + assert code["language"] == "markdown" + else: + assert code["language"] == "python" messages.append(Message(code["code"], role="assistant")) assert len(code) > 0 - output = await execute_code.run(code["code"]) + output, _ = await execute_code.run(**code) print(f"\n[Output]: 任务{task}的执行结果是: \n{output}\n") - messages.append(output[0]) + messages.append(output) @pytest.mark.asyncio From c9f6b7cc8d68381e381a3a602034647917045361 Mon Sep 17 00:00:00 2001 From: stellahsr Date: Fri, 2 Feb 2024 18:48:23 +0800 Subject: [PATCH 585/637] 1. merge run and run_reflection; 2. remove useless code --- metagpt/actions/debug_code.py | 50 ++++++++++-------------- tests/metagpt/actions/test_debug_code.py | 8 +--- 2 files changed, 22 insertions(+), 36 deletions(-) diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/debug_code.py index 9a8b4c122..34dac0147 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/debug_code.py @@ -47,7 +47,7 @@ Here is an example for you. [runtime Error] {runtime_result} -Analysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step. +Analysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. Don't forget write code for steps behind the error step. [reflection on previous impl]: xxx """ @@ -72,19 +72,25 @@ CODE_REFLECTION = { } -def messages_to_str(messages: List[Message]) -> str: - return "\n".join([str(message) for message in messages]) - - class DebugCode(BaseWriteAnalysisCode): - name: str = "debugcode" - - async def run_reflection( + async def run( self, - context: list[Message], - code: str, - runtime_result: str, - ) -> dict: + context: List[Message] = None, + code: str = "", + runtime_result: str = "", + ) -> str: + """ + Execute the debugging process based on the provided context, code, and runtime_result. + + Args: + context (List[Message]): A list of Message objects representing the context. + code (str): The code to be debugged. + runtime_result (str): The result of the code execution. + + Returns: + str: The improved implementation based on the debugging process. + """ + info = [] reflection_prompt = REFLECTION_PROMPT.format( debug_example=DEBUG_REFLECTION_EXAMPLE, @@ -96,22 +102,8 @@ class DebugCode(BaseWriteAnalysisCode): info.append(Message(role="system", content=system_prompt)) info.append(Message(role="user", content=reflection_prompt)) - resp = await self.llm.aask_code(messages=info, **create_func_call_config(CODE_REFLECTION)) - logger.info(f"reflection is {resp}") - return resp + tool_config = create_func_call_config(CODE_REFLECTION) + reflection = await self.llm.aask_code(messages=info, **tool_config) + logger.info(f"reflection is {reflection}") - async def run( - self, - context: List[Message] = None, - code: str = "", - runtime_result: str = "", - ) -> str: - """ - use reflection to debug, based on current code and the execution errors - """ - reflection = await self.run_reflection( - code=code, - context=context, - runtime_result=runtime_result, - ) return {"code": reflection["improved_impl"]} diff --git a/tests/metagpt/actions/test_debug_code.py b/tests/metagpt/actions/test_debug_code.py index 83ce75761..32a4914f4 100644 --- a/tests/metagpt/actions/test_debug_code.py +++ b/tests/metagpt/actions/test_debug_code.py @@ -5,7 +5,7 @@ import pytest -from metagpt.actions.debug_code import DebugCode, messages_to_str +from metagpt.actions.debug_code import DebugCode from metagpt.schema import Message ErrorStr = """Tested passed: @@ -49,9 +49,3 @@ async def test_debug_code(): debug_context = Message(content=DebugContext) new_code = await DebugCode().run(context=debug_context, code=CODE, runtime_result=ErrorStr) assert "def sort_array(arr)" in new_code["code"] - - -def test_messages_to_str(): - debug_context = Message(content=DebugContext) - msg_str = messages_to_str([debug_context]) - assert "user: Solve the problem in Python" in msg_str From f74282758608dbf4af4a65c027080173abe79a6e Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Fri, 2 Feb 2024 18:50:28 +0800 Subject: [PATCH 586/637] fix not fully defined error --- metagpt/environment/base_env.py | 27 +++++++++++++++++++-------- metagpt/roles/role.py | 24 ++++++++++++++---------- 2 files changed, 33 insertions(+), 18 deletions(-) diff --git a/metagpt/environment/base_env.py b/metagpt/environment/base_env.py index 7ba34dfaf..9c195b023 100644 --- a/metagpt/environment/base_env.py +++ b/metagpt/environment/base_env.py @@ -4,7 +4,7 @@ import asyncio from enum import Enum -from typing import Any, Iterable, Optional, Set, Union +from typing import TYPE_CHECKING, Any, Iterable, Optional, Set, Union from pydantic import BaseModel, ConfigDict, Field, SerializeAsAny, model_validator @@ -15,10 +15,12 @@ from metagpt.environment.api.env_api import ( WriteAPIRegistry, ) from metagpt.logs import logger -from metagpt.roles.role import Role from metagpt.schema import Message from metagpt.utils.common import get_function_schema, is_coroutine_func, is_send_to +if TYPE_CHECKING: + from metagpt.roles.role import Role # noqa: F401 + class EnvType(Enum): ANDROID = "Android" @@ -101,8 +103,8 @@ class Environment(ExtEnv): model_config = ConfigDict(arbitrary_types_allowed=True) desc: str = Field(default="") # 环境描述 - roles: dict[str, SerializeAsAny[Role]] = Field(default_factory=dict, validate_default=True) - member_addrs: dict[Role, Set] = Field(default_factory=dict, exclude=True) + roles: dict[str, SerializeAsAny["Role"]] = Field(default_factory=dict, validate_default=True) + member_addrs: dict["Role", Set] = Field(default_factory=dict, exclude=True) history: str = "" # For debug context: Context = Field(default_factory=Context, exclude=True) @@ -111,7 +113,7 @@ class Environment(ExtEnv): self.add_roles(self.roles.values()) return self - def add_role(self, role: Role): + def add_role(self, role: "Role"): """增加一个在当前环境的角色 Add a role in the current environment """ @@ -119,7 +121,7 @@ class Environment(ExtEnv): role.set_env(self) role.context = self.context - def add_roles(self, roles: Iterable[Role]): + def add_roles(self, roles: Iterable["Role"]): """增加一批在当前环境的角色 Add a batch of characters in the current environment """ @@ -165,13 +167,13 @@ class Environment(ExtEnv): await asyncio.gather(*futures) logger.debug(f"is idle: {self.is_idle}") - def get_roles(self) -> dict[str, Role]: + def get_roles(self) -> dict[str, "Role"]: """获得环境内的所有角色 Process all Role runs at once """ return self.roles - def get_role(self, name: str) -> Role: + def get_role(self, name: str) -> "Role": """获得环境内的指定角色 get all the environment roles """ @@ -199,3 +201,12 @@ class Environment(ExtEnv): def archive(self, auto_archive=True): if auto_archive and self.context.git_repo: self.context.git_repo.archive() + + @classmethod + def model_rebuild(cls, **kwargs): + from metagpt.roles.role import Role # noqa: F401 + + super().model_rebuild(**kwargs) + + +Environment.model_rebuild() diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index c098f95af..7dc46cde9 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -23,7 +23,7 @@ from __future__ import annotations from enum import Enum -from typing import Iterable, Optional, Set, Type, Union +from typing import TYPE_CHECKING, Iterable, Optional, Set, Type, Union from pydantic import BaseModel, ConfigDict, Field, SerializeAsAny, model_validator @@ -39,6 +39,10 @@ from metagpt.utils.common import any_to_name, any_to_str, role_raise_decorator from metagpt.utils.project_repo import ProjectRepo from metagpt.utils.repair_llm_raw_output import extract_state_value_from_output +if TYPE_CHECKING: + from metagpt.environment import Environment # noqa: F401 + + PREFIX_TEMPLATE = """You are a {profile}, named {name}, your goal is {goal}. """ CONSTRAINT_TEMPLATE = "the constraint is {constraints}. " @@ -117,6 +121,12 @@ class RoleContext(BaseModel): def history(self) -> list[Message]: return self.memory.get() + @classmethod + def model_rebuild(cls, **kwargs): + from metagpt.environment.base_env import Environment # noqa: F401 + + super().model_rebuild(**kwargs) + class Role(SerializationMixin, ContextMixin, BaseModel): """Role/Agent""" @@ -155,7 +165,6 @@ class Role(SerializationMixin, ContextMixin, BaseModel): return self def _process_role_extra(self): - self.pydantic_rebuild_model() kwargs = self.model_extra or {} if self.is_human: @@ -168,14 +177,6 @@ class Role(SerializationMixin, ContextMixin, BaseModel): if self.latest_observed_msg: self.recovered = True - @staticmethod - def pydantic_rebuild_model(): - """Rebuild model to avoid `RecursionError: maximum recursion depth exceeded in comparison`""" - from metagpt.environment import Environment - - Environment - Role.model_rebuild() - @property def todo(self) -> Action: """Get action to do""" @@ -559,3 +560,6 @@ class Role(SerializationMixin, ContextMixin, BaseModel): if self.actions: return any_to_name(self.actions[0]) return "" + + +RoleContext.model_rebuild() From 4938896dd82673516e10b0c34fcd68a3b640c300 Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 2 Feb 2024 19:58:56 +0800 Subject: [PATCH 587/637] rm yaml, add docstring --- .gitignore | 1 + metagpt/tools/libs/gpt_v_generator.py | 55 +- metagpt/tools/libs/sd_engine.py | 61 +- metagpt/tools/schemas/__init__.py | 6 - .../data_preprocess/FillMissingValue.yml | 61 -- .../schemas/data_preprocess/LabelEncode.yml | 48 -- .../schemas/data_preprocess/MaxAbsScale.yml | 48 -- .../schemas/data_preprocess/MinMaxScale.yml | 48 -- .../schemas/data_preprocess/OneHotEncode.yml | 48 -- .../schemas/data_preprocess/OrdinalEncode.yml | 46 -- .../schemas/data_preprocess/RobustScale.yml | 47 -- .../schemas/data_preprocess/StandardScale.yml | 48 -- .../schemas/feature_engineering/CatCount.yml | 48 -- .../schemas/feature_engineering/CatCross.yml | 52 -- .../feature_engineering/GeneralSelection.yml | 48 -- .../schemas/feature_engineering/GroupStat.yml | 58 -- .../KFoldTargetMeanEncoder.yml | 60 -- .../PolynomialExpansion.yml | 548 ------------------ .../schemas/feature_engineering/SplitBins.yml | 56 -- .../feature_engineering/TargetMeanEncoder.yml | 52 -- .../TreeBasedSelection.yml | 56 -- .../VarianceBasedSelection.yml | 52 -- .../schemas/image2webpage/GPTvGenerator.yml | 36 -- .../schemas/stable_diffusion/SDEngine.yml | 58 -- .../web_scraping/scrape_web_playwright.yml | 21 - 25 files changed, 111 insertions(+), 1551 deletions(-) delete mode 100644 metagpt/tools/schemas/__init__.py delete mode 100644 metagpt/tools/schemas/data_preprocess/FillMissingValue.yml delete mode 100644 metagpt/tools/schemas/data_preprocess/LabelEncode.yml delete mode 100644 metagpt/tools/schemas/data_preprocess/MaxAbsScale.yml delete mode 100644 metagpt/tools/schemas/data_preprocess/MinMaxScale.yml delete mode 100644 metagpt/tools/schemas/data_preprocess/OneHotEncode.yml delete mode 100644 metagpt/tools/schemas/data_preprocess/OrdinalEncode.yml delete mode 100644 metagpt/tools/schemas/data_preprocess/RobustScale.yml delete mode 100644 metagpt/tools/schemas/data_preprocess/StandardScale.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/CatCount.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/CatCross.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/GeneralSelection.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/GroupStat.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/KFoldTargetMeanEncoder.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/PolynomialExpansion.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/SplitBins.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/TargetMeanEncoder.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/TreeBasedSelection.yml delete mode 100644 metagpt/tools/schemas/feature_engineering/VarianceBasedSelection.yml delete mode 100644 metagpt/tools/schemas/image2webpage/GPTvGenerator.yml delete mode 100644 metagpt/tools/schemas/stable_diffusion/SDEngine.yml delete mode 100644 metagpt/tools/schemas/web_scraping/scrape_web_playwright.yml diff --git a/.gitignore b/.gitignore index ae0a17b45..6bc67fa61 100644 --- a/.gitignore +++ b/.gitignore @@ -178,3 +178,4 @@ cov.xml *.faiss *-structure.csv *-structure.json +metagpt/tools/schemas \ No newline at end of file diff --git a/metagpt/tools/libs/gpt_v_generator.py b/metagpt/tools/libs/gpt_v_generator.py index 6a620f7e8..63fda3e81 100644 --- a/metagpt/tools/libs/gpt_v_generator.py +++ b/metagpt/tools/libs/gpt_v_generator.py @@ -30,9 +30,18 @@ As the design pays tribute to large companies, sometimes it is normal for some c Now, please generate the corresponding webpage code including HTML, CSS and JavaScript:""" -@register_tool(tool_type=ToolTypes.IMAGE2WEBPAGE.type_name) +@register_tool( + tool_type=ToolTypes.IMAGE2WEBPAGE.type_name, include_functions=["__init__", "generate_webpages", "save_webpages"] +) class GPTvGenerator: + """Class for generating webpages at once. + + This class provides methods to generate webpages including all code (HTML, CSS, and JavaScript) based on an image. + It utilizes a vision model to analyze the layout from an image and generate webpage codes accordingly. + """ + def __init__(self): + """Initialize GPTvGenerator class with default values from the configuration.""" from metagpt.config2 import config self.api_key = config.llm.api_key @@ -41,15 +50,42 @@ class GPTvGenerator: self.max_tokens = config.vision_max_tokens def analyze_layout(self, image_path): + """Analyze the layout of the given image and return the result. + + This is a helper method to generate a layout description based on the image. + + Args: + image_path (str): Path of the image to analyze. + + Returns: + str: The layout analysis result. + """ return self.get_result(image_path, ANALYZE_LAYOUT_PROMPT) def generate_webpages(self, image_path): + """Generate webpages including all code (HTML, CSS, and JavaScript) in one go based on the image. + + Args: + image_path (str): The path of the image file. + + Returns: + str: Generated webpages content. + """ layout = self.analyze_layout(image_path) prompt = GENERATE_PROMPT + "\n\n # Context\n The layout information of the sketch image is: \n" + layout result = self.get_result(image_path, prompt) return result def get_result(self, image_path, prompt): + """Get the result from the vision model based on the given image path and prompt. + + Args: + image_path (str): Path of the image to analyze. + prompt (str): Prompt to use for the analysis. + + Returns: + str: The model's response as a string. + """ base64_image = self.encode_image(image_path) headers = {"Content-Type": "application/json", "Authorization": f"Bearer {self.api_key}"} payload = { @@ -74,11 +110,28 @@ class GPTvGenerator: @staticmethod def encode_image(image_path): + """Encode the image at the given path to a base64 string. + + Args: + image_path (str): Path of the image to encode. + + Returns: + str: The base64 encoded string of the image. + """ with open(image_path, "rb") as image_file: return base64.b64encode(image_file.read()).decode("utf-8") @staticmethod def save_webpages(image_path, webpages) -> Path: + """Save webpages including all code (HTML, CSS, and JavaScript) at once. + + Args: + image_path (str): The path of the image file. + webpages (str): The generated webpages content. + + Returns: + Path: The path of the saved webpages. + """ # 在workspace目录下,创建一个名为下webpages的文件夹,用于存储html、css和js文件 webpages_path = DEFAULT_WORKSPACE_ROOT / "webpages" / Path(image_path).stem os.makedirs(webpages_path, exist_ok=True) diff --git a/metagpt/tools/libs/sd_engine.py b/metagpt/tools/libs/sd_engine.py index 6fb16993e..6229a60e3 100644 --- a/metagpt/tools/libs/sd_engine.py +++ b/metagpt/tools/libs/sd_engine.py @@ -53,10 +53,22 @@ payload = { default_negative_prompt = "(easynegative:0.8),black, dark,Low resolution" -@register_tool(tool_type=ToolTypes.STABLE_DIFFUSION.type_name) +@register_tool( + tool_type=ToolTypes.STABLE_DIFFUSION.type_name, + include_functions=["__init__", "simple_run_t2i", "run_t2i", "construct_payload", "save"], +) class SDEngine: + """Generate image using stable diffusion model. + + This class provides methods to interact with a stable diffusion service to generate images based on text inputs. + """ + def __init__(self, sd_url=""): - # Initialize the SDEngine with configuration + """Initialize the SDEngine instance with configuration. + + Args: + sd_url (str, optional): URL of the stable diffusion service. Defaults to "". + """ self.sd_url = sd_url self.sd_t2i_url = f"{self.sd_url}/sdapi/v1/txt2img" # Define default payload settings for SD API @@ -71,7 +83,18 @@ class SDEngine: height=512, sd_model="galaxytimemachinesGTM_photoV20", ): - # Configure the payload with provided inputs + """Modify and set the API parameters for image generation. + + Args: + prompt (str): Text input for image generation. + negtive_prompt (str, optional): Text input for negative prompts. Defaults to None. + width (int, optional): Width of the generated image in pixels. Defaults to 512. + height (int, optional): Height of the generated image in pixels. Defaults to 512. + sd_model (str, optional): The model to use for image generation. Defaults to "galaxytimemachinesGTM_photoV20". + + Returns: + dict: Updated parameters for the stable diffusion API. + """ self.payload["prompt"] = prompt self.payload["negative_prompt"] = negtive_prompt self.payload["width"] = width @@ -81,12 +104,27 @@ class SDEngine: return self.payload def save(self, imgs, save_name=""): + """Save generated images to the output directory. + + Args: + imgs (str): Generated images. + save_name (str, optional): Output image name. Default is empty. + """ save_dir = SOURCE_ROOT / SD_OUTPUT_FILE_REPO if not save_dir.exists(): save_dir.mkdir(parents=True, exist_ok=True) batch_decode_base64_to_image(imgs, str(save_dir), save_name=save_name) def simple_run_t2i(self, payload: dict, auto_save: bool = True): + """Run the stable diffusion API for multiple prompts, calling the stable diffusion API to generate images. + + Args: + payload (dict): Dictionary of input parameters for the stable diffusion API. + auto_save (bool, optional): Save generated images automatically. Defaults to True. + + Returns: + list: The generated images as a result of the API call. + """ with requests.Session() as session: logger.debug(self.sd_t2i_url) rsp = session.post(self.sd_t2i_url, json=payload, timeout=600) @@ -98,7 +136,11 @@ class SDEngine: return results async def run_t2i(self, payloads: List): - # Asynchronously run the SD API for multiple prompts + """Run the stable diffusion API for multiple prompts asynchronously. + + Args: + payloads (list): List of payload, each payload is a dictionary of input parameters for the stable diffusion API. + """ session = ClientSession() for payload_idx, payload in enumerate(payloads): results = await self.run(url=self.sd_t2i_url, payload=payload, session=session) @@ -106,7 +148,16 @@ class SDEngine: await session.close() async def run(self, url, payload, session): - # Perform the HTTP POST request to the SD API + """Perform the HTTP POST request to the SD API. + + Args: + url (str): The API URL. + payload (dict): The payload for the request. + session (ClientSession): The session for making HTTP requests. + + Returns: + list: Images generated by the stable diffusion API. + """ async with session.post(url, json=payload, timeout=600) as rsp: data = await rsp.read() diff --git a/metagpt/tools/schemas/__init__.py b/metagpt/tools/schemas/__init__.py deleted file mode 100644 index e50f67d6f..000000000 --- a/metagpt/tools/schemas/__init__.py +++ /dev/null @@ -1,6 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Time : 2023/11/16 16:33 -# @Author : lidanyang -# @File : __init__.py -# @Desc : diff --git a/metagpt/tools/schemas/data_preprocess/FillMissingValue.yml b/metagpt/tools/schemas/data_preprocess/FillMissingValue.yml deleted file mode 100644 index 44c830a1e..000000000 --- a/metagpt/tools/schemas/data_preprocess/FillMissingValue.yml +++ /dev/null @@ -1,61 +0,0 @@ -FillMissingValue: - type: class - description: "Completing missing values with simple strategies" - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "columns to be processed" - strategy: - type: str - description: "the imputation strategy, notice mean/median can only be used for numeric features" - default: mean - enum: - - mean - - median - - most_frequent - - constant - fill_value: - type: int - description: "fill_value is used to replace all occurrences of missing_values" - default: null - required: - - features - fit: - description: "Fit the FillMissingValue model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." diff --git a/metagpt/tools/schemas/data_preprocess/LabelEncode.yml b/metagpt/tools/schemas/data_preprocess/LabelEncode.yml deleted file mode 100644 index 419ef60a8..000000000 --- a/metagpt/tools/schemas/data_preprocess/LabelEncode.yml +++ /dev/null @@ -1,48 +0,0 @@ -LabelEncode: - type: class - description: "Apply label encoding to specified categorical columns in-place." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "Categorical columns to be label encoded" - required: - - features - fit: - description: "Fit the LabelEncode model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." diff --git a/metagpt/tools/schemas/data_preprocess/MaxAbsScale.yml b/metagpt/tools/schemas/data_preprocess/MaxAbsScale.yml deleted file mode 100644 index 3e17cfdd0..000000000 --- a/metagpt/tools/schemas/data_preprocess/MaxAbsScale.yml +++ /dev/null @@ -1,48 +0,0 @@ -MaxAbsScale: - type: class - description: "cale each feature by its maximum absolute value" - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "columns to be processed" - required: - - features - fit: - description: "Fit the MaxAbsScale model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/schemas/data_preprocess/MinMaxScale.yml b/metagpt/tools/schemas/data_preprocess/MinMaxScale.yml deleted file mode 100644 index 8f050d942..000000000 --- a/metagpt/tools/schemas/data_preprocess/MinMaxScale.yml +++ /dev/null @@ -1,48 +0,0 @@ -MinMaxScale: - type: class - description: "Transform features by scaling each feature to a range, witch is (0, 1)" - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "columns to be processed" - required: - - features - fit: - description: "Fit the MinMaxScale model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." diff --git a/metagpt/tools/schemas/data_preprocess/OneHotEncode.yml b/metagpt/tools/schemas/data_preprocess/OneHotEncode.yml deleted file mode 100644 index f499b2cb8..000000000 --- a/metagpt/tools/schemas/data_preprocess/OneHotEncode.yml +++ /dev/null @@ -1,48 +0,0 @@ -OneHotEncode: - type: class - description: "Apply one-hot encoding to specified categorical columns, the original columns will be dropped." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "Categorical columns to be one-hot encoded and dropped" - required: - - features - fit: - description: "Fit the OneHotEncoding model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." diff --git a/metagpt/tools/schemas/data_preprocess/OrdinalEncode.yml b/metagpt/tools/schemas/data_preprocess/OrdinalEncode.yml deleted file mode 100644 index 79ebaf37c..000000000 --- a/metagpt/tools/schemas/data_preprocess/OrdinalEncode.yml +++ /dev/null @@ -1,46 +0,0 @@ -OrdinalEncode: - type: class - description: Encode categorical features as ordinal integers. - methods: - __init__: - description: 'Initialize the OrdinalEncode instance with feature names. ' - parameters: - properties: - features: - type: list - description: List of categorical feature names to be encoded. - required: - - features - fit: - description: 'Learn the ordinal encodings for the features. ' - parameters: - properties: - df: - type: pd.DataFrame - description: Dataframe containing the categorical features. - required: - - df - fit_transform: - description: 'Fit and transform the input DataFrame. ' - parameters: - properties: - df: - type: pd.DataFrame - description: The input DataFrame. - required: - - df - returns: - - type: pd.DataFrame - description: The transformed DataFrame. - transform: - description: 'Convert the categorical features to ordinal integers. ' - parameters: - properties: - df: - type: pd.DataFrame - description: Dataframe containing the categorical features to be encoded. - required: - - df - returns: - - type: pd.DataFrame - description: A new dataframe with the encoded features. diff --git a/metagpt/tools/schemas/data_preprocess/RobustScale.yml b/metagpt/tools/schemas/data_preprocess/RobustScale.yml deleted file mode 100644 index 6d5dfaf3a..000000000 --- a/metagpt/tools/schemas/data_preprocess/RobustScale.yml +++ /dev/null @@ -1,47 +0,0 @@ -RobustScale: - type: class - description: Apply the RobustScaler to scale features using statistics that are - robust to outliers. - methods: - __init__: - description: 'Initialize the RobustScale instance with feature names. ' - parameters: - properties: - features: - type: list - description: List of feature names to be scaled. - required: - - features - fit: - description: 'Compute the median and IQR for scaling. ' - parameters: - properties: - df: - type: pd.DataFrame - description: Dataframe containing the features. - required: - - df - fit_transform: - description: 'Fit and transform the input DataFrame. ' - parameters: - properties: - df: - type: pd.DataFrame - description: The input DataFrame. - required: - - df - returns: - - type: pd.DataFrame - description: The transformed DataFrame. - transform: - description: 'Scale features using the previously computed median and IQR. ' - parameters: - properties: - df: - type: pd.DataFrame - description: Dataframe containing the features to be scaled. - required: - - df - returns: - - type: pd.DataFrame - description: A new dataframe with scaled features. diff --git a/metagpt/tools/schemas/data_preprocess/StandardScale.yml b/metagpt/tools/schemas/data_preprocess/StandardScale.yml deleted file mode 100644 index cf6e7d57b..000000000 --- a/metagpt/tools/schemas/data_preprocess/StandardScale.yml +++ /dev/null @@ -1,48 +0,0 @@ -StandardScale: - type: class - description: "Standardize features by removing the mean and scaling to unit variance" - methods: - __init__: - description: "Initialize self." - parameters: - properties: - features: - type: list - description: "columns to be processed" - required: - - features - fit: - description: "Fit the StandardScale model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." diff --git a/metagpt/tools/schemas/feature_engineering/CatCount.yml b/metagpt/tools/schemas/feature_engineering/CatCount.yml deleted file mode 100644 index 049fc7879..000000000 --- a/metagpt/tools/schemas/feature_engineering/CatCount.yml +++ /dev/null @@ -1,48 +0,0 @@ -CatCount: - type: class - description: "Add value counts of a categorical column as new feature." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - col: - type: str - description: "Column for value counts." - required: - - col - fit: - description: "Fit the CatCount model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/schemas/feature_engineering/CatCross.yml b/metagpt/tools/schemas/feature_engineering/CatCross.yml deleted file mode 100644 index 5d6303439..000000000 --- a/metagpt/tools/schemas/feature_engineering/CatCross.yml +++ /dev/null @@ -1,52 +0,0 @@ -CatCross: - type: class - description: "Add pairwise crossed features and convert them to numerical features." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - cols: - type: list - description: "Columns to be pairwise crossed, at least 2 columns." - max_cat_num: - type: int - description: "Maximum unique categories per crossed feature." - default: 100 - required: - - cols - fit: - description: "Fit the CatCross model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/schemas/feature_engineering/GeneralSelection.yml b/metagpt/tools/schemas/feature_engineering/GeneralSelection.yml deleted file mode 100644 index 2ebf5b397..000000000 --- a/metagpt/tools/schemas/feature_engineering/GeneralSelection.yml +++ /dev/null @@ -1,48 +0,0 @@ -GeneralSelection: - type: class - description: "Drop all nan feats and feats with only one unique value." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - label_col: - type: str - description: "Label column name." - required: - - label_col - fit: - description: "Fit the GeneralSelection model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/schemas/feature_engineering/GroupStat.yml b/metagpt/tools/schemas/feature_engineering/GroupStat.yml deleted file mode 100644 index 6e0ba2877..000000000 --- a/metagpt/tools/schemas/feature_engineering/GroupStat.yml +++ /dev/null @@ -1,58 +0,0 @@ -GroupStat: - type: class - description: "Aggregate specified column in a DataFrame grouped by another column, adding new features named '__by_'." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - group_col: - type: str - description: "Column used for grouping." - agg_col: - type: str - description: "Column on which aggregation is performed." - agg_funcs: - type: list - description: >- - List of aggregation functions to apply, such as ['mean', 'std']. - Each function must be supported by pandas. - required: - - group_col - - agg_col - - agg_funcs - fit: - description: "Fit the GroupStat model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/schemas/feature_engineering/KFoldTargetMeanEncoder.yml b/metagpt/tools/schemas/feature_engineering/KFoldTargetMeanEncoder.yml deleted file mode 100644 index 79a673f9f..000000000 --- a/metagpt/tools/schemas/feature_engineering/KFoldTargetMeanEncoder.yml +++ /dev/null @@ -1,60 +0,0 @@ -KFoldTargetMeanEncoder: - type: class - description: "Adds a new feature to the DataFrame by k-fold mean encoding of a categorical column using the label column." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - col: - type: str - description: "Column to be k-fold mean encoded." - label: - type: str - description: "Predicted label column." - n_splits: - type: int - description: "Number of splits for K-fold." - default: 5 - random_state: - type: int - description: "Random seed." - default: 2021 - required: - - col - - label - fit: - description: "Fit the KFoldTargetMeanEncoder model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/schemas/feature_engineering/PolynomialExpansion.yml b/metagpt/tools/schemas/feature_engineering/PolynomialExpansion.yml deleted file mode 100644 index 62e6ad5b3..000000000 --- a/metagpt/tools/schemas/feature_engineering/PolynomialExpansion.yml +++ /dev/null @@ -1,548 +0,0 @@ -PolynomialExpansion: - type: class - description: "Add polynomial and interaction features from selected numeric columns to input DataFrame." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - cols: - type: list - description: "Columns for polynomial expansion." - label_col: - type: str - description: "Label column name." - degree: - type: int - description: "The degree of the polynomial features." - default: 2 - required: - - cols - - label_col - fit: - description: "Fit the PolynomialExpansion model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame without duplicated columns." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame without duplicated columns." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -CatCount: - type: class - description: "Add value counts of a categorical column as new feature." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - col: - type: str - description: "Column for value counts." - required: - - col - fit: - description: "Fit the CatCount model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -TargetMeanEncoder: - type: class - description: "Encodes a categorical column by the mean of the label column, and adds the result as a new feature." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - col: - type: str - description: "Column to be mean encoded." - label: - type: str - description: "Predicted label column." - required: - - col - - label - fit: - description: "Fit the TargetMeanEncoder model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -KFoldTargetMeanEncoder: - type: class - description: "Adds a new feature to the DataFrame by k-fold mean encoding of a categorical column using the label column." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - col: - type: str - description: "Column to be k-fold mean encoded." - label: - type: str - description: "Predicted label column." - n_splits: - type: int - description: "Number of splits for K-fold." - default: 5 - random_state: - type: int - description: "Random seed." - default: 2021 - required: - - col - - label - fit: - description: "Fit the KFoldTargetMeanEncoder model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -CatCross: - type: class - description: "Add pairwise crossed features and convert them to numerical features." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - cols: - type: list - description: "Columns to be pairwise crossed, at least 2 columns." - max_cat_num: - type: int - description: "Maximum unique categories per crossed feature." - default: 100 - required: - - cols - fit: - description: "Fit the CatCross model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -GroupStat: - type: class - description: "Aggregate specified column in a DataFrame grouped by another column, adding new features named '__by_'." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - group_col: - type: str - description: "Column used for grouping." - agg_col: - type: str - description: "Column on which aggregation is performed." - agg_funcs: - type: list - description: >- - List of aggregation functions to apply, such as ['mean', 'std']. - Each function must be supported by pandas. - required: - - group_col - - agg_col - - agg_funcs - fit: - description: "Fit the GroupStat model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -SplitBins: - type: class - description: "Inplace binning of continuous data into intervals, returning integer-encoded bin identifiers directly." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - cols: - type: list - description: "Columns to be binned inplace." - strategy: - type: str - description: "Strategy used to define the widths of the bins." - default: quantile - enum: - - quantile - - uniform - - kmeans - required: - - cols - fit: - description: "Fit the SplitBins model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - -GeneralSelection: - type: class - description: "Drop all nan feats and feats with only one unique value." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - label_col: - type: str - description: "Label column name." - required: - - label_col - fit: - description: "Fit the GeneralSelection model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - - -TreeBasedSelection: - type: class - description: "Select features based on tree-based model and remove features with low importance." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - label_col: - type: str - description: "Label column name." - task_type: - type: str - description: "Task type, 'cls' for classification, 'mcls' for multi-class classification, 'reg' for regression." - enum: - - cls - - mcls - - reg - required: - - label_col - - task_type - fit: - description: "Fit the TreeBasedSelection model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." - -VarianceBasedSelection: - type: class - description: "Select features based on variance and remove features with low variance." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - label_col: - type: str - description: "Label column name." - threshold: - type: float - description: "Threshold for variance." - default: 0.0 - required: - - label_col - fit: - description: "Fit the VarianceBasedSelection model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." \ No newline at end of file diff --git a/metagpt/tools/schemas/feature_engineering/SplitBins.yml b/metagpt/tools/schemas/feature_engineering/SplitBins.yml deleted file mode 100644 index 4e0171406..000000000 --- a/metagpt/tools/schemas/feature_engineering/SplitBins.yml +++ /dev/null @@ -1,56 +0,0 @@ -SplitBins: - type: class - description: "Inplace binning of continuous data into intervals, returning integer-encoded bin identifiers directly." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - cols: - type: list - description: "Columns to be binned inplace." - strategy: - type: str - description: "Strategy used to define the widths of the bins." - default: quantile - enum: - - quantile - - uniform - - kmeans - required: - - cols - fit: - description: "Fit the SplitBins model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/schemas/feature_engineering/TargetMeanEncoder.yml b/metagpt/tools/schemas/feature_engineering/TargetMeanEncoder.yml deleted file mode 100644 index 86416ccbb..000000000 --- a/metagpt/tools/schemas/feature_engineering/TargetMeanEncoder.yml +++ /dev/null @@ -1,52 +0,0 @@ -TargetMeanEncoder: - type: class - description: "Encodes a categorical column by the mean of the label column, and adds the result as a new feature." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - col: - type: str - description: "Column to be mean encoded." - label: - type: str - description: "Predicted label column." - required: - - col - - label - fit: - description: "Fit the TargetMeanEncoder model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame." \ No newline at end of file diff --git a/metagpt/tools/schemas/feature_engineering/TreeBasedSelection.yml b/metagpt/tools/schemas/feature_engineering/TreeBasedSelection.yml deleted file mode 100644 index c210effea..000000000 --- a/metagpt/tools/schemas/feature_engineering/TreeBasedSelection.yml +++ /dev/null @@ -1,56 +0,0 @@ -TreeBasedSelection: - type: class - description: "Select features based on tree-based model and remove features with low importance." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - label_col: - type: str - description: "Label column name." - task_type: - type: str - description: "Task type, 'cls' for classification, 'mcls' for multi-class classification, 'reg' for regression." - enum: - - cls - - mcls - - reg - required: - - label_col - - task_type - fit: - description: "Fit the TreeBasedSelection model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." \ No newline at end of file diff --git a/metagpt/tools/schemas/feature_engineering/VarianceBasedSelection.yml b/metagpt/tools/schemas/feature_engineering/VarianceBasedSelection.yml deleted file mode 100644 index 6da4c3e7f..000000000 --- a/metagpt/tools/schemas/feature_engineering/VarianceBasedSelection.yml +++ /dev/null @@ -1,52 +0,0 @@ -VarianceBasedSelection: - type: class - description: "Select features based on variance and remove features with low variance." - methods: - __init__: - description: "Initialize self." - parameters: - properties: - label_col: - type: str - description: "Label column name." - threshold: - type: float - description: "Threshold for variance." - default: 0.0 - required: - - label_col - fit: - description: "Fit the VarianceBasedSelection model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - transform: - description: "Transform the input DataFrame with the fitted model." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." - fit_transform: - description: "Fit and transform the input DataFrame." - parameters: - properties: - df: - type: DataFrame - description: "The input DataFrame." - required: - - df - returns: - df: - type: DataFrame - description: "The transformed DataFrame contain label_col." \ No newline at end of file diff --git a/metagpt/tools/schemas/image2webpage/GPTvGenerator.yml b/metagpt/tools/schemas/image2webpage/GPTvGenerator.yml deleted file mode 100644 index 1ba2c2b08..000000000 --- a/metagpt/tools/schemas/image2webpage/GPTvGenerator.yml +++ /dev/null @@ -1,36 +0,0 @@ -GPTvGenerator: - type: class - description: "Class for generating webpages at once." - methods: - __init__: - description: "Initialize Vision class with default values." - - generate_webpages: - description: "Generate webpages including all code(HTML, CSS and JavaScript) in one go based on the image." - parameters: - properties: - image_path: - type: str - description: "The path of the image file" - required: - - image_path - returns: - type: str - description: "Generated webpages content." - - save_webpages: - description: "Save webpages including all code(HTML, CSS and JavaScript) at once" - parameters: - properties: - image_path: - type: str - description: "The path of the image file" - webpages: - type: str - description: "The generated webpages content" - required: - - image_path - - webpages - returns: - type: Path - description: "The path of the saved webpages" \ No newline at end of file diff --git a/metagpt/tools/schemas/stable_diffusion/SDEngine.yml b/metagpt/tools/schemas/stable_diffusion/SDEngine.yml deleted file mode 100644 index a93742a1d..000000000 --- a/metagpt/tools/schemas/stable_diffusion/SDEngine.yml +++ /dev/null @@ -1,58 +0,0 @@ -SDEngine: - type: class - description: "Generate image using stable diffusion model" - methods: - __init__: - description: "Initialize the SDEngine instance." - parameters: - properties: - sd_url: - type: str - description: "URL of the stable diffusion service." - simple_run_t2i: - description: "Run the stable diffusion API for multiple prompts, calling the stable diffusion API to generate images." - parameters: - properties: - payload: - type: dict - description: "Dictionary of input parameters for the stable diffusion API." - auto_save: - type: bool - description: "Save generated images automatically." - required: - - prompts - run_t2i: - type: async function - description: "Run the stable diffusion API for multiple prompts, calling the stable diffusion API to generate images." - parameters: - properties: - payloads: - type: list - description: "List of payload, each payload is a dictionary of input parameters for the stable diffusion API." - required: - - payloads - construct_payload: - description: "Modify and set the API parameters for image generation." - parameters: - properties: - prompt: - type: str - description: "Text input for image generation." - required: - - prompt - returns: - payload: - type: dict - description: "Updated parameters for the stable diffusion API." - save: - description: "Save generated images to the output directory." - parameters: - properties: - imgs: - type: str - description: "Generated images." - save_name: - type: str - description: "Output image name. Default is empty." - required: - - imgs diff --git a/metagpt/tools/schemas/web_scraping/scrape_web_playwright.yml b/metagpt/tools/schemas/web_scraping/scrape_web_playwright.yml deleted file mode 100644 index a6ff7d6c7..000000000 --- a/metagpt/tools/schemas/web_scraping/scrape_web_playwright.yml +++ /dev/null @@ -1,21 +0,0 @@ -scrape_web_playwright: - type: async funciton - description: "Scrape and save the HTML structure and inner text content of a web page using Playwright." - parameters: - properties: - url: - type: str - description: "web url" - \*url: - type: Non-Keyword Arguments - description: "other web urls, you can assagin sub url link to it." - required: - - url - returns: - inner_text: - type: str - description: The inner text content of the web page. - html: - type: str - description: The html structure of the web page. - From 6754d890a12b7725d1cc740901679d59d63e316a Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 2 Feb 2024 20:28:08 +0800 Subject: [PATCH 588/637] update ci cache --- tests/data/rsp_cache.json | 162 +++++++++++++------------------------- 1 file changed, 54 insertions(+), 108 deletions(-) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index e81a9d073..6ee41e076 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -142,104 +142,6 @@ "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.sh\n## Development Code\n```python\necho 'Hello World'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\necho Hello World\n## Running Output\nstandard output: \n```text\nHello World\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "The running result indicates no error. The code works fine.\n\n## File To Rewrite:\nNo file needs to be rewritten.\n\n## Status:\nPASS\n\n## Send To:\nNoOne\n\n---", "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.py\n## Development Code\n```python\npython -c \"print(1/0)\"\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\npython -c print(1/0)\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\nTraceback (most recent call last):\n File \"\", line 1, in \nZeroDivisionError: division by zero\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write Engineer if the errors are due to problematic development codes, and QaEngineer to problematic test codes, and NoOne if there are no errors,\nWRITE ONLY ONE WORD, Engineer OR QaEngineer OR NoOne, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\nThe error is caused by the development code in file a.py, which attempts to divide by zero. To fix the error, the development code in a.py should be modified to handle the ZeroDivisionError, for example by using a try-except block.\n\nFile To Rewrite:\na.py\n\nStatus:\nFAIL\n\nSend To:\nEngineer\n---", "\nNOTICE\n1. Role: You are a Development Engineer or QA engineer;\n2. Task: You received this message from another Development Engineer or QA engineer who ran or tested your code. \nBased on the message, first, figure out your own role, i.e. Engineer or QaEngineer,\nthen rewrite the development code or the test code based on your role, the error, and the summary, such that all bugs are fixed and the code performs well.\nAttention: Use '##' to split sections, not '#', and '## ' SHOULD WRITE BEFORE the test case or script and triple quotes.\nThe message is as follows:\n# Legacy Code\n```python\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n if self.score > 21 and any(card.rank == 'A' for card in self.hand):\n self.score -= 10\n return self.score\n\n```\n---\n# Unit Test Code\n```python\n\nimport unittest\nfrom blackjack_game.player import Player\nfrom blackjack_game.deck import Deck\nfrom blackjack_game.card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n\n\n```\n---\n# Console logs\n```text\n\n```\n---\nNow you should start rewriting the code:\n## file name of the code to rewrite: Write code with triple quote. Do your best to implement THIS IN ONLY ONE FILE.\n": "```python\n## Development Engineer\n\nfrom typing import List\nfrom deck import Deck\nfrom card import Card\n\nclass Player:\n \"\"\"\n A class representing a player in the Black Jack game.\n \"\"\"\n\n def __init__(self, name: str):\n \"\"\"\n Initialize a Player object.\n \n Args:\n name (str): The name of the player.\n \"\"\"\n self.name = name\n self.hand: List[Card] = []\n self.score = 0\n\n def draw(self, deck: Deck):\n \"\"\"\n Draw a card from the deck and add it to the player's hand.\n \n Args:\n deck (Deck): The deck of cards.\n \"\"\"\n card = deck.draw_card()\n self.hand.append(card)\n self.calculate_score()\n\n def calculate_score(self) -> int:\n \"\"\"\n Calculate the score of the player's hand.\n \n Returns:\n int: The score of the player's hand.\n \"\"\"\n self.score = sum(card.value for card in self.hand)\n # Handle the case where Ace is counted as 11 and causes the score to exceed 21\n ace_count = sum(1 for card in self.hand if card.rank == 'A')\n while self.score > 21 and ace_count > 0:\n self.score -= 10\n ace_count -= 1\n return self.score\n```\n\n```python\n## QaEngineer\n\nimport unittest\nfrom player import Player\nfrom deck import Deck\nfrom card import Card\n\nclass TestPlayer(unittest.TestCase):\n ## Test the Player's initialization\n def test_player_initialization(self):\n player = Player(\"Test Player\")\n self.assertEqual(player.name, \"Test Player\")\n self.assertEqual(player.hand, [])\n self.assertEqual(player.score, 0)\n\n ## Test the Player's draw method\n def test_player_draw(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n self.assertEqual(len(player.hand), 1)\n self.assertEqual(player.score, player.hand[0].value)\n\n ## Test the Player's calculate_score method\n def test_player_calculate_score(self):\n deck = Deck()\n player = Player(\"Test Player\")\n player.draw(deck)\n player.draw(deck)\n self.assertEqual(player.score, sum(card.value for card in player.hand))\n\n ## Test the Player's calculate_score method with Ace card\n def test_player_calculate_score_with_ace(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('K', 'Hearts', 10))\n player.calculate_score()\n self.assertEqual(player.score, 21)\n\n ## Test the Player's calculate_score method with multiple Aces\n def test_player_calculate_score_with_multiple_aces(self):\n player = Player(\"Test Player\")\n player.hand.append(Card('A', 'Hearts', 11))\n player.hand.append(Card('A', 'Diamonds', 11))\n player.calculate_score()\n self.assertEqual(player.score, 12)\n\nif __name__ == '__main__':\n unittest.main()\n```", - "\n # Context:\n user: \n## User Requirement\nThis is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Perform exploratory data analysis on the train dataset to understand the features and target variable.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Preprocess the data by handling missing values, encoding categorical variables, and scaling features.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Conduct feature engineering to create new features that may help improve model performance.\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Select and train a machine learning model using the processed train dataset.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Evaluate the model's accuracy using the eval dataset and report the results.\"\n }\n]\n```", - "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Perform exploratory data analysis on the train dataset to understand the features and target variable.\\nTask 2: Preprocess the data by handling missing values, encoding categorical variables, and scaling features.\\nTask 3: Conduct feature engineering to create new features that may help improve model performance.\\nTask 4: Select and train a machine learning model using the processed train dataset.\\nTask 5: Evaluate the model's accuracy using the eval dataset and report the results.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { - "task_type": [ - "eda", - "data_preprocess", - "feature_engineering", - "model_train", - "model_evaluate" - ] - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n\\n```end\\n\\n## Current Task\\nPerform exploratory data analysis on the train dataset to understand the features and target variable.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { - "code": "# Step 1: Load the training dataset\nimport pandas as pd\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\n\n# Step 2: Display the first few rows of the dataset\ndisplay(train_data.head())\n\n# Step 3: Get a summary of the dataset\ntrain_data.info()\n\n# Step 4: Describe the numerical features\ntrain_data.describe()\n\n# Step 5: Describe the categorical features\ntrain_data.describe(include=['O'])\n\n# Step 6: Check for missing values\nmissing_values = train_data.isnull().sum()\nmissing_values[missing_values > 0]\n\n# Step 7: Visualize the distribution of the target variable 'Survived'\nimport seaborn as sns\nimport matplotlib.pyplot as plt\nsns.countplot(x='Survived', data=train_data)\nplt.title('Distribution of Survival')\nplt.show()\n\n# Step 8: Explore the relationship between features and the target variable\n# Visualize the survival rate by sex\nsns.barplot(x='Sex', y='Survived', data=train_data)\nplt.title('Survival Rate by Sex')\nplt.show()\n\n# Visualize the survival rate by class\nsns.barplot(x='Pclass', y='Survived', data=train_data)\nplt.title('Survival Rate by Class')\nplt.show()\n\n# Visualize the survival rate by embarkation port\nsns.barplot(x='Embarked', y='Survived', data=train_data)\nplt.title('Survival Rate by Embarkation Port')\nplt.show()" - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { - "code": "from metagpt.tools.libs.data_preprocess import get_column_info\n\ncolumn_info = get_column_info(train_data)\nprint(\"column_info\")\nprint(column_info)" - }, - "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nPreprocess the data by handling missing values, encoding categorical variables, and scaling features.\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Completing missing values with simple strategies'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { - "recommend_tools": [ - "FillMissingValue" - ] - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n```end\\n\\n## Current Task\\nPreprocess the data by handling missing values, encoding categorical variables, and scaling features.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Name', 'Sex', 'Ticket', 'Cabin', 'Embarked'], 'Numeric': ['PassengerId', 'Survived', 'Pclass', 'Age', 'SibSp', 'Parch', 'Fare'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}, 'strategy': {'type': 'str', 'description': 'the imputation strategy, notice mean/median can only be used for numeric features', 'default': 'mean', 'enum': ['mean', 'median', 'most_frequent', 'constant']}, 'fill_value': {'type': 'int', 'description': 'fill_value is used to replace all occurrences of missing_values', 'default': None}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { - "code": "# Step 1: Handle missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create copies of the datasets to avoid changing the original data\ntrain_processed = train_data.copy()\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\n\n# Fill missing values for 'Age' with median and 'Embarked' with most frequent value\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\ntrain_processed = fill_missing_age.fit_transform(train_processed)\neval_processed = fill_missing_age.transform(eval_data)\n\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\neval_processed = fill_missing_embarked.transform(eval_data)\n\n# Since 'Cabin' has a lot of missing values, we'll create a binary feature indicating whether cabin info is missing\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\n\n# Step 2: Encode categorical variables\nfrom sklearn.preprocessing import LabelEncoder\n\nlabel_encoder = LabelEncoder()\n\n# Encode 'Sex'\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\n\n# Encode 'Embarked'\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\n\n# Step 3: Scale features\nfrom sklearn.preprocessing import StandardScaler\n\nscaler = StandardScaler()\n\n# Scale 'Age' and 'Fare'\nfor feature in ['Age', 'Fare']:\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\n\n# Display the first few rows of the processed train dataset\ndisplay(train_processed.head())" - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { - "code": "from metagpt.tools.libs.data_preprocess import get_column_info\ncolumn_info = get_column_info(train_processed)\nprint(\"column_info\")\nprint(column_info)" - }, - "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nConduct feature engineering to create new features that may help improve model performance.\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'CatCross': 'Add pairwise crossed features and convert them to numerical features.'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { - "recommend_tools": [ - "CatCross" - ] - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n```end\\n\\n## Current Task\\nConduct feature engineering to create new features that may help improve model performance.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Name', 'Ticket', 'Cabin'], 'Numeric': ['PassengerId', 'Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about feature engineering. when performing it, please adhere to the following principles:\\n- Generate as diverse features as possible to improve the model's performance step-by-step. \\n- If potential impactful features are not included in 'Code Steps', add new steps to generate them.\\n- Avoid creating redundant or excessively numerous features in one step.\\n- Exclude ID columns from feature generation and remove them.\\n- Each step do feature engineering to train, must do same for test separately at the same time.\\n- Avoid using the label column to create features, except for cat encoding.\\n- Use the data from previous task result if exist, do not mock or reload data yourself.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'CatCross': {'type': 'class', 'description': 'Add pairwise crossed features and convert them to numerical features.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'cols': {'type': 'list', 'description': 'Columns to be pairwise crossed, at least 2 columns.'}, 'max_cat_num': {'type': 'int', 'description': 'Maximum unique categories per crossed feature.', 'default': 100}}}, 'required': ['cols']}, 'fit': {'description': 'Fit the CatCross model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/feature_engineering.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { - "code": "# Step 1: Create a feature for the number of family members on board\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\n\n# Step 2: Create a feature to indicate whether a passenger was alone or not\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\n\n# Step 3: Create a feature for the title extracted from the passenger's name\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\.', expand=False)\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\.', expand=False)\n\n# Step 4: Create a feature for the deck extracted from the cabin number\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\n\n# Step 5: Create a feature for the ticket class and sex cross\n# Tools used: ['CatCross']\nfrom metagpt.tools.libs.feature_engineering import CatCross\n\n# Initialize CatCross with the desired columns\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\n\n# Fit and transform the train data\ntrain_processed = cat_cross.fit_transform(train_processed)\n\n# Transform the eval data\neval_processed = cat_cross.transform(eval_processed)\n\n# Remove unnecessary columns\nfor df in [train_processed, eval_processed]:\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)" - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { - "code": "from metagpt.tools.libs.data_preprocess import get_column_info\ncolumn_info = get_column_info(train_processed)\nprint(\"column_info\")\nprint(column_info)" - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n## Current Task\\nSelect and train a machine learning model using the processed train dataset.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Title', 'Deck'], 'Numeric': ['Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind', 'Family_Size', 'Is_Alone', 'Pclass_Sex'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about training a model, please ensure high performance:\\n- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as XGBoost, CatBoost, etc.\\n- If non-numeric columns exist, perform label encode together with all steps.\\n- Use the data from previous task result directly, do not mock or reload data yourself.\\n- Set suitable hyperparameters for the model, make metrics as high as possible.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { - "code": "from sklearn.model_selection import train_test_split\nfrom sklearn.metrics import accuracy_score\nfrom xgboost import XGBClassifier\n\n# Step 1: Prepare the data\n# Separate features and target variable\nX = train_processed.drop('Survived', axis=1)\ny = train_processed['Survived']\n\n# Step 2: Split the data into training and validation sets\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\n\n# Step 3: Initialize and train the XGBoost model\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\nmodel.fit(X_train, y_train)\n\n# Step 4: Evaluate the model\ny_pred = model.predict(X_val)\naccuracy = accuracy_score(y_val, y_pred)\nprint(f'Validation Accuracy: {accuracy:.4f}')" - }, - "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\n[user: \\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n```end\\n\\n## Current Task\\nSelect and train a machine learning model using the processed train dataset.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\ncolumn_info\\n{'Category': ['Title', 'Deck'], 'Numeric': ['Survived', 'Pclass', 'Sex', 'Age', 'SibSp', 'Parch', 'Fare', 'Embarked', 'Cabin_Ind', 'Family_Size', 'Is_Alone', 'Pclass_Sex'], 'Datetime': [], 'Others': []}\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about training a model, please ensure high performance:\\n- Keep in mind that your user prioritizes results and is highly focused on model performance. So, when needed, feel free to use models of any complexity to improve effectiveness, such as XGBoost, CatBoost, etc.\\n- If non-numeric columns exist, perform label encode together with all steps.\\n- Use the data from previous task result directly, do not mock or reload data yourself.\\n- Set suitable hyperparameters for the model, make metrics as high as possible.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n]\\n\\n[previous impl]\\nfrom sklearn.model_selection import train_test_split\\nfrom sklearn.metrics import accuracy_score\\nfrom xgboost import XGBClassifier\\n\\n# Step 1: Prepare the data\\n# Separate features and target variable\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\n\\n# Step 2: Split the data into training and validation sets\\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\\n\\n# Step 3: Initialize and train the XGBoost model\\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X_train, y_train)\\n\\n# Step 4: Evaluate the model\\ny_pred = model.predict(X_val)\\naccuracy = accuracy_score(y_val, y_pred)\\nprint(f'Validation Accuracy: {accuracy:.4f}')\\n[runtime Error]\\n[assistant: from sklearn.model_selection import train_test_split\\nfrom sklearn.metrics import accuracy_score\\nfrom xgboost import XGBClassifier\\n\\n# Step 1: Prepare the data\\n# Separate features and target variable\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\n\\n# Step 2: Split the data into training and validation sets\\nX_train, X_val, y_train, y_val = train_test_split(X, y, test_size=0.2, random_state=42)\\n\\n# Step 3: Initialize and train the XGBoost model\\n# Note: Hyperparameters are set to values that generally perform well, but further tuning could be done.\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X_train, y_train)\\n\\n# Step 4: Evaluate the model\\ny_pred = model.predict(X_val)\\naccuracy = accuracy_score(y_val, y_pred)\\nprint(f'Validation Accuracy: {accuracy:.4f}'), user: Executed code failed, please reflect the cause of bug and then debug. Truncated to show only last 2000 characters\\n= self._temporary_data\\n 622 else:\\n--> 623 new, cat_codes, feature_names, feature_types = _proxy_transform(\\n 624 data,\\n 625 feature_names,\\n 626 feature_types,\\n 627 self._enable_categorical,\\n 628 )\\n 629 # Stage the data, meta info are copied inside C++ MetaInfo.\\n 630 self._temporary_data = (new, cat_codes, feature_names, feature_types)\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:1315, in _proxy_transform(data, feature_names, feature_types, enable_categorical)\\n 1313 data = pd.DataFrame(data)\\n 1314 if _is_pandas_df(data):\\n-> 1315 arr, feature_names, feature_types = _transform_pandas_df(\\n 1316 data, enable_categorical, feature_names, feature_types\\n 1317 )\\n 1318 arr, _ = _ensure_np_dtype(arr, arr.dtype)\\n 1319 return arr, None, feature_names, feature_types\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:490, in _transform_pandas_df(data, enable_categorical, feature_names, feature_types, meta, meta_type)\\n 483 for dtype in data.dtypes:\\n 484 if not (\\n 485 (dtype.name in _pandas_dtype_mapper)\\n 486 or is_pd_sparse_dtype(dtype)\\n 487 or (is_pd_cat_dtype(dtype) and enable_categorical)\\n 488 or is_pa_ext_dtype(dtype)\\n 489 ):\\n--> 490 _invalid_dataframe_dtype(data)\\n 491 if is_pa_ext_dtype(dtype):\\n 492 pyarrow_extension = True\\n\\nFile ~/miniconda3/envs/mg_temp/lib/python3.9/site-packages/xgboost/data.py:308, in _invalid_dataframe_dtype(data)\\n 306 type_err = \\\"DataFrame.dtypes for data must be int, float, bool or category.\\\"\\n 307 msg = f\\\"\\\"\\\"{type_err} {_ENABLE_CAT_ERR} {err}\\\"\\\"\\\"\\n--> 308 raise ValueError(msg)\\n\\nValueError: DataFrame.dtypes for data must be int, float, bool or category. When categorical type is supplied, The experimental DMatrix parameter`enable_categorical` must be set to `True`. Invalid columns:Title: object, Deck: object\\n\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { - "reflection": "The error message indicates that the XGBoost model cannot handle columns with 'object' data types directly. The columns 'Title' and 'Deck' are of 'object' type and need to be encoded as numeric values before they can be used for training the XGBoost model. The error also suggests that if we want to use categorical data, we must set the `enable_categorical` parameter to `True` when creating the DMatrix. However, a more common practice is to convert these object types to numeric types using techniques such as label encoding or one-hot encoding. To resolve this issue, we should encode the 'Title' and 'Deck' columns using label encoding before training the model. Additionally, we should ensure that any other non-numeric columns are also converted to numeric types if they exist.", - "improved_impl": "from sklearn.preprocessing import LabelEncoder\n\n# Step 1: Encode non-numeric columns\nencoder = LabelEncoder()\ntrain_processed['Title'] = encoder.fit_transform(train_processed['Title'])\neval_processed['Title'] = encoder.transform(eval_processed['Title'])\ntrain_processed['Deck'] = encoder.fit_transform(train_processed['Deck'].astype(str))\neval_processed['Deck'] = encoder.transform(eval_processed['Deck'].astype(str))\n\n# Step 2: Prepare the data\n# Separate features and target variable\nX = train_processed.drop('Survived', axis=1)\ny = train_processed['Survived']\n\n# Step 3: Train the XGBoost model\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\nmodel.fit(X, y)\n\n# Step 4: Evaluate the model on evaluation data\nX_eval = eval_processed.drop('Survived', axis=1)\ny_eval = eval_processed['Survived']\ny_pred_eval = model.predict(X_eval)\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')" - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\nimport pandas as pd\\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\\ndisplay(train_data.head())\\ntrain_data.info()\\ntrain_data.describe()\\ntrain_data.describe(include=['O'])\\nmissing_values = train_data.isnull().sum()\\nmissing_values[missing_values > 0]\\nimport seaborn as sns\\nimport matplotlib.pyplot as plt\\nsns.countplot(x='Survived', data=train_data)\\nplt.title('Distribution of Survival')\\nplt.show()\\nsns.barplot(x='Sex', y='Survived', data=train_data)\\nplt.title('Survival Rate by Sex')\\nplt.show()\\nsns.barplot(x='Pclass', y='Survived', data=train_data)\\nplt.title('Survival Rate by Class')\\nplt.show()\\nsns.barplot(x='Embarked', y='Survived', data=train_data)\\nplt.title('Survival Rate by Embarkation Port')\\nplt.show()\\n\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\ntrain_processed = train_data.copy()\\neval_data = pd.read_csv('tests/data/ml_datasets/titanic/split_eval.csv')\\nfill_missing_age = FillMissingValue(features=['Age'], strategy='median')\\ntrain_processed = fill_missing_age.fit_transform(train_processed)\\neval_processed = fill_missing_age.transform(eval_data)\\nfill_missing_embarked = FillMissingValue(features=['Embarked'], strategy='most_frequent')\\ntrain_processed = fill_missing_embarked.fit_transform(train_processed)\\neval_processed = fill_missing_embarked.transform(eval_data)\\ntrain_processed['Cabin_Ind'] = train_processed['Cabin'].isnull().astype(int)\\neval_processed['Cabin_Ind'] = eval_processed['Cabin'].isnull().astype(int)\\nfrom sklearn.preprocessing import LabelEncoder\\nlabel_encoder = LabelEncoder()\\ntrain_processed['Sex'] = label_encoder.fit_transform(train_processed['Sex'])\\neval_processed['Sex'] = label_encoder.transform(eval_processed['Sex'])\\ntrain_processed['Embarked'] = label_encoder.fit_transform(train_processed['Embarked'])\\neval_processed['Embarked'] = label_encoder.transform(eval_processed['Embarked'])\\nfrom sklearn.preprocessing import StandardScaler\\nscaler = StandardScaler()\\nfor feature in ['Age', 'Fare']:\\n train_processed[[feature]] = scaler.fit_transform(train_processed[[feature]])\\n eval_processed[[feature]] = scaler.transform(eval_processed[[feature]])\\ndisplay(train_processed.head())\\n\\ntrain_processed['Family_Size'] = train_processed['SibSp'] + train_processed['Parch']\\neval_processed['Family_Size'] = eval_processed['SibSp'] + eval_processed['Parch']\\ntrain_processed['Is_Alone'] = (train_processed['Family_Size'] == 0).astype(int)\\neval_processed['Is_Alone'] = (eval_processed['Family_Size'] == 0).astype(int)\\ntrain_processed['Title'] = train_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\neval_processed['Title'] = eval_processed['Name'].str.extract(' ([A-Za-z]+)\\\\.', expand=False)\\ntrain_processed['Deck'] = train_processed['Cabin'].str[0]\\neval_processed['Deck'] = eval_processed['Cabin'].str[0]\\nfrom metagpt.tools.libs.feature_engineering import CatCross\\ncat_cross = CatCross(cols=['Pclass', 'Sex'])\\ntrain_processed = cat_cross.fit_transform(train_processed)\\neval_processed = cat_cross.transform(eval_processed)\\nfor df in [train_processed, eval_processed]:\\n df.drop(['PassengerId', 'Name', 'Ticket', 'Cabin'], axis=1, inplace=True)\\n\\nfrom sklearn.preprocessing import LabelEncoder\\nencoder = LabelEncoder()\\ntrain_processed['Title'] = encoder.fit_transform(train_processed['Title'])\\neval_processed['Title'] = encoder.transform(eval_processed['Title'])\\ntrain_processed['Deck'] = encoder.fit_transform(train_processed['Deck'].astype(str))\\neval_processed['Deck'] = encoder.transform(eval_processed['Deck'].astype(str))\\nX = train_processed.drop('Survived', axis=1)\\ny = train_processed['Survived']\\nmodel = XGBClassifier(n_estimators=100, learning_rate=0.05, max_depth=4, subsample=0.8, colsample_bytree=0.8, random_state=42)\\nmodel.fit(X, y)\\nX_eval = eval_processed.drop('Survived', axis=1)\\ny_eval = eval_processed['Survived']\\ny_pred_eval = model.predict(X_eval)\\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')\\n```end\\n\\n## Current Task\\nEvaluate the model's accuracy using the eval dataset and report the results.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about evaluating a model, please note the following:\\n- Ensure that the evaluated data is same processed as the training data. If not, remember use object in 'Done Tasks' to transform the data.\\n- Use trained model from previous task result directly, do not mock or reload model yourself.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { - "code": "# Evaluate the model's accuracy using the eval dataset\nfrom sklearn.metrics import accuracy_score\n\n# Predict the target for the evaluation set\ny_pred_eval = model.predict(X_eval)\n\n# Calculate the accuracy for the evaluation set\naccuracy_eval = accuracy_score(y_eval, y_pred_eval)\n\n# Report the accuracy\nprint(f'Evaluation Accuracy: {accuracy_eval:.4f}')" - }, - "\n # Context:\n user: \n## User Requirement\nRun data analysis on sklearn Iris dataset, include a plot\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Load the sklearn Iris dataset.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis on the Iris dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Create a plot visualizing the Iris dataset features.\"\n }\n]\n```", - "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Load the sklearn Iris dataset.\\nTask 2: Perform exploratory data analysis on the Iris dataset.\\nTask 3: Create a plot visualizing the Iris dataset features.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { - "task_type": [ - "other", - "eda", - "other" - ] - }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"1\\\",\\\"dependent_task_ids\\\":[],\\\"instruction\\\":\\\"Load the sklearn Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from sklearn.datasets import load_iris\niris_data = load_iris()" - }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "import pandas as pd\n\n# Convert the loaded Iris dataset into a DataFrame for easier manipulation\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\niris_df['target'] = iris_data.target\n\n# Display basic statistical details like percentile, mean, std etc. of a data frame\niris_stats = iris_df.describe()\n\n# Display the first few rows of the DataFrame\niris_head = iris_df.head()\n\n# Display the class distribution\niris_target_counts = iris_df['target'].value_counts()\n\n# Output the results\nprint('Basic Statistical Details:\\n', iris_stats)\nprint('\\nFirst Five Rows:\\n', iris_head)\nprint('\\nClass Distribution:\\n', iris_target_counts)" - }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\n\\\\n# Convert the loaded Iris dataset into a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\\\\niris_df['target'] = iris_data.target\\\\n\\\\n# Display basic statistical details like percentile, mean, std etc. of a data frame\\\\niris_stats = iris_df.describe()\\\\n\\\\n# Display the first few rows of the DataFrame\\\\niris_head = iris_df.head()\\\\n\\\\n# Display the class distribution\\\\niris_target_counts = iris_df['target'].value_counts()\\\\n\\\\n# Output the results\\\\nprint('Basic Statistical Details:\\\\\\\\n', iris_stats)\\\\nprint('\\\\\\\\nFirst Five Rows:\\\\\\\\n', iris_head)\\\\nprint('\\\\\\\\nClass Distribution:\\\\\\\\n', iris_target_counts)\\\",\\n \\\"result\\\": \\\"Basic Statistical Details:\\\\n sepal length (cm) sepal width (cm) petal length (cm) \\\\\\\\\\\\ncount 150.000000 150.000000 150.000000 \\\\nmean 5.843333 3.057333 3.758000 \\\\nstd 0.828066 0.435866 1.765298 \\\\nmin 4.300000 2.000000 1.000000 \\\\n25% 5.100000 2.800000 1.600000 \\\\n50% 5.800000 3.000000 4.350000 \\\\n75% 6.400000 3.300000 5.100000 \\\\nmax 7.900000 4.400000 6.900000 \\\\n\\\\n petal width (cm) target \\\\ncount 150.000000 150.000000 \\\\nmean 1.199333 1.000000 \\\\nstd 0.762238 0.819232 \\\\nmin 0.100000 0.000000 \\\\n25% 0.300000 0.000000 \\\\n50% 1.300000 1.000000 \\\\n75% 1.800000 2.000000 \\\\nmax 2.500000 2.000000 \\\\n\\\\nFirst Five Rows:\\\\n sepal length (cm) sepal width (cm) petal length (cm) petal width (cm) \\\\\\\\\\\\n0 5.1 3.5 1.4 0.2 \\\\n1 4.9 3.0 1.4 0.2 \\\\n2 4.7 3.2 1.3 0.2 \\\\n3 4.6 3.1 1.5 0.2 \\\\n4 5.0 3.6 1.4 0.2 \\\\n\\\\n target \\\\n0 0 \\\\n1 0 \\\\n2 0 \\\\n3 0 \\\\n4 0 \\\\n\\\\nClass Distribution:\\\\n target\\\\n0 50\\\\n1 50\\\\n2 50\\\\nName: count, dtype: int64\\\\n\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset features.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "import matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Set the style of seaborn\nsns.set(style=\"whitegrid\")\n\n# Create a pairplot to visualize the relationships between features\npair_plot = sns.pairplot(iris_df, hue='target', markers=['o', 's', 'D'], palette='husl')\n\n# Show the plot\nplt.show()" - }, - "\n # Context:\n user: run analysis on sklearn iris dataset\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 5 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Import the Iris dataset from scikit-learn.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis to understand the dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Preprocess the data if necessary (e.g., scaling, encoding).\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Split the dataset into training and testing sets.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Choose a suitable model and train it on the dataset.\"\n },\n {\n \"task_id\": \"6\",\n \"dependent_task_ids\": [\"5\"],\n \"instruction\": \"Evaluate the model's performance on the test set.\"\n },\n {\n \"task_id\": \"7\",\n \"dependent_task_ids\": [\"6\"],\n \"instruction\": \"Report the results of the analysis.\"\n }\n]\n```", - "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\n对数据集进行数据清洗\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Completing missing values with simple strategies', 'MinMaxScale': 'Transform features by scaling each feature to a range, witch is (0, 1)', 'StandardScale': 'Standardize features by removing the mean and scaling to unit variance', 'MaxAbsScale': 'cale each feature by its maximum absolute value', 'RobustScale': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'OrdinalEncode': 'Encode categorical features as ordinal integers.', 'OneHotEncode': 'Apply one-hot encoding to specified categorical columns, the original columns will be dropped.', 'LabelEncode': 'Apply label encoding to specified categorical columns in-place.'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { - "recommend_tools": [ - "FillMissingValue", - "MinMaxScale", - "StandardScale", - "RobustScale", - "OneHotEncode" - ] - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}, 'strategy': {'type': 'str', 'description': 'the imputation strategy, notice mean/median can only be used for numeric features', 'default': 'mean', 'enum': ['mean', 'median', 'most_frequent', 'constant']}, 'fill_value': {'type': 'int', 'description': 'fill_value is used to replace all occurrences of missing_values', 'default': None}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, witch is (0, 1)', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MinMaxScale model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the StandardScale model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize the RobustScale instance with feature names. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'List of feature names to be scaled.'}}, 'required': ['features']}}, 'fit': {'description': 'Compute the median and IQR for scaling. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Scale features using the previously computed median and IQR. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features to be scaled.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'A new dataframe with scaled features.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'OneHotEncode': {'type': 'class', 'description': 'Apply one-hot encoding to specified categorical columns, the original columns will be dropped.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Categorical columns to be one-hot encoded and dropped'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the OneHotEncoding model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { - "code": "# Step 1: Fill missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\ndf_processed = df.copy()\n\n# Identify numeric columns with missing values\nnumeric_cols_with_na = df_processed.select_dtypes(include=['number']).columns[df_processed.isna().any()].tolist()\n\n# Fill missing values with the mean for numeric columns\nfill_missing_value = FillMissingValue(features=numeric_cols_with_na, strategy='mean')\ndf_processed = fill_missing_value.fit_transform(df_processed)\n\n# Step 2: Scale numeric columns\nfrom metagpt.tools.libs.data_preprocess import MinMaxScale\n\n# Identify numeric columns\nnumeric_cols = df_processed.select_dtypes(include=['number']).columns.tolist()\n\n# Scale numeric columns\nmin_max_scaler = MinMaxScale(features=numeric_cols)\ndf_processed = min_max_scaler.fit_transform(df_processed)\n\n# Note: Since there is no separate test set provided, the scaling is only applied to the existing dataset." - }, - "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nclean and preprocess the data\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Filling missing values', 'SplitBins': 'Bin continuous data into intervals and return the bin identifier encoded as an integer value'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { - "recommend_tools": [ - "FillMissingValue" - ] - }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code_steps='', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code_steps='', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}, 'strategy': {'type': 'str', 'description': 'the imputation strategy, notice mean/median can only be used for numeric features', 'default': 'mean', 'enum': ['mean', 'median', 'most_frequent', 'constant']}, 'fill_value': {'type': 'int', 'description': 'fill_value is used to replace all occurrences of missing_values', 'default': None}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, witch is (0, 1)', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MinMaxScale model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'columns to be processed'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the StandardScale model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize the RobustScale instance with feature names. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'List of feature names to be scaled.'}}, 'required': ['features']}}, 'fit': {'description': 'Compute the median and IQR for scaling. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Scale features using the previously computed median and IQR. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features to be scaled.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'A new dataframe with scaled features.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'OneHotEncode': {'type': 'class', 'description': 'Apply one-hot encoding to specified categorical columns, the original columns will be dropped.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Categorical columns to be one-hot encoded and dropped'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the OneHotEncoding model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': {'df': {'type': 'DataFrame', 'description': 'The transformed DataFrame.'}}}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Instantiate the FillMissingValue class for numeric columns with missing values\ncolumns_with_missing_values = ['b']\nfill_missing_value = FillMissingValue(features=columns_with_missing_values, strategy='mean')\n\n# Apply the fit_transform method to the DataFrame to fill missing values\ndf = fill_missing_value.fit_transform(df)\n" - }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n ## User Requirement\\n read a dataset test.csv and print its head\\n ## Current Plan\\n [\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n }\\n ]\\n ## Current Task\\n {\\\"task_id\\\": \\\"1\\\", \\\"dependent_task_ids\\\": [], \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\", \\\"task_type\\\": \\\"\\\", \\\"code\\\": \\\"\\\", \\\"result\\\": \\\"\\\", \\\"is_finished\\\": false}\\n \"}, {\"role\": \"assistant\", \"content\": \"import pandas as pd\\ndata = pd.read_excel('test.csv')\\ndata\"}, {\"role\": \"user\", \"content\": \"\\n Traceback (most recent call last):\\n File \\\"\\\", line 2, in \\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 478, in read_excel\\n io = ExcelFile(io, storage_options=storage_options, engine=engine)\\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 1500, in __init__\\n raise ValueError(\\n ValueError: Excel file format cannot be determined, you must specify an engine manually.\\n \"}]": { - "language": "python", - "code": "import pandas as pd\ndata = pd.read_csv('test.csv')" - }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n ## User Requirement\\n read a dataset test.csv and print its head\\n ## Current Plan\\n [\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"import pandas as pd\\ndata = pd.read_csv('test.csv')\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n }\\n ]\\n ## Current Task\\n {\\\"task_id\\\": \\\"2\\\", \\\"dependent_task_ids\\\": [\\\"1\\\"], \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\", \\\"task_type\\\": \\\"\\\", \\\"code\\\": \\\"\\\", \\\"result\\\": \\\"\\\", \\\"is_finished\\\": false}\\n \"}]": { - "language": "python", - "code": "print(data.head())" - }, - "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nuser: Solve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\nTested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { - "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not handle negative numbers correctly, as the binary representation of negative numbers in Python includes a minus sign, which is not considered in the count_ones function. To fix this issue, we should modify the count_ones function to handle negative numbers appropriately by ensuring that it only counts the '1's in the binary representation of the absolute value of the number. This will ensure that the function returns the correct output for both positive and negative numbers.", - "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(abs(n)).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\n \n return sorted_arr" - }, "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a {teaching_language} Teacher, named Lily, your goal is writing a {language} teaching plan part by part. the constraint is writing in {language}. \nStatement: Write the \"Teaching Hours\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Hours\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n\n## Teaching Hours\n\n本教学单元共包括 4 课时,每课时 45 分钟。\n\n### 课时安排\n\n- 第一课时:1a 和 1b 部分\n- 第二课时:1c 和 2a 部分\n- 第三课时:2b 和 3a 部分\n- 第四课时:3b 和 3c 部分\n\n[TEACHING_PLAN_END]", "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a {teaching_language} Teacher, named Lily, your goal is writing a {language} teaching plan part by part. the constraint is writing in {language}. \nStatement: Write the \"Teaching Objectives\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Objectives\"!!\n\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n\n## Teaching Objectives\n\n1. Students will be able to listen and understand the names of different people in English.\n2. Students will be able to introduce themselves using the correct structure \"I'm [name]\".\n3. Students will be able to engage in simple conversational exchanges using greetings and introductions.\n4. Students will be able to recognize and match big and small letters in the English alphabet.\n\n[TEACHING_PLAN_END]", "Do not refer to the context of the previous conversation records, start the conversation anew.\n\nFormation: \"Capacity and role\" defines the role you are currently playing;\n\t\"[LESSON_BEGIN]\" and \"[LESSON_END]\" tags enclose the content of textbook;\n\t\"Statement\" defines the work detail you need to complete at this stage;\n\t\"Answer options\" defines the format requirements for your responses;\n\t\"Constraint\" defines the conditions that your responses must comply with.\n\nCapacity and role: You are a {teaching_language} Teacher, named Lily, your goal is writing a {language} teaching plan part by part. the constraint is writing in {language}. \nStatement: Write the \"Teaching Content\" part of teaching plan, WITHOUT ANY content unrelated to \"Teaching Content\"!!\nStatement: \"Teaching Content\" must include vocabulary, analysis, and examples of various grammar structures that appear in the textbook, as well as the listening materials and key points.\nStatement: \"Teaching Content\" must include more examples.\nAnswer options: Enclose the teaching plan content with \"[TEACHING_PLAN_BEGIN]\" and \"[TEACHING_PLAN_END]\" tags.\nAnswer options: Using proper markdown format from second-level header format.\nConstraint: Writing in Chinese.\n[LESSON_BEGIN]\n\n UNIT 1 Making New Friends\n TOPIC 1 Welcome to China!\n Section A\n\n 1a Listen and number the following names.\n Jane Mari Kangkang Michael\n Look, listen and understand. Then practice the conversation.\n Work in groups. Introduce yourself using\n I ’m ... Then practice 1a\n with your own hometown or the following places.\n\n 1b Listen and number the following names\n Jane Michael Maria Kangkang\n 1c Work in groups. Introduce yourself using I ’m ... Then practice 1a with your own hometown or the following places.\n China the USA the UK Hong Kong Beijing\n\n 2a Look, listen and understand. Then practice the conversation\n Hello! \n Hello! \n Hello! \n Hello! Are you Maria? \n No, I’m not. I’m Jane.\n Oh, nice to meet you, Jane\n Nice to meet you, too.\n Hi, Maria!\n Hi, Kangkang!\n Welcome to China!\n Thanks.\n\n 2b Work in groups. Make up a conversation with your own name and the\n following structures.\n A: Hello! / Good morning! / Hi! I’m ... Are you ... ?\n B: ...\n\n 3a Listen, say and trace\n Aa Bb Cc Dd Ee Ff Gg\n\n 3b Listen and number the following letters. Then circle the letters with the same sound as Bb.\n Aa Bb Cc Dd Ee Ff Gg\n\n 3c Match the big letters with the small ones. Then write them on the lines.\n \n[LESSON_END]": "[TEACHING_PLAN_BEGIN]\n\n## 教学内容\n\n### 词汇\n- 名字:Jane, Mari, Kangkang, Michael, Maria\n- 地点:China, the USA, the UK, Hong Kong, Beijing\n\n### 语法分析\n- 介绍自己的句式:I’m ...\n- 问候句式:Hello! / Good morning! / Hi! I’m ... Are you ... ?\n- 回答问候的句式:No, I’m not. I’m Jane. / Oh, nice to meet you, Jane. / Nice to meet you, too. / Hi, Maria! / Hi, Kangkang! / Welcome to China! / Thanks.\n\n### 例句\n- 例句1:Hello! Are you Maria? No, I’m not. I’m Jane.\n- 例句2:Hi, Maria! Hi, Kangkang! Welcome to China! Thanks.\n\n### 听力材料\n- 听力练习1a、1b、2a、3a\n\n### 关键点\n- 学生能够用英语介绍自己的名字和来自的地方\n- 学生能够用正确的问候方式和回答方式进行交流\n\n[TEACHING_PLAN_END]", @@ -376,13 +278,6 @@ "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 2:\nConsider a flat and stable surface, such as a grassy field or a gym mat.\n": "Choice 2: Considering a flat and stable surface, such as a grassy field or a gym mat, is the most promising option. This choice provides a safe and secure environment for the activity, reducing the risk of injury. A grassy field offers a natural and soft surface, while a gym mat provides cushioning and stability. Both options allow for comfortable movement and can accommodate various physical activities.\n\nThe best choice is 2.", "Given an instruction and several choices, decide which choice is most promising. Analyze each choice in detail, then conclude in the last line \"The best choice is {s}\", where s the integer id of the choice.\nChoice 4:\nPractice near a wall or with a spotter for safety.\n": "Choice 4: \n\nPractice near a wall or with a spotter for safety.\n\nAnalyzing this choice, practicing near a wall or with a spotter provides a safety measure to prevent accidents or injuries. When practicing a new skill or exercise, having a wall nearby can provide support and stability, reducing the risk of falling or losing balance. Similarly, having a spotter can offer assistance and guidance, ensuring that the practice is done safely and effectively.\n\nThe best choice is 4.", "### Requirements\n1. Please summarize the latest dialogue based on the reference information (secondary) and dialogue history (primary). Do not include text that is irrelevant to the conversation.\n- The context is for reference only. If it is irrelevant to the user's search request history, please reduce its reference and usage.\n2. If there are citable links in the context, annotate them in the main text in the format [main text](citation link). If there are none in the context, do not write links.\n3. The reply should be graceful, clear, non-repetitive, smoothly written, and of moderate length, in {LANG}.\n\n### Dialogue History (For example)\nA: MLOps competitors\n\n### Current Question (For example)\nA: MLOps competitors\n\n### Current Reply (For example)\n1. Alteryx Designer: etc. if any\n2. Matlab: ditto\n3. IBM SPSS Statistics\n4. RapidMiner Studio\n5. DataRobot AI Platform\n6. Databricks Lakehouse Platform\n7. Amazon SageMaker\n8. Dataiku\n#SYSTEM_MSG_END#\n### Reference Information\nABC cleanser is preferred by many with oily skin.\nL'Oreal is a popular brand with many positive reviews.\n\n### Dialogue History\n\nuser: Which facial cleanser is good for oily skin?\n\n### Current Question\nuser: Which facial cleanser is good for oily skin?\n\n### Current Reply: Based on the information, please write the reply to the Question\n\n\n": "Based on the information provided, ABC cleanser is preferred by many with oily skin. It is a popular choice for individuals with oily skin due to its effectiveness. Additionally, L'Oreal is a well-known brand with many positive reviews, and they offer a range of products suitable for oily skin. Both of these options could be good choices for individuals with oily skin.", - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nKeep dataset column information updated before model train.\\n## Done Tasks\\n```python\\n\\n```end\\n\\n# Task\\nUpdate and print the dataset's column information only if the train or test data has changed. Use the following code:\\n```python\\nfrom metagpt.tools.libs.data_preprocess import get_column_info\\n\\ncolumn_info = get_column_info(df)\\nprint(\\\"column_info\\\")\\nprint(column_info)\\n```end\\n\\n# Constraints:\\n- Use the DataFrame variable from 'Done Tasks' in place of df.\\n- Import `get_column_info` only if it's not already imported.\\n\"}]": { - "code": "from metagpt.tools.libs.data_preprocess import get_column_info\n\ncolumn_info = get_column_info(df)\nprint(\"column_info\")\nprint(column_info)" - }, - "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nSolve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\n[user: Tested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] rerquirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { - "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not correctly sort the array based on the number of ones in their binary representation in ascending order. The function `count_ones` correctly counts the number of ones in the binary representation of each number, but the sorting does not handle the case where two numbers have the same number of ones but different decimal values. To fix this issue, we need to ensure that the sorting is stable when the number of ones is the same, so that numbers are then sorted by their decimal value. We can use the `sorted` function with two keys, first sorting by the decimal value, and then by the number of ones, to ensure stability.", - "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(n).count('1')\n \n # First, sort the array by the decimal values to ensure stability\n arr.sort()\n # Then, sort the array using a custom key function\n # The key function returns the number of ones for each element\n # Since the previous sort was stable, elements with the same number of ones will remain sorted by their value\n sorted_arr = sorted(arr, key=count_ones)\n \n return sorted_arr\n" - }, "\n## context\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Required Python packages\": [\n \"flask==1.1.2\",\n \"bcrypt==3.2.0\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"openapi: 3.0.0 ...\",\n \"Shared Knowledge\": \"`game.py` contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on how to start and initialize third-party libraries.\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Required Python packages: typing.List[str] # Provide required Python packages in requirements.txt format.\n- Required Other language third-party packages: typing.List[str] # List down the required packages for languages other than Python.\n- Logic Analysis: typing.List[typing.List[str]] # Provide a list of files with the classes/methods/functions to be implemented, including dependency analysis and imports.\n- Task list: typing.List[str] # Break down the tasks into a list of filenames, prioritized by dependency order.\n- Full API spec: # Describe all APIs using OpenAPI 3.0 spec that may be used by both frontend and backend. If front-end and back-end communication is not required, leave it blank.\n- Shared Knowledge: # Detail any shared knowledge, like common utility functions or configuration variables.\n- Anything UNCLEAR: # Mention any unclear aspects in the project management context and try to clarify them.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Required Python packages\": [\n \"pygame==2.0.1\"\n ],\n \"Required Other language third-party packages\": [\n \"No third-party dependencies required\"\n ],\n \"Logic Analysis\": [\n [\n \"game.py\",\n \"Contains Game class and ... functions\"\n ],\n [\n \"main.py\",\n \"Contains main function, from game import Game\"\n ]\n ],\n \"Task list\": [\n \"game.py\",\n \"main.py\"\n ],\n \"Full API spec\": \"\",\n \"Shared Knowledge\": \"`game.py` contains functions shared across the project.\",\n \"Anything UNCLEAR\": \"Clarification needed on the scoring system and how the game over condition will be handled.\"\n}\n[/CONTENT]", "\nNOTICE\nRole: You are a professional engineer; the main goal is to write google-style, elegant, modular, easy to read and maintain code\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Task\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"`game.py` contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Legacy Code\n```Code\n\n```\n\n## Debug logs\n```text\n\n\n\n```\n\n## Bug Feedback logs\n```text\n\n```\n\n# Format example\n## Code: game.py\n```python\n## game.py\n...\n```\n\n# Instruction: Based on the context, follow \"Format example\", write code.\n\n## Code: game.py. Write code with triple quoto, based on the following attentions and context.\n1. Only One file: do your best to implement THIS ONLY ONE FILE.\n2. COMPLETE CODE: Your code will be part of the entire project, so please implement complete, reliable, reusable code snippets.\n3. Set default value: If there is any setting, ALWAYS SET A DEFAULT VALUE, ALWAYS USE STRONG TYPE AND EXPLICIT VARIABLE. AVOID circular import.\n4. Follow design: YOU MUST FOLLOW \"Data structures and interfaces\". DONT CHANGE ANY DESIGN. Do not use public member functions that do not exist in your design.\n5. CAREFULLY CHECK THAT YOU DONT MISS ANY NECESSARY CLASS/FUNCTION IN THIS FILE.\n6. Before using a external variable/module, make sure you import it first.\n7. Write out EVERY CODE DETAIL, DON'T LEAVE TODO.\n\n": "```python\n## game.py\n\nimport pygame\nimport random\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos]\n self.direction = direction\n self.length = length\n\n def move(self):\n head = self.body[0]\n x, y = head\n if self.direction == 'UP':\n new_head = (x, y - 10)\n elif self.direction == 'DOWN':\n new_head = (x, y + 10)\n elif self.direction == 'LEFT':\n new_head = (x - 10, y)\n elif self.direction == 'RIGHT':\n new_head = (x + 10, y)\n self.body = [new_head] + self.body[:-1]\n\n def change_direction(self, new_direction: str):\n if new_direction in ['UP', 'DOWN', 'LEFT', 'RIGHT']:\n self.direction = new_direction\n\n def grow(self):\n self.length += 1\n self.body.append((0, 0)) # Set default value\n\nclass Game:\n def __init__(self):\n self.snake = Snake((100, 50), 1, 'RIGHT') # Set default value\n self.food = (200, 200) # Set default value\n self.score = 0 # Set default value\n\n def update(self):\n self.snake.move()\n if self.snake.body[0] == self.food:\n self.snake.grow()\n self.food = (random.randrange(0, 400, 10), random.randrange(0, 400, 10)) # Set default value\n\n def draw(self):\n pass # Set default value\n\n def handle_input(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n pygame.quit()\n quit()\n if event.type == pygame.KEYDOWN:\n if event.key == pygame.K_UP:\n self.snake.change_direction('UP')\n elif event.key == pygame.K_DOWN:\n self.snake.change_direction('DOWN')\n elif event.key == pygame.K_LEFT:\n self.snake.change_direction('LEFT')\n elif event.key == pygame.K_RIGHT:\n self.snake.change_direction('RIGHT')\n\n def restart(self):\n self.snake = Snake((100, 50), 1, 'RIGHT') # Set default value\n self.food = (200, 200) # Set default value\n self.score = 0 # Set default value\n```", "\n# System\nRole: You are a professional software engineer, and your main task is to review and revise the code. You need to ensure that the code conforms to the google-style standards, is elegantly designed and modularized, easy to read and maintain.\nLanguage: Please use the same language as the user requirement, but the title and code should be still in English. For example, if the user speaks Chinese, the specific text of your answer should also be in Chinese.\nATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced \"Format example\".\n\n# Context\n## System Design\n{\"Implementation approach\":\"We will use the Pygame library to create the game and handle user input. The game logic will be implemented using Python classes and functions.\",\"File list\":[\"main.py\",\"game.py\"],\"Data structures and interfaces\":\"\\nclassDiagram\\n class Game {\\n -snake: list\\n -food: tuple\\n -score: int\\n +__init__()\\n +update()\\n +draw()\\n +handle_input()\\n +restart()\\n }\\n class Snake {\\n -body: list\\n -direction: str\\n +__init__(start_pos: tuple, length: int, direction: str)\\n +move()\\n +change_direction(new_direction: str)\\n +grow()\\n }\\n Game --> Snake\\n\",\"Program call flow\":\"\\nsequenceDiagram\\n participant M as Main\\n participant G as Game\\n M->>G: initialize game\\n G->>G: update game state\\n G->>G: draw game\\n G->>G: handle user input\\n G->>G: restart game\\n\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Task\n{\"Required Python packages\":[\"pygame==2.0.1\"],\"Required Other language third-party packages\":[\"No third-party dependencies required\"],\"Logic Analysis\":[[\"game.py\",\"Contains Game class and ... functions\"],[\"main.py\",\"Contains main function, from game import Game\"]],\"Task list\":[\"game.py\",\"main.py\"],\"Full API spec\":\"\",\"Shared Knowledge\":\"`game.py` contains functions shared across the project.\",\"Anything UNCLEAR\":\"Clarification needed on the scoring system and how the game over condition will be handled.\"}\n\n## Code Files\n\n\n\n## Code to be Reviewed: game.py\n```Code\n## game.py\n\nimport pygame\nimport random\n\nclass Snake:\n def __init__(self, start_pos: tuple, length: int, direction: str):\n self.body = [start_pos]\n self.direction = direction\n self.length = length\n\n def move(self):\n head = self.body[0]\n x, y = head\n if self.direction == 'UP':\n new_head = (x, y - 10)\n elif self.direction == 'DOWN':\n new_head = (x, y + 10)\n elif self.direction == 'LEFT':\n new_head = (x - 10, y)\n elif self.direction == 'RIGHT':\n new_head = (x + 10, y)\n self.body = [new_head] + self.body[:-1]\n\n def change_direction(self, new_direction: str):\n if new_direction in ['UP', 'DOWN', 'LEFT', 'RIGHT']:\n self.direction = new_direction\n\n def grow(self):\n self.length += 1\n self.body.append((0, 0)) # Set default value\n\nclass Game:\n def __init__(self):\n self.snake = Snake((100, 50), 1, 'RIGHT') # Set default value\n self.food = (200, 200) # Set default value\n self.score = 0 # Set default value\n\n def update(self):\n self.snake.move()\n if self.snake.body[0] == self.food:\n self.snake.grow()\n self.food = (random.randrange(0, 400, 10), random.randrange(0, 400, 10)) # Set default value\n\n def draw(self):\n pass # Set default value\n\n def handle_input(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n pygame.quit()\n quit()\n if event.type == pygame.KEYDOWN:\n if event.key == pygame.K_UP:\n self.snake.change_direction('UP')\n elif event.key == pygame.K_DOWN:\n self.snake.change_direction('DOWN')\n elif event.key == pygame.K_LEFT:\n self.snake.change_direction('LEFT')\n elif event.key == pygame.K_RIGHT:\n self.snake.change_direction('RIGHT')\n\n def restart(self):\n self.snake = Snake((100, 50), 1, 'RIGHT') # Set default value\n self.food = (200, 200) # Set default value\n self.score = 0 # Set default value\n\n```\n\n\n\n# Format example 1\n## Code Review: game.py\n1. No, we should fix the logic of class A due to ...\n2. ...\n3. ...\n4. No, function B is not implemented, ...\n5. ...\n6. ...\n\n## Actions\n1. Fix the `handle_events` method to update the game state only if a move is successful.\n ```python\n def handle_events(self):\n for event in pygame.event.get():\n if event.type == pygame.QUIT:\n return False\n if event.type == pygame.KEYDOWN:\n moved = False\n if event.key == pygame.K_UP:\n moved = self.game.move('UP')\n elif event.key == pygame.K_DOWN:\n moved = self.game.move('DOWN')\n elif event.key == pygame.K_LEFT:\n moved = self.game.move('LEFT')\n elif event.key == pygame.K_RIGHT:\n moved = self.game.move('RIGHT')\n if moved:\n # Update the game state only if a move was successful\n self.render()\n return True\n ```\n2. Implement function B\n\n## Code Review Result\nLBTM\n\n# Format example 2\n## Code Review: game.py\n1. Yes.\n2. Yes.\n3. Yes.\n4. Yes.\n5. Yes.\n6. Yes.\n\n## Actions\npass\n\n## Code Review Result\nLGTM\n\n\n\n# Instruction: Based on the actual code situation, follow one of the \"Format example\". Return only 1 file under review.\n\n## Code Review: Ordered List. Based on the \"Code to be Reviewed\", provide key, clear, concise, and specific answer. If any answer is no, explain how to fix it step by step.\n1. Is the code implemented as per the requirements? If not, how to achieve it? Analyse it step by step.\n2. Is the code logic completely correct? If there are errors, please indicate how to correct them.\n3. Does the existing code follow the \"Data structures and interfaces\"?\n4. Are all functions implemented? If there is no implementation, please indicate how to achieve it step by step.\n5. Have all necessary pre-dependencies been imported? If not, indicate which ones need to be imported\n6. Are methods from other files being reused correctly?\n\n## Actions: Ordered List. Things that should be done after CR, such as implementing class A and function B\n\n## Code Review Result: str. If the code doesn't have bugs, we don't need to rewrite it, so answer LGTM and stop. ONLY ANSWER LGTM/LBTM.\nLGTM/LBTM\n\n": "## Code Review: game.py\n1. The code is implemented as per the requirements. The Snake and Game classes are defined, and the necessary methods for updating, drawing, handling input, and restarting the game are implemented.\n2. The code logic is correct. The Snake class methods for moving, changing direction, and growing the snake are implemented properly. The Game class methods for updating the game state, handling input, and restarting the game are also correctly implemented.\n3. The existing code follows the defined \"Data structures and interfaces\" with the Snake and Game classes and their respective methods.\n4. All necessary functions are implemented as per the requirements.\n5. The necessary pre-dependencies, including the pygame package, are imported.\n6. The methods from other files are not being reused in the current code.\n\n## Actions\nNo actions required.\n\n## Code Review Result\nLGTM", @@ -418,6 +313,42 @@ "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Product Goals\": [\n \"Create an engaging user experience\",\n \"Improve accessibility, be responsive\",\n \"More beautiful UI\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Product Goals: typing.List[str] # Provide up to three clear, orthogonal product goals.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Product Goals\": [\n \"Create a challenging and addictive gameplay\",\n \"Enhance accessibility and responsiveness for all users\",\n \"Implement visually appealing UI design\"\n ]\n}\n[/CONTENT]", "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Competitive Analysis: typing.List[str] # Provide 5 to 7 competitive products.\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Competitive Analysis\": [\n \"2048 Game A: Simple interface, lacks responsive features\",\n \"play2048.co: Beautiful and responsive UI with my best score shown\",\n \"2048game.com: Responsive UI with my best score shown, but many ads\"\n ]\n}\n[/CONTENT]", "\n## context\nCreate a 2048 game\n\n-----\n\n## format example\n[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code ...\"\n ],\n [\n \"P0\",\n \"The game algorithm ...\"\n ]\n ]\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- Requirement Pool: typing.List[typing.List[str]] # List down the top-5 requirements with their priority (P0, P1, P2).\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"Requirement Pool\": [\n [\n \"P0\",\n \"The main code should handle user input and update the game board accordingly.\"\n ],\n [\n \"P0\",\n \"The game algorithm should handle the merging of tiles and the generation of new tiles with values of 2 or 4.\"\n ],\n [\n \"P1\",\n \"The game should end when the player achieves the 2048 tile or when there are no possible moves left.\"\n ],\n [\n \"P1\",\n \"The game should display the current score and the highest tile achieved by the player.\"\n ],\n [\n \"P2\",\n \"The game should have a smooth and visually appealing user interface.\"\n ]\n ]\n}\n[/CONTENT]", + "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.txt\n## Development Code\n```python\nresult = 'helloworld'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\n\n## Running Output\nstandard output: \n```text\nhelloworld\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nNo errors detected. The development code runs successfully and outputs the expected result without any errors.\n## File To Rewrite:\nNone\n## Status:\nPASS\n## Send To:\nNoOne\n---", + "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.sh\n## Development Code\n```python\necho 'Hello World'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\necho Hello World\n## Running Output\nstandard output: \n```text\nHello World\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nNo errors detected in the running result. The development code executed successfully, and the output matches the expected result. Since there is no test code provided, no test execution was performed. Therefore, no specific instructions are needed for correction.\n## File To Rewrite:\nNone\n## Status:\nPASS\n## Send To:\nNoOne\n---", + "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.py\n## Development Code\n```python\npython -c \"print(1/0)\"\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\npython -c print(1/0)\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\nTraceback (most recent call last):\n File \"\", line 1, in \nZeroDivisionError: division by zero\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nThe error is caused by attempting to divide by zero in the development code. To fix this error, you should modify the division operation to avoid division by zero. For example, you can add a condition to check if the denominator is zero before performing the division.\n\n## File To Rewrite:\na.py\n\n## Status:\nFAIL\n\n## Send To:\nEngineer\n---", + "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nuser: Solve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\nTested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { + "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not handle negative numbers correctly. The binary representation of negative numbers in Python uses two's complement, which is not what we want for this sorting task. To fix this issue, we should modify the `count_ones` function to only work with non-negative numbers and ignore negative numbers. We should also ensure that negative numbers are sorted based on their decimal value, as mentioned in the context.", + "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n # Only count ones for non-negative integers\n return bin(n).count('1') if n >= 0 else 0\n \n # Separate the array into non-negative and negative numbers\n non_negatives = [x for x in arr if x >= 0]\n negatives = [x for x in arr if x < 0]\n \n # Sort non-negative numbers using a custom key function\n sorted_non_negatives = sorted(non_negatives, key=lambda x: (count_ones(x), x))\n \n # Sort negative numbers based on their decimal value\n sorted_negatives = sorted(negatives)\n \n # Combine the sorted non-negative and negative numbers\n sorted_arr = sorted_non_negatives + sorted_negatives\n \n return sorted_arr" + }, + "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\n对数据集进行数据清洗\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Completing missing values with simple strategies.', 'MinMaxScale': 'Transform features by scaling each feature to a range, which is (0, 1).', 'StandardScale': 'Standardize features by removing the mean and scaling to unit variance.', 'MaxAbsScale': 'Scale each feature by its maximum absolute value.', 'RobustScale': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'OrdinalEncode': 'Encode categorical features as ordinal integers.', 'OneHotEncode': 'Apply one-hot encoding to specified categorical columns, the original columns will be dropped.', 'LabelEncode': 'Apply label encoding to specified categorical columns in-place.'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { + "recommend_tools": [ + "FillMissingValue", + "MinMaxScale", + "StandardScale", + "MaxAbsScale", + "RobustScale" + ] + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MinMaxScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the StandardScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MaxAbsScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize the RobustScale instance with feature names. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'List of feature names to be scaled.'}}, 'required': ['features']}}, 'fit': {'description': 'Compute the median and IQR for scaling. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Scale features using the previously computed median and IQR. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features to be scaled.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'A new dataframe with scaled features.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { + "code": "# Step 1: Fill missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create a copy of the dataframe to avoid modifying the original one\ndf_processed = df.copy()\n\n# Identify numeric columns to fill missing values with mean\nnumeric_cols = df_processed.select_dtypes(include='number').columns.tolist()\nfill_missing_value_numeric = FillMissingValue(features=numeric_cols, strategy='mean')\ndf_processed = fill_missing_value_numeric.fit_transform(df_processed)\n\n# Step 2: Scale numeric features\nfrom metagpt.tools.libs.data_preprocess import MinMaxScale\n\n# Initialize the MinMaxScaler and scale the numeric columns\nminmax_scaler = MinMaxScale(features=numeric_cols)\ndf_processed = minmax_scaler.fit_transform(df_processed)\n\n# Step 3: Encode categorical features\n# For simplicity, we'll use label encoding for this example\nfrom sklearn.preprocessing import LabelEncoder\n\nlabel_encoder = LabelEncoder()\ndf_processed['c'] = label_encoder.fit_transform(df_processed['c'])\n\n# The dataset is now preprocessed and ready for further analysis or modeling." + }, + "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nclean and preprocess the data\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Filling missing values', 'SplitBins': 'Bin continuous data into intervals and return the bin identifier encoded as an integer value'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { + "recommend_tools": [ + "FillMissingValue" + ] + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code_steps='', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code_steps='', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MinMaxScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the StandardScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MaxAbsScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize the RobustScale instance with feature names. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'List of feature names to be scaled.'}}, 'required': ['features']}}, 'fit': {'description': 'Compute the median and IQR for scaling. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Scale features using the previously computed median and IQR. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features to be scaled.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'A new dataframe with scaled features.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Fill missing values in column 'b' with the mean value\nfill_missing = FillMissingValue(features=['b'], strategy='mean')\ndf = fill_missing.fit_transform(df)\n\n# Check the DataFrame after filling missing values\ndf" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n ## User Requirement\\n read a dataset test.csv and print its head\\n ## Current Plan\\n [\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n }\\n ]\\n ## Current Task\\n {\\\"task_id\\\": \\\"1\\\", \\\"dependent_task_ids\\\": [], \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\", \\\"task_type\\\": \\\"\\\", \\\"code\\\": \\\"\\\", \\\"result\\\": \\\"\\\", \\\"is_finished\\\": false}\\n \"}, {\"role\": \"assistant\", \"content\": \"import pandas as pd\\ndata = pd.read_excel('test.csv')\\ndata\"}, {\"role\": \"user\", \"content\": \"\\n Traceback (most recent call last):\\n File \\\"\\\", line 2, in \\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 478, in read_excel\\n io = ExcelFile(io, storage_options=storage_options, engine=engine)\\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 1500, in __init__\\n raise ValueError(\\n ValueError: Excel file format cannot be determined, you must specify an engine manually.\\n \"}]": { + "language": "python", + "code": "import pandas as pd\ndata = pd.read_csv('test.csv')" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n ## User Requirement\\n read a dataset test.csv and print its head\\n ## Current Plan\\n [\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"import pandas as pd\\ndata = pd.read_csv('test.csv')\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n }\\n ]\\n ## Current Task\\n {\\\"task_id\\\": \\\"2\\\", \\\"dependent_task_ids\\\": [\\\"1\\\"], \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\", \\\"task_type\\\": \\\"\\\", \\\"code\\\": \\\"\\\", \\\"result\\\": \\\"\\\", \\\"is_finished\\\": false}\\n \"}]": { + "language": "python", + "code": "print(data.head())" + }, + "\n # Context:\n user: run analysis on sklearn iris dataset\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 5 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Import the Iris dataset from scikit-learn.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis to understand the dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Preprocess the data if necessary (e.g., scaling, encoding).\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Split the dataset into training and testing sets.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Choose a suitable model and train it on the dataset.\"\n },\n {\n \"task_id\": \"6\",\n \"dependent_task_ids\": [\"5\"],\n \"instruction\": \"Evaluate the model's performance on the test set.\"\n },\n {\n \"task_id\": \"7\",\n \"dependent_task_ids\": [\"6\"],\n \"instruction\": \"Report the results of the analysis.\"\n }\n]\n```", "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Import the Iris dataset from scikit-learn.\\nTask 2: Perform exploratory data analysis to understand the dataset.\\nTask 3: Preprocess the data if necessary (e.g., scaling, encoding).\\nTask 4: Split the dataset into training and testing sets.\\nTask 5: Choose a suitable model and train it on the dataset.\\nTask 6: Evaluate the model's performance on the test set.\\nTask 7: Report the results of the analysis.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { "task_type": [ "other", @@ -429,13 +360,28 @@ "other" ] }, + "\n # Context:\n user: \n## User Requirement\nRun data analysis on sklearn Iris dataset, include a plot\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Load the sklearn Iris dataset.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis on the Iris dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Create a plot visualizing the Iris dataset features.\"\n }\n]\n```", + "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Load the sklearn Iris dataset.\\nTask 2: Perform exploratory data analysis on the Iris dataset.\\nTask 3: Create a plot visualizing the Iris dataset features.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { + "task_type": [ + "other", + "eda", + "other" + ] + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"1\\\",\\\"dependent_task_ids\\\":[],\\\"instruction\\\":\\\"Load the sklearn Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from sklearn.datasets import load_iris\niris_data = load_iris()" + }, "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { "code": "import pandas as pd\n\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\niris_df['target'] = iris_data.target\n\n# Display basic information about the dataset\niris_df_info = iris_df.info()\n\n# Display statistical summary of the dataset\niris_df_description = iris_df.describe()\n\n# Show the first few rows of the dataset\ndf_head = iris_df.head()\n\n# Output the results\ndisplay(iris_df_info, iris_df_description, df_head)" }, "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\n\\\\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\\\\niris_df['target'] = iris_data.target\\\\n\\\\n# Display basic information about the dataset\\\\niris_df_info = iris_df.info()\\\\n\\\\n# Display statistical summary of the dataset\\\\niris_df_description = iris_df.describe()\\\\n\\\\n# Show the first few rows of the dataset\\\\ndf_head = iris_df.head()\\\\n\\\\n# Output the results\\\\ndisplay(iris_df_info, iris_df_description, df_head)\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset features.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { "code": "import seaborn as sns\nimport matplotlib.pyplot as plt\n\n# Pairplot to visualize the relationships between features\nsns.pairplot(iris_df, hue='target', diag_kind='kde')\n\n# Show the plot\nplt.show()" }, - "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.txt\n## Development Code\n```python\nresult = 'helloworld'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\n\n## Running Output\nstandard output: \n```text\nhelloworld\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nNo errors detected. The development code runs successfully and outputs the expected result without any errors.\n## File To Rewrite:\nNone\n## Status:\nPASS\n## Send To:\nNoOne\n---", - "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.sh\n## Development Code\n```python\necho 'Hello World'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\necho Hello World\n## Running Output\nstandard output: \n```text\nHello World\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nNo errors detected in the running result. The development code executed successfully, and the output matches the expected result. Since there is no test code provided, no test execution was performed. Therefore, no specific instructions are needed for correction.\n## File To Rewrite:\nNone\n## Status:\nPASS\n## Send To:\nNoOne\n---", - "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.py\n## Development Code\n```python\npython -c \"print(1/0)\"\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\npython -c print(1/0)\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\nTraceback (most recent call last):\n File \"\", line 1, in \nZeroDivisionError: division by zero\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nThe error is caused by attempting to divide by zero in the development code. To fix this error, you should modify the division operation to avoid division by zero. For example, you can add a condition to check if the denominator is zero before performing the division.\n\n## File To Rewrite:\na.py\n\n## Status:\nFAIL\n\n## Send To:\nEngineer\n---" + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n\\n```end\\n\\n## Current Task\\nPerform exploratory data analysis on the train dataset to understand the features and target variable.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { + "code": "# Step 1: Load the training dataset\nimport pandas as pd\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\n\n# Step 2: Display the first few rows of the dataset\ndisplay(train_data.head())\n\n# Step 3: Get a summary of the dataset\ntrain_data.info()\n\n# Step 4: Describe the numerical features\ntrain_data.describe()\n\n# Step 5: Describe the categorical features\ntrain_data.describe(include=['O'])\n\n# Step 6: Check for missing values\nmissing_values = train_data.isnull().sum()\nmissing_values[missing_values > 0]\n\n# Step 7: Visualize the distribution of the target variable 'Survived'\nimport seaborn as sns\nimport matplotlib.pyplot as plt\nsns.countplot(x='Survived', data=train_data)\nplt.title('Distribution of Survival')\nplt.show()\n\n# Step 8: Explore the relationship between features and the target variable\n# Visualize the survival rate by sex\nsns.barplot(x='Sex', y='Survived', data=train_data)\nplt.title('Survival Rate by Sex')\nplt.show()\n\n# Visualize the survival rate by class\nsns.barplot(x='Pclass', y='Survived', data=train_data)\nplt.title('Survival Rate by Class')\nplt.show()\n\n# Visualize the survival rate by embarkation port\nsns.barplot(x='Embarked', y='Survived', data=train_data)\nplt.title('Survival Rate by Embarkation Port')\nplt.show()" + }, + "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nSolve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\n[user: Tested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { + "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not correctly sort the array based on the number of ones in their binary representation followed by their decimal value. The error in the output indicates that the numbers 3 and 4 are not in the expected positions. The binary representations of 3 and 4 are '11' and '100', respectively, which means 3 has more ones than 4 and should come before 4 in the sorted array. To fix this issue, we need to ensure that the sorting is stable so that when two numbers have the same number of ones, they retain their original order if their decimal values are the same. We can achieve this by using the 'stable' sorting algorithm provided by Python's sorted function. Additionally, we need to handle negative numbers correctly by taking the absolute value before counting the ones in their binary representation.", + "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(abs(n)).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n # Using 'stable' sorting algorithm to maintain the original order when necessary\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x), reverse=False)\n \n return sorted_arr" + } } \ No newline at end of file From c6ac7ef8b8a076706acdd172cb88ae9923593874 Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 2 Feb 2024 20:52:56 +0800 Subject: [PATCH 589/637] fix web scrape init --- metagpt/tools/libs/web_scraping.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/tools/libs/web_scraping.py b/metagpt/tools/libs/web_scraping.py index b6db62d67..f983c1215 100644 --- a/metagpt/tools/libs/web_scraping.py +++ b/metagpt/tools/libs/web_scraping.py @@ -16,7 +16,7 @@ async def scrape_web_playwright(url, *urls): (dict): The inner text content and html structure of the web page, key are : 'inner_text', 'html'. """ # Create a PlaywrightWrapper instance for the Chromium browser - web = await PlaywrightWrapper("chromium").run(url, *urls) + web = await PlaywrightWrapper().run(url, *urls) # Return the inner text content of the web page return {"inner_text": web.inner_text.strip(), "html": web.html.strip()} From a758782d278cef39ba19944a29943b13bcdff6e0 Mon Sep 17 00:00:00 2001 From: voidking Date: Fri, 2 Feb 2024 22:43:41 +0800 Subject: [PATCH 590/637] bugfix: unittest to fulltest --- .github/workflows/fulltest.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/fulltest.yaml b/.github/workflows/fulltest.yaml index 68d3c382f..f5c6049e1 100644 --- a/.github/workflows/fulltest.yaml +++ b/.github/workflows/fulltest.yaml @@ -1,4 +1,4 @@ -name: Unit Tests +name: Full Tests on: workflow_dispatch: From 2a096ad3aa80070abd2ffeab197141e90c09281b Mon Sep 17 00:00:00 2001 From: yzlin Date: Fri, 2 Feb 2024 23:00:16 +0800 Subject: [PATCH 591/637] rm unnecessary test datasets --- tests/data/ml_datasets/titanic/split_eval.csv | 180 ----- .../data/ml_datasets/titanic/split_train.csv | 713 ------------------ 2 files changed, 893 deletions(-) delete mode 100644 tests/data/ml_datasets/titanic/split_eval.csv delete mode 100644 tests/data/ml_datasets/titanic/split_train.csv diff --git a/tests/data/ml_datasets/titanic/split_eval.csv b/tests/data/ml_datasets/titanic/split_eval.csv deleted file mode 100644 index 6da6ff6b3..000000000 --- a/tests/data/ml_datasets/titanic/split_eval.csv +++ /dev/null @@ -1,180 +0,0 @@ -PassengerId,Survived,Pclass,Name,Sex,Age,SibSp,Parch,Ticket,Fare,Cabin,Embarked -206,0,3,"Strom, Miss. Telma Matilda",female,2.0,0,1,347054,10.4625,G6,S -45,1,3,"Devaney, Miss. Margaret Delia",female,19.0,0,0,330958,7.8792,,Q -822,1,3,"Lulic, Mr. Nikola",male,27.0,0,0,315098,8.6625,,S -459,1,2,"Toomey, Miss. Ellen",female,50.0,0,0,F.C.C. 13531,10.5,,S -796,0,2,"Otter, Mr. Richard",male,39.0,0,0,28213,13.0,,S -119,0,1,"Baxter, Mr. Quigg Edmond",male,24.0,0,1,PC 17558,247.5208,B58 B60,C -425,0,3,"Rosblom, Mr. Viktor Richard",male,18.0,1,1,370129,20.2125,,S -679,0,3,"Goodwin, Mrs. Frederick (Augusta Tyler)",female,43.0,1,6,CA 2144,46.9,,S -270,1,1,"Bissette, Miss. Amelia",female,35.0,0,0,PC 17760,135.6333,C99,S -230,0,3,"Lefebre, Miss. Mathilde",female,,3,1,4133,25.4667,,S -690,1,1,"Madill, Miss. Georgette Alexandra",female,15.0,0,1,24160,211.3375,B5,S -321,0,3,"Dennis, Mr. Samuel",male,22.0,0,0,A/5 21172,7.25,,S -406,0,2,"Gale, Mr. Shadrach",male,34.0,1,0,28664,21.0,,S -41,0,3,"Ahlin, Mrs. Johan (Johanna Persdotter Larsson)",female,40.0,1,0,7546,9.475,,S -25,0,3,"Palsson, Miss. Torborg Danira",female,8.0,3,1,349909,21.075,,S -554,1,3,"Leeni, Mr. Fahim (""Philip Zenni"")",male,22.0,0,0,2620,7.225,,C -413,1,1,"Minahan, Miss. Daisy E",female,33.0,1,0,19928,90.0,C78,Q -513,1,1,"McGough, Mr. James Robert",male,36.0,0,0,PC 17473,26.2875,E25,S -756,1,2,"Hamalainen, Master. Viljo",male,0.67,1,1,250649,14.5,,S -392,1,3,"Jansson, Mr. Carl Olof",male,21.0,0,0,350034,7.7958,,S -602,0,3,"Slabenoff, Mr. Petco",male,,0,0,349214,7.8958,,S -326,1,1,"Young, Miss. Marie Grice",female,36.0,0,0,PC 17760,135.6333,C32,C -373,0,3,"Beavan, Mr. William Thomas",male,19.0,0,0,323951,8.05,,S -377,1,3,"Landergren, Miss. Aurora Adelia",female,22.0,0,0,C 7077,7.25,,S -201,0,3,"Vande Walle, Mr. Nestor Cyriel",male,28.0,0,0,345770,9.5,,S -512,0,3,"Webber, Mr. James",male,,0,0,SOTON/OQ 3101316,8.05,,S -601,1,2,"Jacobsohn, Mrs. Sidney Samuel (Amy Frances Christy)",female,24.0,2,1,243847,27.0,,S -631,1,1,"Barkworth, Mr. Algernon Henry Wilson",male,80.0,0,0,27042,30.0,A23,S -364,0,3,"Asim, Mr. Adola",male,35.0,0,0,SOTON/O.Q. 3101310,7.05,,S -144,0,3,"Burke, Mr. Jeremiah",male,19.0,0,0,365222,6.75,,Q -202,0,3,"Sage, Mr. Frederick",male,,8,2,CA. 2343,69.55,,S -134,1,2,"Weisz, Mrs. Leopold (Mathilde Francoise Pede)",female,29.0,1,0,228414,26.0,,S -431,1,1,"Bjornstrom-Steffansson, Mr. Mauritz Hakan",male,28.0,0,0,110564,26.55,C52,S -419,0,2,"Matthews, Mr. William John",male,30.0,0,0,28228,13.0,,S -782,1,1,"Dick, Mrs. Albert Adrian (Vera Gillespie)",female,17.0,1,0,17474,57.0,B20,S -705,0,3,"Hansen, Mr. Henrik Juul",male,26.0,1,0,350025,7.8542,,S -536,1,2,"Hart, Miss. Eva Miriam",female,7.0,0,2,F.C.C. 13529,26.25,,S -335,1,1,"Frauenthal, Mrs. Henry William (Clara Heinsheimer)",female,,1,0,PC 17611,133.65,,S -273,1,2,"Mellinger, Mrs. (Elizabeth Anne Maidment)",female,41.0,0,1,250644,19.5,,S -108,1,3,"Moss, Mr. Albert Johan",male,,0,0,312991,7.775,,S -403,0,3,"Jussila, Miss. Mari Aina",female,21.0,1,0,4137,9.825,,S -307,1,1,"Fleming, Miss. Margaret",female,,0,0,17421,110.8833,,C -218,0,2,"Jacobsohn, Mr. Sidney Samuel",male,42.0,1,0,243847,27.0,,S -789,1,3,"Dean, Master. Bertram Vere",male,1.0,1,2,C.A. 2315,20.575,,S -160,0,3,"Sage, Master. Thomas Henry",male,,8,2,CA. 2343,69.55,,S -20,1,3,"Masselmani, Mrs. Fatima",female,,0,0,2649,7.225,,C -174,0,3,"Sivola, Mr. Antti Wilhelm",male,21.0,0,0,STON/O 2. 3101280,7.925,,S -311,1,1,"Hays, Miss. Margaret Bechstein",female,24.0,0,0,11767,83.1583,C54,C -595,0,2,"Chapman, Mr. John Henry",male,37.0,1,0,SC/AH 29037,26.0,,S -592,1,1,"Stephenson, Mrs. Walter Bertram (Martha Eustis)",female,52.0,1,0,36947,78.2667,D20,C -164,0,3,"Calic, Mr. Jovo",male,17.0,0,0,315093,8.6625,,S -563,0,2,"Norman, Mr. Robert Douglas",male,28.0,0,0,218629,13.5,,S -172,0,3,"Rice, Master. Arthur",male,4.0,4,1,382652,29.125,,Q -871,0,3,"Balkic, Mr. Cerin",male,26.0,0,0,349248,7.8958,,S -176,0,3,"Klasen, Mr. Klas Albin",male,18.0,1,1,350404,7.8542,,S -434,0,3,"Kallio, Mr. Nikolai Erland",male,17.0,0,0,STON/O 2. 3101274,7.125,,S -462,0,3,"Morley, Mr. William",male,34.0,0,0,364506,8.05,,S -49,0,3,"Samaan, Mr. Youssef",male,,2,0,2662,21.6792,,C -126,1,3,"Nicola-Yarred, Master. Elias",male,12.0,1,0,2651,11.2417,,C -125,0,1,"White, Mr. Percival Wayland",male,54.0,0,1,35281,77.2875,D26,S -266,0,2,"Reeves, Mr. David",male,36.0,0,0,C.A. 17248,10.5,,S -550,1,2,"Davies, Master. John Morgan Jr",male,8.0,1,1,C.A. 33112,36.75,,S -589,0,3,"Gilinski, Mr. Eliezer",male,22.0,0,0,14973,8.05,,S -779,0,3,"Kilgannon, Mr. Thomas J",male,,0,0,36865,7.7375,,Q -179,0,2,"Hale, Mr. Reginald",male,30.0,0,0,250653,13.0,,S -107,1,3,"Salkjelsvik, Miss. Anna Kristine",female,21.0,0,0,343120,7.65,,S -624,0,3,"Hansen, Mr. Henry Damsgaard",male,21.0,0,0,350029,7.8542,,S -115,0,3,"Attalah, Miss. Malake",female,17.0,0,0,2627,14.4583,,C -42,0,2,"Turpin, Mrs. William John Robert (Dorothy Ann Wonnacott)",female,27.0,1,0,11668,21.0,,S -664,0,3,"Coleff, Mr. Peju",male,36.0,0,0,349210,7.4958,,S -661,1,1,"Frauenthal, Dr. Henry William",male,50.0,2,0,PC 17611,133.65,,S -762,0,3,"Nirva, Mr. Iisakki Antino Aijo",male,41.0,0,0,SOTON/O2 3101272,7.125,,S -580,1,3,"Jussila, Mr. Eiriik",male,32.0,0,0,STON/O 2. 3101286,7.925,,S -265,0,3,"Henry, Miss. Delia",female,,0,0,382649,7.75,,Q -757,0,3,"Carlsson, Mr. August Sigfrid",male,28.0,0,0,350042,7.7958,,S -666,0,2,"Hickman, Mr. Lewis",male,32.0,2,0,S.O.C. 14879,73.5,,S -634,0,1,"Parr, Mr. William Henry Marsh",male,,0,0,112052,0.0,,S -532,0,3,"Toufik, Mr. Nakli",male,,0,0,2641,7.2292,,C -640,0,3,"Thorneycroft, Mr. Percival",male,,1,0,376564,16.1,,S -599,0,3,"Boulos, Mr. Hanna",male,,0,0,2664,7.225,,C -220,0,2,"Harris, Mr. Walter",male,30.0,0,0,W/C 14208,10.5,,S -150,0,2,"Byles, Rev. Thomas Roussel Davids",male,42.0,0,0,244310,13.0,,S -269,1,1,"Graham, Mrs. William Thompson (Edith Junkins)",female,58.0,0,1,PC 17582,153.4625,C125,S -670,1,1,"Taylor, Mrs. Elmer Zebley (Juliet Cummins Wright)",female,,1,0,19996,52.0,C126,S -578,1,1,"Silvey, Mrs. William Baird (Alice Munger)",female,39.0,1,0,13507,55.9,E44,S -786,0,3,"Harmer, Mr. Abraham (David Lishin)",male,25.0,0,0,374887,7.25,,S -82,1,3,"Sheerlinck, Mr. Jan Baptist",male,29.0,0,0,345779,9.5,,S -400,1,2,"Trout, Mrs. William H (Jessie L)",female,28.0,0,0,240929,12.65,,S -135,0,2,"Sobey, Mr. Samuel James Hayden",male,25.0,0,0,C.A. 29178,13.0,,S -223,0,3,"Green, Mr. George Henry",male,51.0,0,0,21440,8.05,,S -693,1,3,"Lam, Mr. Ali",male,,0,0,1601,56.4958,,S -280,1,3,"Abbott, Mrs. Stanton (Rosa Hunt)",female,35.0,1,1,C.A. 2673,20.25,,S -102,0,3,"Petroff, Mr. Pastcho (""Pentcho"")",male,,0,0,349215,7.8958,,S -288,0,3,"Naidenoff, Mr. Penko",male,22.0,0,0,349206,7.8958,,S -711,1,1,"Mayne, Mlle. Berthe Antonine (""Mrs de Villiers"")",female,24.0,0,0,PC 17482,49.5042,C90,C -256,1,3,"Touma, Mrs. Darwis (Hanne Youssef Razi)",female,29.0,0,2,2650,15.2458,,C -23,1,3,"McGowan, Miss. Anna ""Annie""",female,15.0,0,0,330923,8.0292,,Q -582,1,1,"Thayer, Mrs. John Borland (Marian Longstreth Morris)",female,39.0,1,1,17421,110.8833,C68,C -564,0,3,"Simmons, Mr. John",male,,0,0,SOTON/OQ 392082,8.05,,S -405,0,3,"Oreskovic, Miss. Marija",female,20.0,0,0,315096,8.6625,,S -429,0,3,"Flynn, Mr. James",male,,0,0,364851,7.75,,Q -848,0,3,"Markoff, Mr. Marin",male,35.0,0,0,349213,7.8958,,C -726,0,3,"Oreskovic, Mr. Luka",male,20.0,0,0,315094,8.6625,,S -721,1,2,"Harper, Miss. Annie Jessie ""Nina""",female,6.0,0,1,248727,33.0,,S -637,0,3,"Leinonen, Mr. Antti Gustaf",male,32.0,0,0,STON/O 2. 3101292,7.925,,S -863,1,1,"Swift, Mrs. Frederick Joel (Margaret Welles Barron)",female,48.0,0,0,17466,25.9292,D17,S -615,0,3,"Brocklebank, Mr. William Alfred",male,35.0,0,0,364512,8.05,,S -199,1,3,"Madigan, Miss. Margaret ""Maggie""",female,,0,0,370370,7.75,,Q -787,1,3,"Sjoblom, Miss. Anna Sofia",female,18.0,0,0,3101265,7.4958,,S -156,0,1,"Williams, Mr. Charles Duane",male,51.0,0,1,PC 17597,61.3792,,C -190,0,3,"Turcin, Mr. Stjepan",male,36.0,0,0,349247,7.8958,,S -556,0,1,"Wright, Mr. George",male,62.0,0,0,113807,26.55,,S -890,1,1,"Behr, Mr. Karl Howell",male,26.0,0,0,111369,30.0,C148,C -827,0,3,"Lam, Mr. Len",male,,0,0,1601,56.4958,,S -534,1,3,"Peter, Mrs. Catherine (Catherine Rizk)",female,,0,2,2668,22.3583,,C -834,0,3,"Augustsson, Mr. Albert",male,23.0,0,0,347468,7.8542,,S -279,0,3,"Rice, Master. Eric",male,7.0,4,1,382652,29.125,,Q -189,0,3,"Bourke, Mr. John",male,40.0,1,1,364849,15.5,,Q -561,0,3,"Morrow, Mr. Thomas Rowan",male,,0,0,372622,7.75,,Q -375,0,3,"Palsson, Miss. Stina Viola",female,3.0,3,1,349909,21.075,,S -322,0,3,"Danoff, Mr. Yoto",male,27.0,0,0,349219,7.8958,,S -158,0,3,"Corn, Mr. Harry",male,30.0,0,0,SOTON/OQ 392090,8.05,,S -524,1,1,"Hippach, Mrs. Louis Albert (Ida Sophia Fischer)",female,44.0,0,1,111361,57.9792,B18,C -175,0,1,"Smith, Mr. James Clinch",male,56.0,0,0,17764,30.6958,A7,C -117,0,3,"Connors, Mr. Patrick",male,70.5,0,0,370369,7.75,,Q -810,1,1,"Chambers, Mrs. Norman Campbell (Bertha Griggs)",female,33.0,1,0,113806,53.1,E8,S -472,0,3,"Cacic, Mr. Luka",male,38.0,0,0,315089,8.6625,,S -228,0,3,"Lovell, Mr. John Hall (""Henry"")",male,20.5,0,0,A/5 21173,7.25,,S -330,1,1,"Hippach, Miss. Jean Gertrude",female,16.0,0,1,111361,57.9792,B18,C -147,1,3,"Andersson, Mr. August Edvard (""Wennerstrom"")",male,27.0,0,0,350043,7.7958,,S -98,1,1,"Greenfield, Mr. William Bertram",male,23.0,0,1,PC 17759,63.3583,D10 D12,C -493,0,1,"Molson, Mr. Harry Markland",male,55.0,0,0,113787,30.5,C30,S -73,0,2,"Hood, Mr. Ambrose Jr",male,21.0,0,0,S.O.C. 14879,73.5,,S -645,1,3,"Baclini, Miss. Eugenie",female,0.75,2,1,2666,19.2583,,C -303,0,3,"Johnson, Mr. William Cahoone Jr",male,19.0,0,0,LINE,0.0,,S -699,0,1,"Thayer, Mr. John Borland",male,49.0,1,1,17421,110.8833,C68,C -704,0,3,"Gallagher, Mr. Martin",male,25.0,0,0,36864,7.7417,,Q -639,0,3,"Panula, Mrs. Juha (Maria Emilia Ojala)",female,41.0,0,5,3101295,39.6875,,S -99,1,2,"Doling, Mrs. John T (Ada Julia Bone)",female,34.0,0,1,231919,23.0,,S -74,0,3,"Chronopoulos, Mr. Apostolos",male,26.0,1,0,2680,14.4542,,C -157,1,3,"Gilnagh, Miss. Katherine ""Katie""",female,16.0,0,0,35851,7.7333,,Q -475,0,3,"Strandberg, Miss. Ida Sofia",female,22.0,0,0,7553,9.8375,,S -240,0,2,"Hunt, Mr. George Henry",male,33.0,0,0,SCO/W 1585,12.275,,S -801,0,2,"Ponesell, Mr. Martin",male,34.0,0,0,250647,13.0,,S -829,1,3,"McCormack, Mr. Thomas Joseph",male,,0,0,367228,7.75,,Q -208,1,3,"Albimona, Mr. Nassef Cassem",male,26.0,0,0,2699,18.7875,,C -29,1,3,"O'Dwyer, Miss. Ellen ""Nellie""",female,,0,0,330959,7.8792,,Q -616,1,2,"Herman, Miss. Alice",female,24.0,1,2,220845,65.0,,S -309,0,2,"Abelson, Mr. Samuel",male,30.0,1,0,P/PP 3381,24.0,,C -382,1,3,"Nakid, Miss. Maria (""Mary"")",female,1.0,0,2,2653,15.7417,,C -703,0,3,"Barbara, Miss. Saiide",female,18.0,0,1,2691,14.4542,,C -623,1,3,"Nakid, Mr. Sahid",male,20.0,1,1,2653,15.7417,,C -26,1,3,"Asplund, Mrs. Carl Oscar (Selma Augusta Emilia Johansson)",female,38.0,1,5,347077,31.3875,,S -519,1,2,"Angle, Mrs. William A (Florence ""Mary"" Agnes Hughes)",female,36.0,1,0,226875,26.0,,S -638,0,2,"Collyer, Mr. Harvey",male,31.0,1,1,C.A. 31921,26.25,,S -360,1,3,"Mockler, Miss. Helen Mary ""Ellie""",female,,0,0,330980,7.8792,,Q -736,0,3,"Williams, Mr. Leslie",male,28.5,0,0,54636,16.1,,S -101,0,3,"Petranec, Miss. Matilda",female,28.0,0,0,349245,7.8958,,S -165,0,3,"Panula, Master. Eino Viljami",male,1.0,4,1,3101295,39.6875,,S -591,0,3,"Rintamaki, Mr. Matti",male,35.0,0,0,STON/O 2. 3101273,7.125,,S -11,1,3,"Sandstrom, Miss. Marguerite Rut",female,4.0,1,1,PP 9549,16.7,G6,S -217,1,3,"Honkanen, Miss. Eliina",female,27.0,0,0,STON/O2. 3101283,7.925,,S -734,0,2,"Berriman, Mr. William John",male,23.0,0,0,28425,13.0,,S -385,0,3,"Plotcharsky, Mr. Vasil",male,,0,0,349227,7.8958,,S -854,1,1,"Lines, Miss. Mary Conover",female,16.0,0,1,PC 17592,39.4,D28,S -860,0,3,"Razi, Mr. Raihed",male,,0,0,2629,7.2292,,C -359,1,3,"McGovern, Miss. Mary",female,,0,0,330931,7.8792,,Q -448,1,1,"Seward, Mr. Frederic Kimber",male,34.0,0,0,113794,26.55,,S -214,0,2,"Givard, Mr. Hans Kristensen",male,30.0,0,0,250646,13.0,,S -652,1,2,"Doling, Miss. Elsie",female,18.0,0,1,231919,23.0,,S -192,0,2,"Carbines, Mr. William",male,19.0,0,0,28424,13.0,,S -57,1,2,"Rugg, Miss. Emily",female,21.0,0,0,C.A. 31026,10.5,,S -868,0,1,"Roebling, Mr. Washington Augustus II",male,31.0,0,0,PC 17590,50.4958,A24,S -531,1,2,"Quick, Miss. Phyllis May",female,2.0,1,1,26360,26.0,,S -248,1,2,"Hamalainen, Mrs. William (Anna)",female,24.0,0,2,250649,14.5,,S -260,1,2,"Parrish, Mrs. (Lutie Davis)",female,50.0,0,1,230433,26.0,,S -354,0,3,"Arnold-Franchi, Mr. Josef",male,25.0,1,0,349237,17.8,,S -784,0,3,"Johnston, Mr. Andrew G",male,,1,2,W./C. 6607,23.45,,S -853,0,3,"Boulos, Miss. Nourelain",female,9.0,1,1,2678,15.2458,,C diff --git a/tests/data/ml_datasets/titanic/split_train.csv b/tests/data/ml_datasets/titanic/split_train.csv deleted file mode 100644 index a48680208..000000000 --- a/tests/data/ml_datasets/titanic/split_train.csv +++ /dev/null @@ -1,713 +0,0 @@ -PassengerId,Survived,Pclass,Name,Sex,Age,SibSp,Parch,Ticket,Fare,Cabin,Embarked -409,0,3,"Birkeland, Mr. Hans Martin Monsen",male,21.0,0,0,312992,7.775,,S -481,0,3,"Goodwin, Master. Harold Victor",male,9.0,5,2,CA 2144,46.9,,S -511,1,3,"Daly, Mr. Eugene Patrick",male,29.0,0,0,382651,7.75,,Q -610,1,1,"Shutes, Miss. Elizabeth W",female,40.0,0,0,PC 17582,153.4625,C125,S -548,1,2,"Padro y Manent, Mr. Julian",male,,0,0,SC/PARIS 2146,13.8625,,C -710,1,3,"Moubarek, Master. Halim Gonios (""William George"")",male,,1,1,2661,15.2458,,C -153,0,3,"Meo, Mr. Alfonzo",male,55.5,0,0,A.5. 11206,8.05,,S -494,0,1,"Artagaveytia, Mr. Ramon",male,71.0,0,0,PC 17609,49.5042,,C -393,0,3,"Gustafsson, Mr. Johan Birger",male,28.0,2,0,3101277,7.925,,S -824,1,3,"Moor, Mrs. (Beila)",female,27.0,0,1,392096,12.475,E121,S -577,1,2,"Garside, Miss. Ethel",female,34.0,0,0,243880,13.0,,S -773,0,2,"Mack, Mrs. (Mary)",female,57.0,0,0,S.O./P.P. 3,10.5,E77,S -745,1,3,"Stranden, Mr. Juho",male,31.0,0,0,STON/O 2. 3101288,7.925,,S -328,1,2,"Ball, Mrs. (Ada E Hall)",female,36.0,0,0,28551,13.0,D,S -460,0,3,"O'Connor, Mr. Maurice",male,,0,0,371060,7.75,,Q -222,0,2,"Bracken, Mr. James H",male,27.0,0,0,220367,13.0,,S -851,0,3,"Andersson, Master. Sigvard Harald Elias",male,4.0,4,2,347082,31.275,,S -558,0,1,"Robbins, Mr. Victor",male,,0,0,PC 17757,227.525,,C -47,0,3,"Lennon, Mr. Denis",male,,1,0,370371,15.5,,Q -449,1,3,"Baclini, Miss. Marie Catherine",female,5.0,2,1,2666,19.2583,,C -371,1,1,"Harder, Mr. George Achilles",male,25.0,1,0,11765,55.4417,E50,C -196,1,1,"Lurette, Miss. Elise",female,58.0,0,0,PC 17569,146.5208,B80,C -761,0,3,"Garfirth, Mr. John",male,,0,0,358585,14.5,,S -55,0,1,"Ostby, Mr. Engelhart Cornelius",male,65.0,0,1,113509,61.9792,B30,C -573,1,1,"Flynn, Mr. John Irwin (""Irving"")",male,36.0,0,0,PC 17474,26.3875,E25,S -379,0,3,"Betros, Mr. Tannous",male,20.0,0,0,2648,4.0125,,C -198,0,3,"Olsen, Mr. Karl Siegwart Andreas",male,42.0,0,1,4579,8.4042,,S -396,0,3,"Johansson, Mr. Erik",male,22.0,0,0,350052,7.7958,,S -111,0,1,"Porter, Mr. Walter Chamberlain",male,47.0,0,0,110465,52.0,C110,S -138,0,1,"Futrelle, Mr. Jacques Heath",male,37.0,1,0,113803,53.1,C123,S -312,1,1,"Ryerson, Miss. Emily Borie",female,18.0,2,2,PC 17608,262.375,B57 B59 B63 B66,C -391,1,1,"Carter, Mr. William Ernest",male,36.0,1,2,113760,120.0,B96 B98,S -24,1,1,"Sloper, Mr. William Thompson",male,28.0,0,0,113788,35.5,A6,S -818,0,2,"Mallet, Mr. Albert",male,31.0,1,1,S.C./PARIS 2079,37.0042,,C -110,1,3,"Moran, Miss. Bertha",female,,1,0,371110,24.15,,Q -302,1,3,"McCoy, Mr. Bernard",male,,2,0,367226,23.25,,Q -104,0,3,"Johansson, Mr. Gustaf Joel",male,33.0,0,0,7540,8.6542,,S -875,1,2,"Abelson, Mrs. Samuel (Hannah Wizosky)",female,28.0,1,0,P/PP 3381,24.0,,C -62,1,1,"Icard, Miss. Amelie",female,38.0,0,0,113572,80.0,B28, -154,0,3,"van Billiard, Mr. Austin Blyler",male,40.5,0,2,A/5. 851,14.5,,S -289,1,2,"Hosono, Mr. Masabumi",male,42.0,0,0,237798,13.0,,S -245,0,3,"Attalah, Mr. Sleiman",male,30.0,0,0,2694,7.225,,C -681,0,3,"Peters, Miss. Katie",female,,0,0,330935,8.1375,,Q -797,1,1,"Leader, Dr. Alice (Farnham)",female,49.0,0,0,17465,25.9292,D17,S -226,0,3,"Berglund, Mr. Karl Ivar Sven",male,22.0,0,0,PP 4348,9.35,,S -857,1,1,"Wick, Mrs. George Dennick (Mary Hitchcock)",female,45.0,1,1,36928,164.8667,,S -621,0,3,"Yasbeck, Mr. Antoni",male,27.0,1,0,2659,14.4542,,C -451,0,2,"West, Mr. Edwy Arthur",male,36.0,1,2,C.A. 34651,27.75,,S -424,0,3,"Danbom, Mrs. Ernst Gilbert (Anna Sigrid Maria Brogren)",female,28.0,1,1,347080,14.4,,S -450,1,1,"Peuchen, Major. Arthur Godfrey",male,52.0,0,0,113786,30.5,C104,S -161,0,3,"Cribb, Mr. John Hatfield",male,44.0,0,1,371362,16.1,,S -743,1,1,"Ryerson, Miss. Susan Parker ""Suzette""",female,21.0,2,2,PC 17608,262.375,B57 B59 B63 B66,C -651,0,3,"Mitkoff, Mr. Mito",male,,0,0,349221,7.8958,,S -250,0,2,"Carter, Rev. Ernest Courtenay",male,54.0,1,0,244252,26.0,,S -540,1,1,"Frolicher, Miss. Hedwig Margaritha",female,22.0,0,2,13568,49.5,B39,C -414,0,2,"Cunningham, Mr. Alfred Fleming",male,,0,0,239853,0.0,,S -207,0,3,"Backstrom, Mr. Karl Alfred",male,32.0,1,0,3101278,15.85,,S -828,1,2,"Mallet, Master. Andre",male,1.0,0,2,S.C./PARIS 2079,37.0042,,C -484,1,3,"Turkula, Mrs. (Hedwig)",female,63.0,0,0,4134,9.5875,,S -607,0,3,"Karaic, Mr. Milan",male,30.0,0,0,349246,7.8958,,S -185,1,3,"Kink-Heilmann, Miss. Luise Gretchen",female,4.0,0,2,315153,22.025,,S -683,0,3,"Olsvigen, Mr. Thor Anderson",male,20.0,0,0,6563,9.225,,S -794,0,1,"Hoyt, Mr. William Fisher",male,,0,0,PC 17600,30.6958,,C -13,0,3,"Saundercock, Mr. William Henry",male,20.0,0,0,A/5. 2151,8.05,,S -118,0,2,"Turpin, Mr. William John Robert",male,29.0,1,0,11668,21.0,,S -483,0,3,"Rouse, Mr. Richard Henry",male,50.0,0,0,A/5 3594,8.05,,S -421,0,3,"Gheorgheff, Mr. Stanio",male,,0,0,349254,7.8958,,C -543,0,3,"Andersson, Miss. Sigrid Elisabeth",female,11.0,4,2,347082,31.275,,S -884,0,2,"Banfield, Mr. Frederick James",male,28.0,0,0,C.A./SOTON 34068,10.5,,S -877,0,3,"Gustafsson, Mr. Alfred Ossian",male,20.0,0,0,7534,9.8458,,S -109,0,3,"Rekic, Mr. Tido",male,38.0,0,0,349249,7.8958,,S -603,0,1,"Harrington, Mr. Charles H",male,,0,0,113796,42.4,,S -575,0,3,"Rush, Mr. Alfred George John",male,16.0,0,0,A/4. 20589,8.05,,S -253,0,1,"Stead, Mr. William Thomas",male,62.0,0,0,113514,26.55,C87,S -712,0,1,"Klaber, Mr. Herman",male,,0,0,113028,26.55,C124,S -397,0,3,"Olsson, Miss. Elina",female,31.0,0,0,350407,7.8542,,S -194,1,2,"Navratil, Master. Michel M",male,3.0,1,1,230080,26.0,F2,S -567,0,3,"Stoytcheff, Mr. Ilia",male,19.0,0,0,349205,7.8958,,S -204,0,3,"Youseff, Mr. Gerious",male,45.5,0,0,2628,7.225,,C -491,0,3,"Hagland, Mr. Konrad Mathias Reiersen",male,,1,0,65304,19.9667,,S -815,0,3,"Tomlin, Mr. Ernest Portage",male,30.5,0,0,364499,8.05,,S -219,1,1,"Bazzani, Miss. Albina",female,32.0,0,0,11813,76.2917,D15,C -446,1,1,"Dodge, Master. Washington",male,4.0,0,2,33638,81.8583,A34,S -490,1,3,"Coutts, Master. Eden Leslie ""Neville""",male,9.0,1,1,C.A. 37671,15.9,,S -112,0,3,"Zabour, Miss. Hileni",female,14.5,1,0,2665,14.4542,,C -731,1,1,"Allen, Miss. Elisabeth Walton",female,29.0,0,0,24160,211.3375,B5,S -106,0,3,"Mionoff, Mr. Stoytcho",male,28.0,0,0,349207,7.8958,,S -480,1,3,"Hirvonen, Miss. Hildur E",female,2.0,0,1,3101298,12.2875,,S -278,0,2,"Parkes, Mr. Francis ""Frank""",male,,0,0,239853,0.0,,S -70,0,3,"Kink, Mr. Vincenz",male,26.0,2,0,315151,8.6625,,S -86,1,3,"Backstrom, Mrs. Karl Alfred (Maria Mathilda Gustafsson)",female,33.0,3,0,3101278,15.85,,S -795,0,3,"Dantcheff, Mr. Ristiu",male,25.0,0,0,349203,7.8958,,S -162,1,2,"Watt, Mrs. James (Elizabeth ""Bessie"" Inglis Milne)",female,40.0,0,0,C.A. 33595,15.75,,S -816,0,1,"Fry, Mr. Richard",male,,0,0,112058,0.0,B102,S -517,1,2,"Lemore, Mrs. (Amelia Milley)",female,34.0,0,0,C.A. 34260,10.5,F33,S -300,1,1,"Baxter, Mrs. James (Helene DeLaudeniere Chaput)",female,50.0,0,1,PC 17558,247.5208,B58 B60,C -455,0,3,"Peduzzi, Mr. Joseph",male,,0,0,A/5 2817,8.05,,S -60,0,3,"Goodwin, Master. William Frederick",male,11.0,5,2,CA 2144,46.9,,S -880,1,1,"Potter, Mrs. Thomas Jr (Lily Alexenia Wilson)",female,56.0,0,1,11767,83.1583,C50,C -43,0,3,"Kraeff, Mr. Theodor",male,,0,0,349253,7.8958,,C -500,0,3,"Svensson, Mr. Olof",male,24.0,0,0,350035,7.7958,,S -236,0,3,"Harknett, Miss. Alice Phoebe",female,,0,0,W./C. 6609,7.55,,S -255,0,3,"Rosblom, Mrs. Viktor (Helena Wilhelmina)",female,41.0,0,2,370129,20.2125,,S -346,1,2,"Brown, Miss. Amelia ""Mildred""",female,24.0,0,0,248733,13.0,F33,S -105,0,3,"Gustafsson, Mr. Anders Vilhelm",male,37.0,2,0,3101276,7.925,,S -316,1,3,"Nilsson, Miss. Helmina Josefina",female,26.0,0,0,347470,7.8542,,S -873,0,1,"Carlsson, Mr. Frans Olof",male,33.0,0,0,695,5.0,B51 B53 B55,S -4,1,1,"Futrelle, Mrs. Jacques Heath (Lily May Peel)",female,35.0,1,0,113803,53.1,C123,S -805,1,3,"Hedman, Mr. Oskar Arvid",male,27.0,0,0,347089,6.975,,S -225,1,1,"Hoyt, Mr. Frederick Maxfield",male,38.0,1,0,19943,90.0,C93,S -772,0,3,"Jensen, Mr. Niels Peder",male,48.0,0,0,350047,7.8542,,S -539,0,3,"Risien, Mr. Samuel Beard",male,,0,0,364498,14.5,,S -249,1,1,"Beckwith, Mr. Richard Leonard",male,37.0,1,1,11751,52.5542,D35,S -32,1,1,"Spencer, Mrs. William Augustus (Marie Eugenie)",female,,1,0,PC 17569,146.5208,B78,C -268,1,3,"Persson, Mr. Ernst Ulrik",male,25.0,1,0,347083,7.775,,S -544,1,2,"Beane, Mr. Edward",male,32.0,1,0,2908,26.0,,S -685,0,2,"Brown, Mr. Thomas William Solomon",male,60.0,1,1,29750,39.0,,S -608,1,1,"Daniel, Mr. Robert Williams",male,27.0,0,0,113804,30.5,,S -749,0,1,"Marvin, Mr. Daniel Warner",male,19.0,1,0,113773,53.1,D30,S -234,1,3,"Asplund, Miss. Lillian Gertrud",female,5.0,4,2,347077,31.3875,,S -641,0,3,"Jensen, Mr. Hans Peder",male,20.0,0,0,350050,7.8542,,S -707,1,2,"Kelly, Mrs. Florence ""Fannie""",female,45.0,0,0,223596,13.5,,S -611,0,3,"Andersson, Mrs. Anders Johan (Alfrida Konstantia Brogren)",female,39.0,1,5,347082,31.275,,S -647,0,3,"Cor, Mr. Liudevit",male,19.0,0,0,349231,7.8958,,S -148,0,3,"Ford, Miss. Robina Maggie ""Ruby""",female,9.0,2,2,W./C. 6608,34.375,,S -574,1,3,"Kelly, Miss. Mary",female,,0,0,14312,7.75,,Q -809,0,2,"Meyer, Mr. August",male,39.0,0,0,248723,13.0,,S -535,0,3,"Cacic, Miss. Marija",female,30.0,0,0,315084,8.6625,,S -588,1,1,"Frolicher-Stehli, Mr. Maxmillian",male,60.0,1,1,13567,79.2,B41,C -331,1,3,"McCoy, Miss. Agnes",female,,2,0,367226,23.25,,Q -569,0,3,"Doharr, Mr. Tannous",male,,0,0,2686,7.2292,,C -725,1,1,"Chambers, Mr. Norman Campbell",male,27.0,1,0,113806,53.1,E8,S -100,0,2,"Kantor, Mr. Sinai",male,34.0,1,0,244367,26.0,,S -708,1,1,"Calderhead, Mr. Edward Pennington",male,42.0,0,0,PC 17476,26.2875,E24,S -277,0,3,"Lindblom, Miss. Augusta Charlotta",female,45.0,0,0,347073,7.75,,S -418,1,2,"Silven, Miss. Lyyli Karoliina",female,18.0,0,2,250652,13.0,,S -463,0,1,"Gee, Mr. Arthur H",male,47.0,0,0,111320,38.5,E63,S -665,1,3,"Lindqvist, Mr. Eino William",male,20.0,1,0,STON/O 2. 3101285,7.925,,S -718,1,2,"Troutt, Miss. Edwina Celia ""Winnie""",female,27.0,0,0,34218,10.5,E101,S -850,1,1,"Goldenberg, Mrs. Samuel L (Edwiga Grabowska)",female,,1,0,17453,89.1042,C92,C -516,0,1,"Walker, Mr. William Anderson",male,47.0,0,0,36967,34.0208,D46,S -633,1,1,"Stahelin-Maeglin, Dr. Max",male,32.0,0,0,13214,30.5,B50,C -538,1,1,"LeRoy, Miss. Bertha",female,30.0,0,0,PC 17761,106.425,,C -151,0,2,"Bateman, Rev. Robert James",male,51.0,0,0,S.O.P. 1166,12.525,,S -79,1,2,"Caldwell, Master. Alden Gates",male,0.83,0,2,248738,29.0,,S -10,1,2,"Nasser, Mrs. Nicholas (Adele Achem)",female,14.0,1,0,237736,30.0708,,C -143,1,3,"Hakkarainen, Mrs. Pekka Pietari (Elin Matilda Dolck)",female,24.0,1,0,STON/O2. 3101279,15.85,,S -76,0,3,"Moen, Mr. Sigurd Hansen",male,25.0,0,0,348123,7.65,F G73,S -254,0,3,"Lobb, Mr. William Arthur",male,30.0,1,0,A/5. 3336,16.1,,S -30,0,3,"Todoroff, Mr. Lalio",male,,0,0,349216,7.8958,,S -170,0,3,"Ling, Mr. Lee",male,28.0,0,0,1601,56.4958,,S -747,0,3,"Abbott, Mr. Rossmore Edward",male,16.0,1,1,C.A. 2673,20.25,,S -212,1,2,"Cameron, Miss. Clear Annie",female,35.0,0,0,F.C.C. 13528,21.0,,S -636,1,2,"Davis, Miss. Mary",female,28.0,0,0,237668,13.0,,S -689,0,3,"Fischer, Mr. Eberhard Thelander",male,18.0,0,0,350036,7.7958,,S -600,1,1,"Duff Gordon, Sir. Cosmo Edmund (""Mr Morgan"")",male,49.0,1,0,PC 17485,56.9292,A20,C -423,0,3,"Zimmerman, Mr. Leo",male,29.0,0,0,315082,7.875,,S -59,1,2,"West, Miss. Constance Mirium",female,5.0,1,2,C.A. 34651,27.75,,S -504,0,3,"Laitinen, Miss. Kristina Sofia",female,37.0,0,0,4135,9.5875,,S -352,0,1,"Williams-Lambert, Mr. Fletcher Fellows",male,,0,0,113510,35.0,C128,S -542,0,3,"Andersson, Miss. Ingeborg Constanzia",female,9.0,4,2,347082,31.275,,S -89,1,1,"Fortune, Miss. Mabel Helen",female,23.0,3,2,19950,263.0,C23 C25 C27,S -433,1,2,"Louch, Mrs. Charles Alexander (Alice Adelaide Slow)",female,42.0,1,0,SC/AH 3085,26.0,,S -566,0,3,"Davies, Mr. Alfred J",male,24.0,2,0,A/4 48871,24.15,,S -502,0,3,"Canavan, Miss. Mary",female,21.0,0,0,364846,7.75,,Q -128,1,3,"Madsen, Mr. Fridtjof Arne",male,24.0,0,0,C 17369,7.1417,,S -688,0,3,"Dakic, Mr. Branko",male,19.0,0,0,349228,10.1708,,S -329,1,3,"Goldsmith, Mrs. Frank John (Emily Alice Brown)",female,31.0,1,1,363291,20.525,,S -845,0,3,"Culumovic, Mr. Jeso",male,17.0,0,0,315090,8.6625,,S -886,0,3,"Rice, Mrs. William (Margaret Norton)",female,39.0,0,5,382652,29.125,,Q -581,1,2,"Christy, Miss. Julie Rachel",female,25.0,1,1,237789,30.0,,S -568,0,3,"Palsson, Mrs. Nils (Alma Cornelia Berglund)",female,29.0,0,4,349909,21.075,,S -152,1,1,"Pears, Mrs. Thomas (Edith Wearne)",female,22.0,1,0,113776,66.6,C2,S -342,1,1,"Fortune, Miss. Alice Elizabeth",female,24.0,3,2,19950,263.0,C23 C25 C27,S -272,1,3,"Tornquist, Mr. William Henry",male,25.0,0,0,LINE,0.0,,S -737,0,3,"Ford, Mrs. Edward (Margaret Ann Watson)",female,48.0,1,3,W./C. 6608,34.375,,S -700,0,3,"Humblen, Mr. Adolf Mathias Nicolai Olsen",male,42.0,0,0,348121,7.65,F G63,S -291,1,1,"Barber, Miss. Ellen ""Nellie""",female,26.0,0,0,19877,78.85,,S -141,0,3,"Boulos, Mrs. Joseph (Sultana)",female,,0,2,2678,15.2458,,C -261,0,3,"Smith, Mr. Thomas",male,,0,0,384461,7.75,,Q -163,0,3,"Bengtsson, Mr. John Viktor",male,26.0,0,0,347068,7.775,,S -232,0,3,"Larsson, Mr. Bengt Edvin",male,29.0,0,0,347067,7.775,,S -802,1,2,"Collyer, Mrs. Harvey (Charlotte Annie Tate)",female,31.0,1,1,C.A. 31921,26.25,,S -844,0,3,"Lemberopolous, Mr. Peter L",male,34.5,0,0,2683,6.4375,,C -691,1,1,"Dick, Mr. Albert Adrian",male,31.0,1,0,17474,57.0,B20,S -649,0,3,"Willey, Mr. Edward",male,,0,0,S.O./P.P. 751,7.55,,S -137,1,1,"Newsom, Miss. Helen Monypeny",female,19.0,0,2,11752,26.2833,D47,S -570,1,3,"Jonsson, Mr. Carl",male,32.0,0,0,350417,7.8542,,S -862,0,2,"Giles, Mr. Frederick Edward",male,21.0,1,0,28134,11.5,,S -445,1,3,"Johannesen-Bratthammer, Mr. Bernt",male,,0,0,65306,8.1125,,S -697,0,3,"Kelly, Mr. James",male,44.0,0,0,363592,8.05,,S -674,1,2,"Wilhelms, Mr. Charles",male,31.0,0,0,244270,13.0,,S -748,1,2,"Sinkkonen, Miss. Anna",female,30.0,0,0,250648,13.0,,S -367,1,1,"Warren, Mrs. Frank Manley (Anna Sophia Atkinson)",female,60.0,1,0,110813,75.25,D37,C -626,0,1,"Sutton, Mr. Frederick",male,61.0,0,0,36963,32.3208,D50,S -741,1,1,"Hawksford, Mr. Walter James",male,,0,0,16988,30.0,D45,S -821,1,1,"Hays, Mrs. Charles Melville (Clara Jennings Gregg)",female,52.0,1,1,12749,93.5,B69,S -282,0,3,"Olsson, Mr. Nils Johan Goransson",male,28.0,0,0,347464,7.8542,,S -546,0,1,"Nicholson, Mr. Arthur Ernest",male,64.0,0,0,693,26.0,,S -237,0,2,"Hold, Mr. Stephen",male,44.0,1,0,26707,26.0,,S -16,1,2,"Hewlett, Mrs. (Mary D Kingcome) ",female,55.0,0,0,248706,16.0,,S -565,0,3,"Meanwell, Miss. (Marion Ogden)",female,,0,0,SOTON/O.Q. 392087,8.05,,S -798,1,3,"Osman, Mrs. Mara",female,31.0,0,0,349244,8.6833,,S -740,0,3,"Nankoff, Mr. Minko",male,,0,0,349218,7.8958,,S -549,0,3,"Goldsmith, Mr. Frank John",male,33.0,1,1,363291,20.525,,S -663,0,1,"Colley, Mr. Edward Pomeroy",male,47.0,0,0,5727,25.5875,E58,S -482,0,2,"Frost, Mr. Anthony Wood ""Archie""",male,,0,0,239854,0.0,,S -113,0,3,"Barton, Mr. David John",male,22.0,0,0,324669,8.05,,S -458,1,1,"Kenyon, Mrs. Frederick R (Marion)",female,,1,0,17464,51.8625,D21,S -842,0,2,"Mudd, Mr. Thomas Charles",male,16.0,0,0,S.O./P.P. 3,10.5,,S -518,0,3,"Ryan, Mr. Patrick",male,,0,0,371110,24.15,,Q -553,0,3,"O'Brien, Mr. Timothy",male,,0,0,330979,7.8292,,Q -388,1,2,"Buss, Miss. Kate",female,36.0,0,0,27849,13.0,,S -514,1,1,"Rothschild, Mrs. Martin (Elizabeth L. Barrett)",female,54.0,1,0,PC 17603,59.4,,C -560,1,3,"de Messemaeker, Mrs. Guillaume Joseph (Emma)",female,36.0,1,0,345572,17.4,,S -701,1,1,"Astor, Mrs. John Jacob (Madeleine Talmadge Force)",female,18.0,1,0,PC 17757,227.525,C62 C64,C -241,0,3,"Zabour, Miss. Thamine",female,,1,0,2665,14.4542,,C -428,1,2,"Phillips, Miss. Kate Florence (""Mrs Kate Louise Phillips Marshall"")",female,19.0,0,0,250655,26.0,,S -593,0,3,"Elsbury, Mr. William James",male,47.0,0,0,A/5 3902,7.25,,S -116,0,3,"Pekoniemi, Mr. Edvard",male,21.0,0,0,STON/O 2. 3101294,7.925,,S -686,0,2,"Laroche, Mr. Joseph Philippe Lemercier",male,25.0,1,2,SC/Paris 2123,41.5792,,C -155,0,3,"Olsen, Mr. Ole Martin",male,,0,0,Fa 265302,7.3125,,S -308,1,1,"Penasco y Castellana, Mrs. Victor de Satode (Maria Josefa Perez de Soto y Vallejo)",female,17.0,1,0,PC 17758,108.9,C65,C -765,0,3,"Eklund, Mr. Hans Linus",male,16.0,0,0,347074,7.775,,S -597,1,2,"Leitch, Miss. Jessie Wills",female,,0,0,248727,33.0,,S -242,1,3,"Murphy, Miss. Katherine ""Kate""",female,,1,0,367230,15.5,,Q -823,0,1,"Reuchlin, Jonkheer. John George",male,38.0,0,0,19972,0.0,,S -380,0,3,"Gustafsson, Mr. Karl Gideon",male,19.0,0,0,347069,7.775,,S -336,0,3,"Denkoff, Mr. Mitto",male,,0,0,349225,7.8958,,S -488,0,1,"Kent, Mr. Edward Austin",male,58.0,0,0,11771,29.7,B37,C -672,0,1,"Davidson, Mr. Thornton",male,31.0,1,0,F.C. 12750,52.0,B71,S -791,0,3,"Keane, Mr. Andrew ""Andy""",male,,0,0,12460,7.75,,Q -340,0,1,"Blackwell, Mr. Stephen Weart",male,45.0,0,0,113784,35.5,T,S -879,0,3,"Laleff, Mr. Kristo",male,,0,0,349217,7.8958,,S -464,0,2,"Milling, Mr. Jacob Christian",male,48.0,0,0,234360,13.0,,S -717,1,1,"Endres, Miss. Caroline Louise",female,38.0,0,0,PC 17757,227.525,C45,C -343,0,2,"Collander, Mr. Erik Gustaf",male,28.0,0,0,248740,13.0,,S -276,1,1,"Andrews, Miss. Kornelia Theodosia",female,63.0,1,0,13502,77.9583,D7,S -530,0,2,"Hocking, Mr. Richard George",male,23.0,2,1,29104,11.5,,S -861,0,3,"Hansen, Mr. Claus Peter",male,41.0,2,0,350026,14.1083,,S -8,0,3,"Palsson, Master. Gosta Leonard",male,2.0,3,1,349909,21.075,,S -841,0,3,"Alhomaki, Mr. Ilmari Rudolf",male,20.0,0,0,SOTON/O2 3101287,7.925,,S -231,1,1,"Harris, Mrs. Henry Birkhardt (Irene Wallach)",female,35.0,1,0,36973,83.475,C83,S -338,1,1,"Burns, Miss. Elizabeth Margaret",female,41.0,0,0,16966,134.5,E40,C -286,0,3,"Stankovic, Mr. Ivan",male,33.0,0,0,349239,8.6625,,C -381,1,1,"Bidois, Miss. Rosalie",female,42.0,0,0,PC 17757,227.525,,C -468,0,1,"Smart, Mr. John Montgomery",male,56.0,0,0,113792,26.55,,S -838,0,3,"Sirota, Mr. Maurice",male,,0,0,392092,8.05,,S -742,0,1,"Cavendish, Mr. Tyrell William",male,36.0,1,0,19877,78.85,C46,S -617,0,3,"Danbom, Mr. Ernst Gilbert",male,34.0,1,1,347080,14.4,,S -485,1,1,"Bishop, Mr. Dickinson H",male,25.0,1,0,11967,91.0792,B49,C -437,0,3,"Ford, Miss. Doolina Margaret ""Daisy""",female,21.0,2,2,W./C. 6608,34.375,,S -885,0,3,"Sutehall, Mr. Henry Jr",male,25.0,0,0,SOTON/OQ 392076,7.05,,S -28,0,1,"Fortune, Mr. Charles Alexander",male,19.0,3,2,19950,263.0,C23 C25 C27,S -751,1,2,"Wells, Miss. Joan",female,4.0,1,1,29103,23.0,,S -97,0,1,"Goldschmidt, Mr. George B",male,71.0,0,0,PC 17754,34.6542,A5,C -6,0,3,"Moran, Mr. James",male,,0,0,330877,8.4583,,Q -271,0,1,"Cairns, Mr. Alexander",male,,0,0,113798,31.0,,S -301,1,3,"Kelly, Miss. Anna Katherine ""Annie Kate""",female,,0,0,9234,7.75,,Q -366,0,3,"Adahl, Mr. Mauritz Nils Martin",male,30.0,0,0,C 7076,7.25,,S -200,0,2,"Yrois, Miss. Henriette (""Mrs Harbeck"")",female,24.0,0,0,248747,13.0,,S -776,0,3,"Myhrman, Mr. Pehr Fabian Oliver Malkolm",male,18.0,0,0,347078,7.75,,S -178,0,1,"Isham, Miss. Ann Elizabeth",female,50.0,0,0,PC 17595,28.7125,C49,C -728,1,3,"Mannion, Miss. Margareth",female,,0,0,36866,7.7375,,Q -167,1,1,"Chibnall, Mrs. (Edith Martha Bowerman)",female,,0,1,113505,55.0,E33,S -869,0,3,"van Melkebeke, Mr. Philemon",male,,0,0,345777,9.5,,S -313,0,2,"Lahtinen, Mrs. William (Anna Sylfven)",female,26.0,1,1,250651,26.0,,S -285,0,1,"Smith, Mr. Richard William",male,,0,0,113056,26.0,A19,S -495,0,3,"Stanley, Mr. Edward Roland",male,21.0,0,0,A/4 45380,8.05,,S -33,1,3,"Glynn, Miss. Mary Agatha",female,,0,0,335677,7.75,,Q -417,1,2,"Drew, Mrs. James Vivian (Lulu Thorne Christian)",female,34.0,1,1,28220,32.5,,S -887,0,2,"Montvila, Rev. Juozas",male,27.0,0,0,211536,13.0,,S -559,1,1,"Taussig, Mrs. Emil (Tillie Mandelbaum)",female,39.0,1,1,110413,79.65,E67,S -806,0,3,"Johansson, Mr. Karl Johan",male,31.0,0,0,347063,7.775,,S -294,0,3,"Haas, Miss. Aloisia",female,24.0,0,0,349236,8.85,,S -209,1,3,"Carr, Miss. Helen ""Ellen""",female,16.0,0,0,367231,7.75,,Q -85,1,2,"Ilett, Miss. Bertha",female,17.0,0,0,SO/C 14885,10.5,,S -38,0,3,"Cann, Mr. Ernest Charles",male,21.0,0,0,A./5. 2152,8.05,,S -7,0,1,"McCarthy, Mr. Timothy J",male,54.0,0,0,17463,51.8625,E46,S -426,0,3,"Wiseman, Mr. Phillippe",male,,0,0,A/4. 34244,7.25,,S -790,0,1,"Guggenheim, Mr. Benjamin",male,46.0,0,0,PC 17593,79.2,B82 B84,C -389,0,3,"Sadlier, Mr. Matthew",male,,0,0,367655,7.7292,,Q -258,1,1,"Cherry, Miss. Gladys",female,30.0,0,0,110152,86.5,B77,S -643,0,3,"Skoog, Miss. Margit Elizabeth",female,2.0,3,2,347088,27.9,,S -355,0,3,"Yousif, Mr. Wazli",male,,0,0,2647,7.225,,C -830,1,1,"Stone, Mrs. George Nelson (Martha Evelyn)",female,62.0,0,0,113572,80.0,B28, -781,1,3,"Ayoub, Miss. Banoura",female,13.0,0,0,2687,7.2292,,C -267,0,3,"Panula, Mr. Ernesti Arvid",male,16.0,4,1,3101295,39.6875,,S -506,0,1,"Penasco y Castellana, Mr. Victor de Satode",male,18.0,1,0,PC 17758,108.9,C65,C -52,0,3,"Nosworthy, Mr. Richard Cater",male,21.0,0,0,A/4. 39886,7.8,,S -401,1,3,"Niskanen, Mr. Juha",male,39.0,0,0,STON/O 2. 3101289,7.925,,S -533,0,3,"Elias, Mr. Joseph Jr",male,17.0,1,1,2690,7.2292,,C -283,0,3,"de Pelsmaeker, Mr. Alfons",male,16.0,0,0,345778,9.5,,S -442,0,3,"Hampe, Mr. Leon",male,20.0,0,0,345769,9.5,,S -361,0,3,"Skoog, Mr. Wilhelm",male,40.0,1,4,347088,27.9,,S -840,1,1,"Marechal, Mr. Pierre",male,,0,0,11774,29.7,C47,C -509,0,3,"Olsen, Mr. Henry Margido",male,28.0,0,0,C 4001,22.525,,S -121,0,2,"Hickman, Mr. Stanley George",male,21.0,2,0,S.O.C. 14879,73.5,,S -320,1,1,"Spedden, Mrs. Frederic Oakley (Margaretta Corning Stone)",female,40.0,1,1,16966,134.5,E34,C -858,1,1,"Daly, Mr. Peter Denis ",male,51.0,0,0,113055,26.55,E17,S -501,0,3,"Calic, Mr. Petar",male,17.0,0,0,315086,8.6625,,S -91,0,3,"Christmann, Mr. Emil",male,29.0,0,0,343276,8.05,,S -727,1,2,"Renouf, Mrs. Peter Henry (Lillian Jefferys)",female,30.0,3,0,31027,21.0,,S -671,1,2,"Brown, Mrs. Thomas William Solomon (Elizabeth Catherine Ford)",female,40.0,1,1,29750,39.0,,S -456,1,3,"Jalsevac, Mr. Ivan",male,29.0,0,0,349240,7.8958,,C -427,1,2,"Clarke, Mrs. Charles V (Ada Maria Winfield)",female,28.0,1,0,2003,26.0,,S -63,0,1,"Harris, Mr. Henry Birkhardt",male,45.0,1,0,36973,83.475,C83,S -51,0,3,"Panula, Master. Juha Niilo",male,7.0,4,1,3101295,39.6875,,S -454,1,1,"Goldenberg, Mr. Samuel L",male,49.0,1,0,17453,89.1042,C92,C -394,1,1,"Newell, Miss. Marjorie",female,23.0,1,0,35273,113.275,D36,C -188,1,1,"Romaine, Mr. Charles Hallace (""Mr C Rolmane"")",male,45.0,0,0,111428,26.55,,S -368,1,3,"Moussa, Mrs. (Mantoura Boulos)",female,,0,0,2626,7.2292,,C -759,0,3,"Theobald, Mr. Thomas Leonard",male,34.0,0,0,363294,8.05,,S -804,1,3,"Thomas, Master. Assad Alexander",male,0.42,0,1,2625,8.5167,,C -510,1,3,"Lang, Mr. Fang",male,26.0,0,0,1601,56.4958,,S -788,0,3,"Rice, Master. George Hugh",male,8.0,4,1,382652,29.125,,Q -298,0,1,"Allison, Miss. Helen Loraine",female,2.0,1,2,113781,151.55,C22 C26,S -92,0,3,"Andreasson, Mr. Paul Edvin",male,20.0,0,0,347466,7.8542,,S -754,0,3,"Jonkoff, Mr. Lalio",male,23.0,0,0,349204,7.8958,,S -547,1,2,"Beane, Mrs. Edward (Ethel Clarke)",female,19.0,1,0,2908,26.0,,S -492,0,3,"Windelov, Mr. Einar",male,21.0,0,0,SOTON/OQ 3101317,7.25,,S -2,1,1,"Cumings, Mrs. John Bradley (Florence Briggs Thayer)",female,38.0,1,0,PC 17599,71.2833,C85,C -777,0,3,"Tobin, Mr. Roger",male,,0,0,383121,7.75,F38,Q -473,1,2,"West, Mrs. Edwy Arthur (Ada Mary Worth)",female,33.0,1,2,C.A. 34651,27.75,,S -252,0,3,"Strom, Mrs. Wilhelm (Elna Matilda Persson)",female,29.0,1,1,347054,10.4625,G6,S -93,0,1,"Chaffee, Mr. Herbert Fuller",male,46.0,1,0,W.E.P. 5734,61.175,E31,S -635,0,3,"Skoog, Miss. Mabel",female,9.0,3,2,347088,27.9,,S -44,1,2,"Laroche, Miss. Simonne Marie Anne Andree",female,3.0,1,2,SC/Paris 2123,41.5792,,C -835,0,3,"Allum, Mr. Owen George",male,18.0,0,0,2223,8.3,,S -48,1,3,"O'Driscoll, Miss. Bridget",female,,0,0,14311,7.75,,Q -891,0,3,"Dooley, Mr. Patrick",male,32.0,0,0,370376,7.75,,Q -264,0,1,"Harrison, Mr. William",male,40.0,0,0,112059,0.0,B94,S -356,0,3,"Vanden Steen, Mr. Leo Peter",male,28.0,0,0,345783,9.5,,S -528,0,1,"Farthing, Mr. John",male,,0,0,PC 17483,221.7792,C95,S -339,1,3,"Dahl, Mr. Karl Edwart",male,45.0,0,0,7598,8.05,,S -780,1,1,"Robert, Mrs. Edward Scott (Elisabeth Walton McMillan)",female,43.0,0,1,24160,211.3375,B3,S -21,0,2,"Fynney, Mr. Joseph J",male,35.0,0,0,239865,26.0,,S -723,0,2,"Gillespie, Mr. William Henry",male,34.0,0,0,12233,13.0,,S -677,0,3,"Sawyer, Mr. Frederick Charles",male,24.5,0,0,342826,8.05,,S -349,1,3,"Coutts, Master. William Loch ""William""",male,3.0,1,1,C.A. 37671,15.9,,S -817,0,3,"Heininen, Miss. Wendla Maria",female,23.0,0,0,STON/O2. 3101290,7.925,,S -334,0,3,"Vander Planke, Mr. Leo Edmondus",male,16.0,2,0,345764,18.0,,S -470,1,3,"Baclini, Miss. Helene Barbara",female,0.75,2,1,2666,19.2583,,C -130,0,3,"Ekstrom, Mr. Johan",male,45.0,0,0,347061,6.975,,S -191,1,2,"Pinsky, Mrs. (Rosa)",female,32.0,0,0,234604,13.0,,S -760,1,1,"Rothes, the Countess. of (Lucy Noel Martha Dyer-Edwards)",female,33.0,0,0,110152,86.5,B77,S -520,0,3,"Pavlovic, Mr. Stefo",male,32.0,0,0,349242,7.8958,,S -67,1,2,"Nye, Mrs. (Elizabeth Ramell)",female,29.0,0,0,C.A. 29395,10.5,F33,S -487,1,1,"Hoyt, Mrs. Frederick Maxfield (Jane Anne Forby)",female,35.0,1,0,19943,90.0,C93,S -19,0,3,"Vander Planke, Mrs. Julius (Emelia Maria Vandemoortele)",female,31.0,1,0,345763,18.0,,S -702,1,1,"Silverthorne, Mr. Spencer Victor",male,35.0,0,0,PC 17475,26.2875,E24,S -826,0,3,"Flynn, Mr. John",male,,0,0,368323,6.95,,Q -333,0,1,"Graham, Mr. George Edward",male,38.0,0,1,PC 17582,153.4625,C91,S -855,0,2,"Carter, Mrs. Ernest Courtenay (Lilian Hughes)",female,44.0,1,0,244252,26.0,,S -441,1,2,"Hart, Mrs. Benjamin (Esther Ada Bloomfield)",female,45.0,1,1,F.C.C. 13529,26.25,,S -775,1,2,"Hocking, Mrs. Elizabeth (Eliza Needs)",female,54.0,1,3,29105,23.0,,S -675,0,2,"Watson, Mr. Ennis Hastings",male,,0,0,239856,0.0,,S -552,0,2,"Sharp, Mr. Percival James R",male,27.0,0,0,244358,26.0,,S -56,1,1,"Woolner, Mr. Hugh",male,,0,0,19947,35.5,C52,S -653,0,3,"Kalvik, Mr. Johannes Halvorsen",male,21.0,0,0,8475,8.4333,,S -849,0,2,"Harper, Rev. John",male,28.0,0,1,248727,33.0,,S -730,0,3,"Ilmakangas, Miss. Pieta Sofia",female,25.0,1,0,STON/O2. 3101271,7.925,,S -233,0,2,"Sjostedt, Mr. Ernst Adolf",male,59.0,0,0,237442,13.5,,S -660,0,1,"Newell, Mr. Arthur Webster",male,58.0,0,2,35273,113.275,D48,C -243,0,2,"Coleridge, Mr. Reginald Charles",male,29.0,0,0,W./C. 14263,10.5,,S -36,0,1,"Holverson, Mr. Alexander Oskar",male,42.0,1,0,113789,52.0,,S -541,1,1,"Crosby, Miss. Harriet R",female,36.0,0,2,WE/P 5735,71.0,B22,S -719,0,3,"McEvoy, Mr. Michael",male,,0,0,36568,15.5,,Q -752,1,3,"Moor, Master. Meier",male,6.0,0,1,392096,12.475,E121,S -888,1,1,"Graham, Miss. Margaret Edith",female,19.0,0,0,112053,30.0,B42,S -122,0,3,"Moore, Mr. Leonard Charles",male,,0,0,A4. 54510,8.05,,S -411,0,3,"Sdycoff, Mr. Todor",male,,0,0,349222,7.8958,,S -353,0,3,"Elias, Mr. Tannous",male,15.0,1,1,2695,7.2292,,C -34,0,2,"Wheadon, Mr. Edward H",male,66.0,0,0,C.A. 24579,10.5,,S -180,0,3,"Leonard, Mr. Lionel",male,36.0,0,0,LINE,0.0,,S -646,1,1,"Harper, Mr. Henry Sleeper",male,48.0,1,0,PC 17572,76.7292,D33,C -819,0,3,"Holm, Mr. John Fredrik Alexander",male,43.0,0,0,C 7075,6.45,,S -22,1,2,"Beesley, Mr. Lawrence",male,34.0,0,0,248698,13.0,D56,S -412,0,3,"Hart, Mr. Henry",male,,0,0,394140,6.8583,,Q -422,0,3,"Charters, Mr. David",male,21.0,0,0,A/5. 13032,7.7333,,Q -584,0,1,"Ross, Mr. John Hugo",male,36.0,0,0,13049,40.125,A10,C -729,0,2,"Bryhl, Mr. Kurt Arnold Gottfrid",male,25.0,1,0,236853,26.0,,S -813,0,2,"Slemen, Mr. Richard James",male,35.0,0,0,28206,10.5,,S -562,0,3,"Sivic, Mr. Husein",male,40.0,0,0,349251,7.8958,,S -332,0,1,"Partner, Mr. Austen",male,45.5,0,0,113043,28.5,C124,S -341,1,2,"Navratil, Master. Edmond Roger",male,2.0,1,1,230080,26.0,F2,S -247,0,3,"Lindahl, Miss. Agda Thorilda Viktoria",female,25.0,0,0,347071,7.775,,S -127,0,3,"McMahon, Mr. Martin",male,,0,0,370372,7.75,,Q -324,1,2,"Caldwell, Mrs. Albert Francis (Sylvia Mae Harbaugh)",female,22.0,1,1,248738,29.0,,S -398,0,2,"McKane, Mr. Peter David",male,46.0,0,0,28403,26.0,,S -46,0,3,"Rogers, Mr. William John",male,,0,0,S.C./A.4. 23567,8.05,,S -65,0,1,"Stewart, Mr. Albert A",male,,0,0,PC 17605,27.7208,,C -262,1,3,"Asplund, Master. Edvin Rojj Felix",male,3.0,4,2,347077,31.3875,,S -372,0,3,"Wiklund, Mr. Jakob Alfred",male,18.0,1,0,3101267,6.4958,,S -376,1,1,"Meyer, Mrs. Edgar Joseph (Leila Saks)",female,,1,0,PC 17604,82.1708,,C -676,0,3,"Edvardsson, Mr. Gustaf Hjalmar",male,18.0,0,0,349912,7.775,,S -471,0,3,"Keefe, Mr. Arthur",male,,0,0,323592,7.25,,S -210,1,1,"Blank, Mr. Henry",male,40.0,0,0,112277,31.0,A31,C -733,0,2,"Knight, Mr. Robert J",male,,0,0,239855,0.0,,S -81,0,3,"Waelens, Mr. Achille",male,22.0,0,0,345767,9.0,,S -609,1,2,"Laroche, Mrs. Joseph (Juliette Marie Louise Lafargue)",female,22.0,1,2,SC/Paris 2123,41.5792,,C -874,0,3,"Vander Cruyssen, Mr. Victor",male,47.0,0,0,345765,9.0,,S -435,0,1,"Silvey, Mr. William Baird",male,50.0,1,0,13507,55.9,E44,S -767,0,1,"Brewe, Dr. Arthur Jackson",male,,0,0,112379,39.6,,C -768,0,3,"Mangan, Miss. Mary",female,30.5,0,0,364850,7.75,,Q -168,0,3,"Skoog, Mrs. William (Anna Bernhardina Karlsson)",female,45.0,1,4,347088,27.9,,S -709,1,1,"Cleaver, Miss. Alice",female,22.0,0,0,113781,151.55,,S -327,0,3,"Nysveen, Mr. Johan Hansen",male,61.0,0,0,345364,6.2375,,S -843,1,1,"Serepeca, Miss. Augusta",female,30.0,0,0,113798,31.0,,C -211,0,3,"Ali, Mr. Ahmed",male,24.0,0,0,SOTON/O.Q. 3101311,7.05,,S -159,0,3,"Smiljanic, Mr. Mile",male,,0,0,315037,8.6625,,S -378,0,1,"Widener, Mr. Harry Elkins",male,27.0,0,2,113503,211.5,C82,C -778,1,3,"Emanuel, Miss. Virginia Ethel",female,5.0,0,0,364516,12.475,,S -457,0,1,"Millet, Mr. Francis Davis",male,65.0,0,0,13509,26.55,E38,S -769,0,3,"Moran, Mr. Daniel J",male,,1,0,371110,24.15,,Q -362,0,2,"del Carlo, Mr. Sebastiano",male,29.0,1,0,SC/PARIS 2167,27.7208,,C -655,0,3,"Hegarty, Miss. Hanora ""Nora""",female,18.0,0,0,365226,6.75,,Q -698,1,3,"Mullens, Miss. Katherine ""Katie""",female,,0,0,35852,7.7333,,Q -444,1,2,"Reynaldo, Ms. Encarnacion",female,28.0,0,0,230434,13.0,,S -203,0,3,"Johanson, Mr. Jakob Alfred",male,34.0,0,0,3101264,6.4958,,S -606,0,3,"Lindell, Mr. Edvard Bengtsson",male,36.0,1,0,349910,15.55,,S -673,0,2,"Mitchell, Mr. Henry Michael",male,70.0,0,0,C.A. 24580,10.5,,S -846,0,3,"Abbing, Mr. Anthony",male,42.0,0,0,C.A. 5547,7.55,,S -374,0,1,"Ringhini, Mr. Sante",male,22.0,0,0,PC 17760,135.6333,,C -667,0,2,"Butler, Mr. Reginald Fenton",male,25.0,0,0,234686,13.0,,S -61,0,3,"Sirayanian, Mr. Orsen",male,22.0,0,0,2669,7.2292,,C -642,1,1,"Sagesser, Mlle. Emma",female,24.0,0,0,PC 17477,69.3,B35,C -469,0,3,"Scanlan, Mr. James",male,,0,0,36209,7.725,,Q -792,0,2,"Gaskell, Mr. Alfred",male,16.0,0,0,239865,26.0,,S -465,0,3,"Maisner, Mr. Simon",male,,0,0,A/S 2816,8.05,,S -551,1,1,"Thayer, Mr. John Borland Jr",male,17.0,0,2,17421,110.8833,C70,C -523,0,3,"Lahoud, Mr. Sarkis",male,,0,0,2624,7.225,,C -369,1,3,"Jermyn, Miss. Annie",female,,0,0,14313,7.75,,Q -864,0,3,"Sage, Miss. Dorothy Edith ""Dolly""",female,,8,2,CA. 2343,69.55,,S -839,1,3,"Chip, Mr. Chang",male,32.0,0,0,1601,56.4958,,S -590,0,3,"Murdlin, Mr. Joseph",male,,0,0,A./5. 3235,8.05,,S -9,1,3,"Johnson, Mrs. Oscar W (Elisabeth Vilhelmina Berg)",female,27.0,0,2,347742,11.1333,,S -505,1,1,"Maioni, Miss. Roberta",female,16.0,0,0,110152,86.5,B79,S -572,1,1,"Appleton, Mrs. Edward Dale (Charlotte Lamson)",female,53.0,2,0,11769,51.4792,C101,S -235,0,2,"Leyson, Mr. Robert William Norman",male,24.0,0,0,C.A. 29566,10.5,,S -345,0,2,"Fox, Mr. Stanley Hubert",male,36.0,0,0,229236,13.0,,S -714,0,3,"Larsson, Mr. August Viktor",male,29.0,0,0,7545,9.4833,,S -477,0,2,"Renouf, Mr. Peter Henry",male,34.0,1,0,31027,21.0,,S -587,0,2,"Jarvis, Mr. John Denzil",male,47.0,0,0,237565,15.0,,S -630,0,3,"O'Connell, Mr. Patrick D",male,,0,0,334912,7.7333,,Q -133,0,3,"Robins, Mrs. Alexander A (Grace Charity Laury)",female,47.0,1,0,A/5. 3337,14.5,,S -27,0,3,"Emir, Mr. Farred Chehab",male,,0,0,2631,7.225,,C -612,0,3,"Jardin, Mr. Jose Neto",male,,0,0,SOTON/O.Q. 3101305,7.05,,S -292,1,1,"Bishop, Mrs. Dickinson H (Helen Walton)",female,19.0,1,0,11967,91.0792,B49,C -293,0,2,"Levy, Mr. Rene Jacques",male,36.0,0,0,SC/Paris 2163,12.875,D,C -40,1,3,"Nicola-Yarred, Miss. Jamila",female,14.0,1,0,2651,11.2417,,C -205,1,3,"Cohen, Mr. Gurshon ""Gus""",male,18.0,0,0,A/5 3540,8.05,,S -832,1,2,"Richards, Master. George Sibley",male,0.83,1,1,29106,18.75,,S -716,0,3,"Soholt, Mr. Peter Andreas Lauritz Andersen",male,19.0,0,0,348124,7.65,F G73,S -596,0,3,"Van Impe, Mr. Jean Baptiste",male,36.0,1,1,345773,24.15,,S -344,0,2,"Sedgwick, Mr. Charles Frederick Waddington",male,25.0,0,0,244361,13.0,,S -687,0,3,"Panula, Mr. Jaako Arnold",male,14.0,4,1,3101295,39.6875,,S -662,0,3,"Badt, Mr. Mohamed",male,40.0,0,0,2623,7.225,,C -66,1,3,"Moubarek, Master. Gerios",male,,1,1,2661,15.2458,,C -820,0,3,"Skoog, Master. Karl Thorsten",male,10.0,3,2,347088,27.9,,S -865,0,2,"Gill, Mr. John William",male,24.0,0,0,233866,13.0,,S -323,1,2,"Slayter, Miss. Hilda Mary",female,30.0,0,0,234818,12.35,,Q -358,0,2,"Funk, Miss. Annie Clemmer",female,38.0,0,0,237671,13.0,,S -129,1,3,"Peter, Miss. Anna",female,,1,1,2668,22.3583,F E69,C -166,1,3,"Goldsmith, Master. Frank John William ""Frankie""",male,9.0,0,2,363291,20.525,,S -799,0,3,"Ibrahim Shawah, Mr. Yousseff",male,30.0,0,0,2685,7.2292,,C -770,0,3,"Gronnestad, Mr. Daniel Danielsen",male,32.0,0,0,8471,8.3625,,S -785,0,3,"Ali, Mr. William",male,25.0,0,0,SOTON/O.Q. 3101312,7.05,,S -399,0,2,"Pain, Dr. Alfred",male,23.0,0,0,244278,10.5,,S -746,0,1,"Crosby, Capt. Edward Gifford",male,70.0,1,1,WE/P 5735,71.0,B22,S -498,0,3,"Shellard, Mr. Frederick William",male,,0,0,C.A. 6212,15.1,,S -297,0,3,"Hanna, Mr. Mansour",male,23.5,0,0,2693,7.2292,,C -295,0,3,"Mineff, Mr. Ivan",male,24.0,0,0,349233,7.8958,,S -545,0,1,"Douglas, Mr. Walter Donald",male,50.0,1,0,PC 17761,106.425,C86,C -755,1,2,"Herman, Mrs. Samuel (Jane Laver)",female,48.0,1,2,220845,65.0,,S -305,0,3,"Williams, Mr. Howard Hugh ""Harry""",male,,0,0,A/5 2466,8.05,,S -682,1,1,"Hassab, Mr. Hammad",male,27.0,0,0,PC 17572,76.7292,D49,C -124,1,2,"Webber, Miss. Susan",female,32.5,0,0,27267,13.0,E101,S -499,0,1,"Allison, Mrs. Hudson J C (Bessie Waldo Daniels)",female,25.0,1,2,113781,151.55,C22 C26,S -870,1,3,"Johnson, Master. Harold Theodor",male,4.0,1,1,347742,11.1333,,S -72,0,3,"Goodwin, Miss. Lillian Amy",female,16.0,5,2,CA 2144,46.9,,S -120,0,3,"Andersson, Miss. Ellis Anna Maria",female,2.0,4,2,347082,31.275,,S -325,0,3,"Sage, Mr. George John Jr",male,,8,2,CA. 2343,69.55,,S -383,0,3,"Tikkanen, Mr. Juho",male,32.0,0,0,STON/O 2. 3101293,7.925,,S -628,1,1,"Longley, Miss. Gretchen Fiske",female,21.0,0,0,13502,77.9583,D9,S -744,0,3,"McNamee, Mr. Neal",male,24.0,1,0,376566,16.1,,S -684,0,3,"Goodwin, Mr. Charles Edward",male,14.0,5,2,CA 2144,46.9,,S -598,0,3,"Johnson, Mr. Alfred",male,49.0,0,0,LINE,0.0,,S -866,1,2,"Bystrom, Mrs. (Karolina)",female,42.0,0,0,236852,13.0,,S -53,1,1,"Harper, Mrs. Henry Sleeper (Myna Haxtun)",female,49.0,1,0,PC 17572,76.7292,D33,C -732,0,3,"Hassan, Mr. Houssein G N",male,11.0,0,0,2699,18.7875,,C -306,1,1,"Allison, Master. Hudson Trevor",male,0.92,1,2,113781,151.55,C22 C26,S -140,0,1,"Giglio, Mr. Victor",male,24.0,0,0,PC 17593,79.2,B86,C -814,0,3,"Andersson, Miss. Ebba Iris Alfrida",female,6.0,4,2,347082,31.275,,S -310,1,1,"Francatelli, Miss. Laura Mabel",female,30.0,0,0,PC 17485,56.9292,E36,C -71,0,2,"Jenkin, Mr. Stephen Curnow",male,32.0,0,0,C.A. 33111,10.5,,S -529,0,3,"Salonen, Mr. Johan Werner",male,39.0,0,0,3101296,7.925,,S -466,0,3,"Goncalves, Mr. Manuel Estanslas",male,38.0,0,0,SOTON/O.Q. 3101306,7.05,,S -319,1,1,"Wick, Miss. Mary Natalie",female,31.0,0,2,36928,164.8667,C7,S -259,1,1,"Ward, Miss. Anna",female,35.0,0,0,PC 17755,512.3292,,C -114,0,3,"Jussila, Miss. Katriina",female,20.0,1,0,4136,9.825,,S -625,0,3,"Bowen, Mr. David John ""Dai""",male,21.0,0,0,54636,16.1,,S -555,1,3,"Ohman, Miss. Velin",female,22.0,0,0,347085,7.775,,S -357,1,1,"Bowerman, Miss. Elsie Edith",female,22.0,0,1,113505,55.0,E33,S -837,0,3,"Pasic, Mr. Jakob",male,21.0,0,0,315097,8.6625,,S -84,0,1,"Carrau, Mr. Francisco M",male,28.0,0,0,113059,47.1,,S -184,1,2,"Becker, Master. Richard F",male,1.0,2,1,230136,39.0,F4,S -183,0,3,"Asplund, Master. Clarence Gustaf Hugo",male,9.0,4,2,347077,31.3875,,S -145,0,2,"Andrew, Mr. Edgardo Samuel",male,18.0,0,0,231945,11.5,,S -859,1,3,"Baclini, Mrs. Solomon (Latifa Qurban)",female,24.0,0,3,2666,19.2583,,C -299,1,1,"Saalfeld, Mr. Adolphe",male,,0,0,19988,30.5,C106,S -658,0,3,"Bourke, Mrs. John (Catherine)",female,32.0,1,1,364849,15.5,,Q -507,1,2,"Quick, Mrs. Frederick Charles (Jane Richards)",female,33.0,0,2,26360,26.0,,S -692,1,3,"Karun, Miss. Manca",female,4.0,0,1,349256,13.4167,,C -88,0,3,"Slocovski, Mr. Selman Francis",male,,0,0,SOTON/OQ 392086,8.05,,S -314,0,3,"Hendekovic, Mr. Ignjac",male,28.0,0,0,349243,7.8958,,S -800,0,3,"Van Impe, Mrs. Jean Baptiste (Rosalie Paula Govaert)",female,30.0,1,1,345773,24.15,,S -614,0,3,"Horgan, Mr. John",male,,0,0,370377,7.75,,Q -12,1,1,"Bonnell, Miss. Elizabeth",female,58.0,0,0,113783,26.55,C103,S -771,0,3,"Lievens, Mr. Rene Aime",male,24.0,0,0,345781,9.5,,S -365,0,3,"O'Brien, Mr. Thomas",male,,1,0,370365,15.5,,Q -876,1,3,"Najib, Miss. Adele Kiamie ""Jane""",female,15.0,0,0,2667,7.225,,C -195,1,1,"Brown, Mrs. James Joseph (Margaret Tobin)",female,44.0,0,0,PC 17610,27.7208,B4,C -594,0,3,"Bourke, Miss. Mary",female,,0,2,364848,7.75,,Q -654,1,3,"O'Leary, Miss. Hanora ""Norah""",female,,0,0,330919,7.8292,,Q -402,0,3,"Adams, Mr. John",male,26.0,0,0,341826,8.05,,S -83,1,3,"McDermott, Miss. Brigdet Delia",female,,0,0,330932,7.7875,,Q -669,0,3,"Cook, Mr. Jacob",male,43.0,0,0,A/5 3536,8.05,,S -878,0,3,"Petroff, Mr. Nedelio",male,19.0,0,0,349212,7.8958,,S -833,0,3,"Saad, Mr. Amin",male,,0,0,2671,7.2292,,C -75,1,3,"Bing, Mr. Lee",male,32.0,0,0,1601,56.4958,,S -722,0,3,"Jensen, Mr. Svend Lauritz",male,17.0,1,0,350048,7.0542,,S -251,0,3,"Reed, Mr. James George",male,,0,0,362316,7.25,,S -238,1,2,"Collyer, Miss. Marjorie ""Lottie""",female,8.0,0,2,C.A. 31921,26.25,,S -146,0,2,"Nicholls, Mr. Joseph Charles",male,19.0,1,1,C.A. 33112,36.75,,S -808,0,3,"Pettersson, Miss. Ellen Natalia",female,18.0,0,0,347087,7.775,,S -131,0,3,"Drazenoic, Mr. Jozef",male,33.0,0,0,349241,7.8958,,C -576,0,3,"Patchett, Mr. George",male,19.0,0,0,358585,14.5,,S -515,0,3,"Coleff, Mr. Satio",male,24.0,0,0,349209,7.4958,,S -847,0,3,"Sage, Mr. Douglas Bullen",male,,8,2,CA. 2343,69.55,,S -648,1,1,"Simonius-Blumer, Col. Oberst Alfons",male,56.0,0,0,13213,35.5,A26,C -443,0,3,"Petterson, Mr. Johan Emil",male,25.0,1,0,347076,7.775,,S -478,0,3,"Braund, Mr. Lewis Richard",male,29.0,1,0,3460,7.0458,,S -537,0,1,"Butt, Major. Archibald Willingham",male,45.0,0,0,113050,26.55,B38,S -169,0,1,"Baumann, Mr. John D",male,,0,0,PC 17318,25.925,,S -149,0,2,"Navratil, Mr. Michel (""Louis M Hoffman"")",male,36.5,0,2,230080,26.0,F2,S -290,1,3,"Connolly, Miss. Kate",female,22.0,0,0,370373,7.75,,Q -15,0,3,"Vestrom, Miss. Hulda Amanda Adolfina",female,14.0,0,0,350406,7.8542,,S -386,0,2,"Davies, Mr. Charles Henry",male,18.0,0,0,S.O.C. 14879,73.5,,S -811,0,3,"Alexander, Mr. William",male,26.0,0,0,3474,7.8875,,S -78,0,3,"Moutal, Mr. Rahamin Haim",male,,0,0,374746,8.05,,S -738,1,1,"Lesurer, Mr. Gustave J",male,35.0,0,0,PC 17755,512.3292,B101,C -452,0,3,"Hagland, Mr. Ingvald Olai Olsen",male,,1,0,65303,19.9667,,S -35,0,1,"Meyer, Mr. Edgar Joseph",male,28.0,1,0,PC 17604,82.1708,,C -347,1,2,"Smith, Miss. Marion Elsie",female,40.0,0,0,31418,13.0,,S -436,1,1,"Carter, Miss. Lucile Polk",female,14.0,1,2,113760,120.0,B96 B98,S -390,1,2,"Lehmann, Miss. Bertha",female,17.0,0,0,SC 1748,12.0,,C -657,0,3,"Radeff, Mr. Alexander",male,,0,0,349223,7.8958,,S -695,0,1,"Weir, Col. John",male,60.0,0,0,113800,26.55,,S -586,1,1,"Taussig, Miss. Ruth",female,18.0,0,2,110413,79.65,E68,S -384,1,1,"Holverson, Mrs. Alexander Oskar (Mary Aline Towner)",female,35.0,1,0,113789,52.0,,S -58,0,3,"Novel, Mr. Mansouer",male,28.5,0,0,2697,7.2292,,C -246,0,1,"Minahan, Dr. William Edward",male,44.0,2,0,19928,90.0,C78,Q -557,1,1,"Duff Gordon, Lady. (Lucille Christiana Sutherland) (""Mrs Morgan"")",female,48.0,1,0,11755,39.6,A16,C -605,1,1,"Homer, Mr. Harry (""Mr E Haven"")",male,35.0,0,0,111426,26.55,,C -350,0,3,"Dimic, Mr. Jovan",male,42.0,0,0,315088,8.6625,,S -659,0,2,"Eitemiller, Mr. George Floyd",male,23.0,0,0,29751,13.0,,S -415,1,3,"Sundman, Mr. Johan Julian",male,44.0,0,0,STON/O 2. 3101269,7.925,,S -713,1,1,"Taylor, Mr. Elmer Zebley",male,48.0,1,0,19996,52.0,C126,S -474,1,2,"Jerwan, Mrs. Amin S (Marie Marthe Thuillard)",female,23.0,0,0,SC/AH Basle 541,13.7917,D,C -139,0,3,"Osen, Mr. Olaf Elon",male,16.0,0,0,7534,9.2167,,S -224,0,3,"Nenkoff, Mr. Christo",male,,0,0,349234,7.8958,,S -221,1,3,"Sunderland, Mr. Victor Francis",male,16.0,0,0,SOTON/OQ 392089,8.05,,S -68,0,3,"Crease, Mr. Ernest James",male,19.0,0,0,S.P. 3464,8.1583,,S -622,1,1,"Kimball, Mr. Edwin Nelson Jr",male,42.0,1,0,11753,52.5542,D19,S -467,0,2,"Campbell, Mr. William",male,,0,0,239853,0.0,,S -525,0,3,"Kassem, Mr. Fared",male,,0,0,2700,7.2292,,C -17,0,3,"Rice, Master. Eugene",male,2.0,4,1,382652,29.125,,Q -430,1,3,"Pickard, Mr. Berk (Berk Trembisky)",male,32.0,0,0,SOTON/O.Q. 392078,8.05,E10,S -90,0,3,"Celotti, Mr. Francesco",male,24.0,0,0,343275,8.05,,S -486,0,3,"Lefebre, Miss. Jeannie",female,,3,1,4133,25.4667,,S -831,1,3,"Yasbeck, Mrs. Antoni (Selini Alexander)",female,15.0,1,0,2659,14.4542,,C -440,0,2,"Kvillner, Mr. Johan Henrik Johannesson",male,31.0,0,0,C.A. 18723,10.5,,S -244,0,3,"Maenpaa, Mr. Matti Alexanteri",male,22.0,0,0,STON/O 2. 3101275,7.125,,S -882,0,3,"Markun, Mr. Johann",male,33.0,0,0,349257,7.8958,,S -287,1,3,"de Mulder, Mr. Theodore",male,30.0,0,0,345774,9.5,,S -735,0,2,"Troupiansky, Mr. Moses Aaron",male,23.0,0,0,233639,13.0,,S -620,0,2,"Gavey, Mr. Lawrence",male,26.0,0,0,31028,10.5,,S -296,0,1,"Lewy, Mr. Ervin G",male,,0,0,PC 17612,27.7208,,C -187,1,3,"O'Brien, Mrs. Thomas (Johanna ""Hannah"" Godfrey)",female,,1,0,370365,15.5,,Q -629,0,3,"Bostandyeff, Mr. Guentcho",male,26.0,0,0,349224,7.8958,,S -123,0,2,"Nasser, Mr. Nicholas",male,32.5,1,0,237736,30.0708,,C -678,1,3,"Turja, Miss. Anna Sofia",female,18.0,0,0,4138,9.8417,,S -263,0,1,"Taussig, Mr. Emil",male,52.0,1,1,110413,79.65,E67,S -439,0,1,"Fortune, Mr. Mark",male,64.0,1,4,19950,263.0,C23 C25 C27,S -410,0,3,"Lefebre, Miss. Ida",female,,3,1,4133,25.4667,,S -497,1,1,"Eustis, Miss. Elizabeth Mussey",female,54.0,1,0,36947,78.2667,D20,C -522,0,3,"Vovk, Mr. Janko",male,22.0,0,0,349252,7.8958,,S -766,1,1,"Hogeboom, Mrs. John C (Anna Andrews)",female,51.0,1,0,13502,77.9583,D11,S -408,1,2,"Richards, Master. William Rowe",male,3.0,1,1,29106,18.75,,S -420,0,3,"Van Impe, Miss. Catharina",female,10.0,0,2,345773,24.15,,S -453,0,1,"Foreman, Mr. Benjamin Laventall",male,30.0,0,0,113051,27.75,C111,C -447,1,2,"Mellinger, Miss. Madeleine Violet",female,13.0,0,1,250644,19.5,,S -197,0,3,"Mernagh, Mr. Robert",male,,0,0,368703,7.75,,Q -227,1,2,"Mellors, Mr. William John",male,19.0,0,0,SW/PP 751,10.5,,S -852,0,3,"Svensson, Mr. Johan",male,74.0,0,0,347060,7.775,,S -763,1,3,"Barah, Mr. Hanna Assi",male,20.0,0,0,2663,7.2292,,C -257,1,1,"Thorne, Mrs. Gertrude Maybelle",female,,0,0,PC 17585,79.2,,C -407,0,3,"Widegren, Mr. Carl/Charles Peter",male,51.0,0,0,347064,7.75,,S -103,0,1,"White, Mr. Richard Frasar",male,21.0,0,1,35281,77.2875,D26,S -315,0,2,"Hart, Mr. Benjamin",male,43.0,1,1,F.C.C. 13529,26.25,,S -77,0,3,"Staneff, Mr. Ivan",male,,0,0,349208,7.8958,,S -632,0,3,"Lundahl, Mr. Johan Svensson",male,51.0,0,0,347743,7.0542,,S -750,0,3,"Connaghton, Mr. Michael",male,31.0,0,0,335097,7.75,,Q -627,0,2,"Kirkland, Rev. Charles Leonard",male,57.0,0,0,219533,12.35,,Q -96,0,3,"Shorney, Mr. Charles Joseph",male,,0,0,374910,8.05,,S -171,0,1,"Van der hoef, Mr. Wyckoff",male,61.0,0,0,111240,33.5,B19,S -881,1,2,"Shelley, Mrs. William (Imanita Parrish Hall)",female,25.0,0,1,230433,26.0,,S -95,0,3,"Coxon, Mr. Daniel",male,59.0,0,0,364500,7.25,,S -215,0,3,"Kiernan, Mr. Philip",male,,1,0,367229,7.75,,Q -39,0,3,"Vander Planke, Miss. Augusta Maria",female,18.0,2,0,345764,18.0,,S -774,0,3,"Elias, Mr. Dibo",male,,0,0,2674,7.225,,C -37,1,3,"Mamee, Mr. Hanna",male,,0,0,2677,7.2292,,C -181,0,3,"Sage, Miss. Constance Gladys",female,,8,2,CA. 2343,69.55,,S -177,0,3,"Lefebre, Master. Henry Forbes",male,,3,1,4133,25.4667,,S -812,0,3,"Lester, Mr. James",male,39.0,0,0,A/4 48871,24.15,,S -496,0,3,"Yousseff, Mr. Gerious",male,,0,0,2627,14.4583,,C -503,0,3,"O'Sullivan, Miss. Bridget Mary",female,,0,0,330909,7.6292,,Q -216,1,1,"Newell, Miss. Madeleine",female,31.0,1,0,35273,113.275,D36,C -395,1,3,"Sandstrom, Mrs. Hjalmar (Agnes Charlotta Bengtsson)",female,24.0,0,2,PP 9549,16.7,G6,S -720,0,3,"Johnson, Mr. Malkolm Joackim",male,33.0,0,0,347062,7.775,,S -213,0,3,"Perkin, Mr. John Henry",male,22.0,0,0,A/5 21174,7.25,,S -644,1,3,"Foo, Mr. Choong",male,,0,0,1601,56.4958,,S -583,0,2,"Downton, Mr. William James",male,54.0,0,0,28403,26.0,,S -132,0,3,"Coelho, Mr. Domingos Fernandeo",male,20.0,0,0,SOTON/O.Q. 3101307,7.05,,S -363,0,3,"Barbara, Mrs. (Catherine David)",female,45.0,0,1,2691,14.4542,,C -461,1,1,"Anderson, Mr. Harry",male,48.0,0,0,19952,26.55,E12,S -186,0,1,"Rood, Mr. Hugh Roscoe",male,,0,0,113767,50.0,A32,S -14,0,3,"Andersson, Mr. Anders Johan",male,39.0,1,5,347082,31.275,,S -1,0,3,"Braund, Mr. Owen Harris",male,22.0,1,0,A/5 21171,7.25,,S -694,0,3,"Saad, Mr. Khalil",male,25.0,0,0,2672,7.225,,C -476,0,1,"Clifford, Mr. George Quincy",male,,0,0,110465,52.0,A14,S -348,1,3,"Davison, Mrs. Thomas Henry (Mary E Finck)",female,,1,0,386525,16.1,,S -489,0,3,"Somerton, Mr. Francis William",male,30.0,0,0,A.5. 18509,8.05,,S -69,1,3,"Andersson, Miss. Erna Alexandra",female,17.0,4,2,3101281,7.925,,S -883,0,3,"Dahlberg, Miss. Gerda Ulrika",female,22.0,0,0,7552,10.5167,,S -18,1,2,"Williams, Mr. Charles Eugene",male,,0,0,244373,13.0,,S -31,0,1,"Uruchurtu, Don. Manuel E",male,40.0,0,0,PC 17601,27.7208,,C -619,1,2,"Becker, Miss. Marion Louise",female,4.0,2,1,230136,39.0,F4,S -526,0,3,"Farrell, Mr. James",male,40.5,0,0,367232,7.75,,Q -585,0,3,"Paulner, Mr. Uscher",male,,0,0,3411,8.7125,,C -274,0,1,"Natsch, Mr. Charles H",male,37.0,0,1,PC 17596,29.7,C118,C -715,0,2,"Greenberg, Mr. Samuel",male,52.0,0,0,250647,13.0,,S -438,1,2,"Richards, Mrs. Sidney (Emily Hocking)",female,24.0,2,3,29106,18.75,,S -193,1,3,"Andersen-Jensen, Miss. Carla Christine Nielsine",female,19.0,1,0,350046,7.8542,,S -275,1,3,"Healy, Miss. Hanora ""Nora""",female,,0,0,370375,7.75,,Q -173,1,3,"Johnson, Miss. Eleanor Ileen",female,1.0,1,1,347742,11.1333,,S -807,0,1,"Andrews, Mr. Thomas Jr",male,39.0,0,0,112050,0.0,A36,S -680,1,1,"Cardeza, Mr. Thomas Drake Martinez",male,36.0,0,1,PC 17755,512.3292,B51 B53 B55,C -304,1,2,"Keane, Miss. Nora A",female,,0,0,226593,12.35,E101,Q -370,1,1,"Aubart, Mme. Leontine Pauline",female,24.0,0,0,PC 17477,69.3,B35,C -239,0,2,"Pengelly, Mr. Frederick William",male,19.0,0,0,28665,10.5,,S -825,0,3,"Panula, Master. Urho Abraham",male,2.0,4,1,3101295,39.6875,,S -284,1,3,"Dorking, Mr. Edward Arthur",male,19.0,0,0,A/5. 10482,8.05,,S -182,0,2,"Pernot, Mr. Rene",male,,0,0,SC/PARIS 2131,15.05,,C -64,0,3,"Skoog, Master. Harald",male,4.0,3,2,347088,27.9,,S -404,0,3,"Hakkarainen, Mr. Pekka Pietari",male,28.0,1,0,STON/O2. 3101279,15.85,,S -479,0,3,"Karlsson, Mr. Nils August",male,22.0,0,0,350060,7.5208,,S -618,0,3,"Lobb, Mrs. William Arthur (Cordelia K Stanlick)",female,26.0,1,0,A/5. 3336,16.1,,S -3,1,3,"Heikkinen, Miss. Laina",female,26.0,0,0,STON/O2. 3101282,7.925,,S -337,0,1,"Pears, Mr. Thomas Clinton",male,29.0,1,0,113776,66.6,C2,S -764,1,1,"Carter, Mrs. William Ernest (Lucile Polk)",female,36.0,1,2,113760,120.0,B96 B98,S -696,0,2,"Chapman, Mr. Charles Henry",male,52.0,0,0,248731,13.5,,S -783,0,1,"Long, Mr. Milton Clyde",male,29.0,0,0,113501,30.0,D6,S -318,0,2,"Moraweck, Dr. Ernest",male,54.0,0,0,29011,14.0,,S -706,0,2,"Morley, Mr. Henry Samuel (""Mr Henry Marshall"")",male,39.0,0,0,250655,26.0,,S -432,1,3,"Thorneycroft, Mrs. Percival (Florence Kate White)",female,,1,0,376564,16.1,,S -50,0,3,"Arnold-Franchi, Mrs. Josef (Josefine Franchi)",female,18.0,1,0,349237,17.8,,S -136,0,2,"Richard, Mr. Emile",male,23.0,0,0,SC/PARIS 2133,15.0458,,C -889,0,3,"Johnston, Miss. Catherine Helen ""Carrie""",female,,1,2,W./C. 6607,23.45,,S -604,0,3,"Torber, Mr. Ernst William",male,44.0,0,0,364511,8.05,,S -5,0,3,"Allen, Mr. William Henry",male,35.0,0,0,373450,8.05,,S -613,1,3,"Murphy, Miss. Margaret Jane",female,,1,0,367230,15.5,,Q -724,0,2,"Hodges, Mr. Henry Price",male,50.0,0,0,250643,13.0,,S -758,0,2,"Bailey, Mr. Percy Andrew",male,18.0,0,0,29108,11.5,,S -142,1,3,"Nysten, Miss. Anna Sofia",female,22.0,0,0,347081,7.75,,S -416,0,3,"Meek, Mrs. Thomas (Annie Louise Rowley)",female,,0,0,343095,8.05,,S -668,0,3,"Rommetvedt, Mr. Knud Paust",male,,0,0,312993,7.775,,S -387,0,3,"Goodwin, Master. Sidney Leonard",male,1.0,5,2,CA 2144,46.9,,S -87,0,3,"Ford, Mr. William Neal",male,16.0,1,3,W./C. 6608,34.375,,S -94,0,3,"Dean, Mr. Bertram Frank",male,26.0,1,2,C.A. 2315,20.575,,S -650,1,3,"Stanley, Miss. Amy Zillah Elsie",female,23.0,0,0,CA. 2314,7.55,,S -508,1,1,"Bradley, Mr. George (""George Arthur Brayton"")",male,,0,0,111427,26.55,,S -571,1,2,"Harris, Mr. George",male,62.0,0,0,S.W./PP 752,10.5,,S -317,1,2,"Kantor, Mrs. Sinai (Miriam Sternin)",female,24.0,1,0,244367,26.0,,S -229,0,2,"Fahlstrom, Mr. Arne Jonas",male,18.0,0,0,236171,13.0,,S -656,0,2,"Hickman, Mr. Leonard Mark",male,24.0,2,0,S.O.C. 14879,73.5,,S -281,0,3,"Duane, Mr. Frank",male,65.0,0,0,336439,7.75,,Q -753,0,3,"Vande Velde, Mr. Johannes Joseph",male,33.0,0,0,345780,9.5,,S -803,1,1,"Carter, Master. William Thornton II",male,11.0,1,2,113760,120.0,B96 B98,S -527,1,2,"Ridsdale, Miss. Lucy",female,50.0,0,0,W./C. 14258,10.5,,S -739,0,3,"Ivanoff, Mr. Kanio",male,,0,0,349201,7.8958,,S -579,0,3,"Caram, Mrs. Joseph (Maria Elias)",female,,1,0,2689,14.4583,,C -54,1,2,"Faunthorpe, Mrs. Lizzie (Elizabeth Anne Wilkinson)",female,29.0,1,0,2926,26.0,,S -867,1,2,"Duran y More, Miss. Asuncion",female,27.0,1,0,SC/PARIS 2149,13.8583,,C -351,0,3,"Odahl, Mr. Nils Martin",male,23.0,0,0,7267,9.225,,S -80,1,3,"Dowdell, Miss. Elizabeth",female,30.0,0,0,364516,12.475,,S -856,1,3,"Aks, Mrs. Sam (Leah Rosen)",female,18.0,0,1,392091,9.35,,S -872,1,1,"Beckwith, Mrs. Richard Leonard (Sallie Monypeny)",female,47.0,1,1,11751,52.5542,D35,S -836,1,1,"Compton, Miss. Sara Rebecca",female,39.0,1,1,PC 17756,83.1583,E49,C -793,0,3,"Sage, Miss. Stella Anna",female,,8,2,CA. 2343,69.55,,S -521,1,1,"Perreault, Miss. Anne",female,30.0,0,0,12749,93.5,B73,S From f2dbb51094484fbbc226c6609e4614a1dd903d20 Mon Sep 17 00:00:00 2001 From: better629 Date: Sat, 3 Feb 2024 13:33:02 +0800 Subject: [PATCH 592/637] update code due to failed unittests --- metagpt/llm.py | 2 +- metagpt/provider/google_gemini_api.py | 4 +++- tests/metagpt/actions/test_action_node.py | 8 +++++++- tests/metagpt/utils/test_repair_llm_raw_output.py | 6 +++--- 4 files changed, 14 insertions(+), 6 deletions(-) diff --git a/metagpt/llm.py b/metagpt/llm.py index a3fc5613a..465e419a1 100644 --- a/metagpt/llm.py +++ b/metagpt/llm.py @@ -16,5 +16,5 @@ def LLM(llm_config: Optional[LLMConfig] = None, context: Context = None) -> Base """get the default llm provider if name is None""" ctx = context or Context() if llm_config is not None: - ctx.llm_with_cost_manager_from_llm_config(llm_config) + return ctx.llm_with_cost_manager_from_llm_config(llm_config) return ctx.llm() diff --git a/metagpt/provider/google_gemini_api.py b/metagpt/provider/google_gemini_api.py index 6df814b55..2647ab16b 100644 --- a/metagpt/provider/google_gemini_api.py +++ b/metagpt/provider/google_gemini_api.py @@ -2,6 +2,8 @@ # -*- coding: utf-8 -*- # @Desc : Google Gemini LLM from https://ai.google.dev/tutorials/python_quickstart +from typing import Optional, Union + import google.generativeai as genai from google.ai import generativelanguage as glm from google.generativeai.generative_models import GenerativeModel @@ -58,7 +60,7 @@ class GeminiLLM(BaseLLM): def __init_gemini(self, config: LLMConfig): genai.configure(api_key=config.api_key) - def _user_msg(self, msg: str) -> dict[str, str]: + def _user_msg(self, msg: str, images: Optional[Union[str, list[str]]] = None) -> dict[str, str]: # Not to change BaseLLM default functions but update with Gemini's conversation format. # You should follow the format. return {"role": "user", "parts": [msg]} diff --git a/tests/metagpt/actions/test_action_node.py b/tests/metagpt/actions/test_action_node.py index 589282879..989e2249c 100644 --- a/tests/metagpt/actions/test_action_node.py +++ b/tests/metagpt/actions/test_action_node.py @@ -244,13 +244,19 @@ def test_create_model_class_with_mapping(): @pytest.mark.asyncio -async def test_action_node_with_image(): +async def test_action_node_with_image(mocker): + # add a mock to update model in unittest, due to the gloabl MockLLM + def _cons_kwargs(self, messages: list[dict], timeout=3, **extra_kwargs) -> dict: + kwargs = {"messages": messages, "temperature": 0.3, "model": "gpt-4-vision-preview"} + return kwargs + invoice = ActionNode( key="invoice", expected_type=bool, instruction="if it's a invoice file, return True else False", example="False" ) invoice_path = Path(__file__).parent.joinpath("..", "..", "data", "invoices", "invoice-2.png") img_base64 = encode_image(invoice_path) + mocker.patch("metagpt.provider.openai_api.OpenAILLM._cons_kwargs", _cons_kwargs) node = await invoice.fill(context="", llm=LLM(), images=[img_base64]) assert node.instruct_content.invoice diff --git a/tests/metagpt/utils/test_repair_llm_raw_output.py b/tests/metagpt/utils/test_repair_llm_raw_output.py index 9eec24727..e28423b91 100644 --- a/tests/metagpt/utils/test_repair_llm_raw_output.py +++ b/tests/metagpt/utils/test_repair_llm_raw_output.py @@ -135,7 +135,7 @@ def test_repair_json_format(): } """ target_output = """{ - "Language": "en_us", + "Language": "en_us", "Programming Language": "Python" }""" output = repair_llm_raw_output(output=raw_output, req_keys=[None], repair_type=RepairType.JSON) @@ -148,7 +148,7 @@ def test_repair_json_format(): } """ target_output = """{ - "Language": "en_us", + "Language": "en_us", "Programming Language": "Python" }""" output = repair_llm_raw_output(output=raw_output, req_keys=[None], repair_type=RepairType.JSON) @@ -161,7 +161,7 @@ def test_repair_json_format(): } """ target_output = """{ - "Language": "#en_us#", + "Language": "#en_us#", "Programming Language": "//Python # Code // Language//" }""" output = repair_llm_raw_output(output=raw_output, req_keys=[None], repair_type=RepairType.JSON) From 05e2f3b397051fef5824772699b9f6c294a3775b Mon Sep 17 00:00:00 2001 From: better629 Date: Sat, 3 Feb 2024 13:39:48 +0800 Subject: [PATCH 593/637] add stanford_town ville env files --- .../the_ville/matrix/maze/arena_maze.csv | 1 + .../the_ville/matrix/maze/collision_maze.csv | 1 + .../matrix/maze/game_object_maze.csv | 1 + .../the_ville/matrix/maze/sector_maze.csv | 1 + .../matrix/maze/spawning_location_maze.csv | 1 + .../the_ville/matrix/maze_meta_info.json | 5 ++ .../matrix/special_blocks/arena_blocks.csv | 63 +++++++++++++++++++ .../special_blocks/game_object_blocks.csv | 46 ++++++++++++++ .../matrix/special_blocks/sector_blocks.csv | 19 ++++++ .../spawning_location_blocks.csv | 40 ++++++++++++ .../matrix/special_blocks/world_blocks.csv | 1 + 11 files changed, 179 insertions(+) create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/maze/arena_maze.csv create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/maze/collision_maze.csv create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/maze/game_object_maze.csv create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/maze/sector_maze.csv create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/maze/spawning_location_maze.csv create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/maze_meta_info.json create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/special_blocks/arena_blocks.csv create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/special_blocks/game_object_blocks.csv create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/special_blocks/sector_blocks.csv create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/special_blocks/spawning_location_blocks.csv create mode 100644 tests/data/environment/stanford_town/the_ville/matrix/special_blocks/world_blocks.csv diff --git a/tests/data/environment/stanford_town/the_ville/matrix/maze/arena_maze.csv b/tests/data/environment/stanford_town/the_ville/matrix/maze/arena_maze.csv new file mode 100644 index 000000000..f9cf65ecd --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/maze/arena_maze.csv @@ -0,0 +1 @@ +0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32200, 32200, 32200, 32200, 32200, 0, 0, 32151, 32151, 32151, 32151, 32151, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32199, 32199, 32199, 32199, 32199, 32199, 0, 32140, 32140, 32140, 0, 0, 32160, 32160, 32160, 32160, 32160, 0, 0, 32170, 32170, 32170, 32170, 32170, 32170, 32170, 32170, 0, 32180, 32180, 32180, 0, 0, 32200, 32200, 32200, 32200, 32200, 0, 0, 32151, 32151, 32151, 32151, 32151, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32199, 32199, 32199, 32199, 32199, 32199, 0, 32140, 32140, 32140, 0, 0, 32160, 32160, 32160, 32160, 32160, 0, 0, 32170, 32170, 32170, 32170, 32170, 32170, 32170, 32170, 0, 32180, 32180, 32180, 0, 0, 32200, 32200, 32200, 32200, 32200, 0, 0, 32151, 32151, 32151, 32151, 32151, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32199, 32199, 32199, 32199, 32199, 32199, 0, 32140, 32140, 32140, 0, 0, 32160, 32160, 32160, 32160, 32160, 0, 0, 32170, 32170, 32170, 32170, 32170, 32170, 32170, 32170, 0, 32180, 32180, 32180, 0, 0, 32200, 32200, 32200, 32200, 32200, 0, 0, 32151, 32151, 32151, 32151, 32151, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32199, 32199, 32199, 32199, 32199, 32199, 32199, 32140, 32140, 32140, 0, 0, 32160, 32160, 32160, 32160, 32160, 0, 0, 32170, 32170, 32170, 32170, 32170, 32170, 32170, 32170, 32170, 32180, 32180, 32180, 0, 0, 0, 0, 32200, 32200, 0, 0, 0, 0, 0, 32151, 32151, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32138, 32138, 32138, 32138, 0, 32148, 32148, 32148, 0, 0, 32158, 32158, 32158, 32158, 0, 32168, 32168, 32168, 0, 0, 32178, 32178, 32178, 32178, 32178, 32178, 32178, 32178, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32199, 32199, 32199, 32199, 32199, 32199, 32199, 32140, 32140, 32140, 0, 0, 0, 0, 32160, 32160, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32170, 32170, 0, 0, 0, 0, 0, 0, 32190, 32190, 32190, 32190, 32190, 0, 0, 32141, 32141, 32141, 32141, 32141, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32138, 32138, 32138, 32138, 0, 32148, 32148, 32148, 0, 0, 32158, 32158, 32158, 32158, 0, 32168, 32168, 32168, 0, 0, 32178, 32178, 32178, 32178, 32178, 32178, 32178, 32178, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32199, 32199, 0, 0, 0, 0, 0, 0, 32150, 32150, 32150, 32150, 32150, 0, 0, 0, 0, 0, 0, 0, 0, 32170, 32170, 0, 0, 0, 0, 0, 0, 32190, 32190, 32190, 32190, 32190, 0, 0, 32141, 32141, 32141, 32141, 32141, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32138, 32138, 32138, 32138, 0, 32148, 32148, 32148, 0, 0, 32158, 32158, 32158, 32158, 0, 32168, 32168, 32168, 0, 0, 32178, 32178, 32178, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32199, 32199, 0, 0, 0, 0, 0, 0, 32150, 32150, 32150, 32150, 32150, 0, 0, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 0, 0, 32190, 32190, 32190, 32190, 32190, 0, 0, 32141, 32141, 32141, 32141, 32141, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 0, 0, 32191, 32191, 32191, 32191, 32191, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32138, 32138, 32138, 32138, 0, 32148, 32148, 32148, 0, 0, 32158, 32158, 32158, 32158, 0, 32168, 32168, 32168, 0, 0, 32178, 32178, 32178, 32188, 32188, 32188, 32188, 32188, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 0, 0, 32150, 32150, 32150, 32150, 32150, 0, 0, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 0, 0, 32190, 32190, 32190, 32190, 32190, 0, 0, 32141, 32141, 32141, 32141, 32141, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 0, 0, 32191, 32191, 32191, 32191, 32191, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32138, 32138, 32138, 32138, 32138, 32148, 32148, 32148, 0, 0, 32158, 32158, 32158, 32158, 32158, 32168, 32168, 32168, 0, 0, 32178, 32178, 32178, 32188, 32188, 32188, 32188, 32188, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 0, 0, 32150, 32150, 32150, 32150, 32150, 0, 0, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 0, 0, 32190, 32190, 32190, 32190, 32190, 0, 0, 32141, 32141, 32141, 32141, 32141, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 0, 0, 32191, 32191, 32191, 32191, 32191, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32138, 32138, 32138, 32138, 32138, 32148, 32148, 32148, 0, 0, 32158, 32158, 32158, 32158, 32158, 32168, 32168, 32168, 0, 0, 32178, 32178, 32178, 32188, 32188, 32188, 32188, 32188, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 0, 0, 32150, 32150, 32150, 32150, 32150, 0, 0, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 0, 0, 32190, 32190, 32190, 32190, 32190, 0, 0, 32141, 32141, 32141, 32141, 32141, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 0, 0, 32191, 32191, 32191, 32191, 32191, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32138, 32138, 0, 0, 0, 0, 0, 0, 0, 0, 32158, 32158, 0, 0, 0, 0, 0, 0, 0, 0, 32178, 32178, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 0, 0, 32150, 32150, 32150, 32150, 32150, 0, 0, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 0, 0, 32190, 32190, 32190, 32190, 32190, 0, 0, 32141, 32141, 32141, 32141, 32141, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 0, 0, 32191, 32191, 32191, 32191, 32191, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32138, 32138, 0, 0, 0, 0, 0, 0, 0, 0, 32158, 32158, 0, 0, 0, 0, 0, 0, 0, 0, 32178, 32178, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 0, 0, 32150, 32150, 32150, 32150, 32150, 0, 0, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 0, 0, 32190, 32190, 32190, 32190, 32190, 0, 0, 32141, 32141, 32141, 32141, 32141, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 0, 0, 32191, 32191, 32191, 32191, 32191, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32189, 32189, 32189, 32189, 32189, 32189, 32189, 32189, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 0, 0, 32150, 32150, 32150, 32150, 32150, 0, 0, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 0, 0, 32190, 32190, 32190, 32190, 32190, 0, 0, 32141, 32141, 32141, 32141, 32141, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 0, 0, 32191, 32191, 32191, 32191, 32191, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32179, 32179, 32179, 32179, 32179, 32179, 32189, 32189, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 32161, 0, 0, 32150, 32150, 32150, 32150, 32150, 0, 0, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 32171, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 0, 0, 32191, 32191, 32191, 32191, 32191, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32198, 32198, 32198, 32198, 32198, 32198, 32139, 32139, 0, 0, 32149, 32149, 32149, 32149, 32149, 32149, 32159, 32159, 0, 32179, 32179, 32179, 32179, 32179, 32179, 32179, 32189, 32189, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 32181, 0, 0, 32191, 32191, 32191, 32191, 32191, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32198, 32198, 32198, 32198, 32198, 32198, 32139, 32139, 0, 0, 32149, 32149, 32149, 32149, 32149, 32149, 32159, 32159, 0, 32179, 32179, 32179, 32179, 32179, 32179, 32179, 32189, 32189, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 0, 0, 0, 0, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32198, 32198, 32198, 32198, 0, 0, 32139, 32139, 0, 0, 32149, 32149, 32149, 32149, 0, 0, 32159, 32159, 0, 32179, 32179, 32179, 32179, 32179, 32179, 32179, 32189, 32189, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32181, 32181, 0, 0, 0, 0, 32191, 32191, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32198, 32198, 32198, 32198, 0, 0, 32139, 32139, 0, 0, 32149, 32149, 32149, 32149, 0, 0, 32159, 32159, 0, 32179, 32179, 32179, 32179, 32179, 32179, 32179, 32179, 32179, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32198, 32198, 32198, 32198, 0, 32139, 32139, 32139, 0, 0, 32149, 32149, 32149, 32149, 0, 32159, 32159, 32159, 0, 32179, 32179, 32179, 32179, 32179, 32179, 32179, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 0, 0, 0, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32198, 32198, 32198, 32198, 0, 32139, 32139, 32139, 0, 0, 32149, 32149, 32149, 32149, 0, 32159, 32159, 32159, 0, 32179, 32179, 32179, 32179, 32179, 32179, 32179, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 0, 0, 0, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32179, 32179, 32179, 32179, 32179, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 0, 0, 0, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32179, 32179, 32179, 32179, 32179, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 32201, 0, 0, 0, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32183, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 0, 0, 0, 0, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 32183, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32172, 32172, 32172, 32172, 32172, 32172, 32172, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32173, 32173, 32173, 32173, 32173, 32173, 32173, 0, 32172, 32172, 32172, 32172, 32172, 32172, 32172, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32163, 32163, 32163, 32163, 32163, 32163, 32173, 32173, 0, 32172, 32172, 32172, 32172, 32172, 32172, 32172, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32163, 32163, 32163, 32163, 32163, 32163, 32173, 32173, 0, 32172, 32172, 32172, 32172, 32172, 32172, 32172, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32163, 32163, 32163, 32163, 32163, 32163, 32173, 32173, 0, 32172, 32172, 32172, 32172, 32172, 32172, 32172, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32143, 32143, 32143, 32143, 32143, 32143, 0, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32173, 32173, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32143, 32143, 32143, 32143, 32143, 32143, 0, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32142, 32142, 0, 0, 0, 0, 32142, 32142, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32143, 32143, 32143, 32143, 32143, 32143, 0, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32153, 32153, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32143, 32143, 32143, 32143, 32143, 32143, 0, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 0, 32182, 32182, 0, 0, 32182, 32182, 0, 0, 32153, 32153, 32153, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 32152, 0, 0, 0, 0, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 32162, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32143, 32143, 0, 0, 0, 0, 0, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 0, 32182, 32182, 32182, 32182, 32182, 32182, 0, 0, 32153, 32153, 32153, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 32163, 0, 32182, 32182, 32182, 32182, 32182, 32182, 0, 0, 32153, 32153, 32153, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32182, 32182, 32182, 32182, 32182, 32182, 0, 0, 32153, 32153, 32153, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32202, 32202, 0, 0, 32202, 32202, 0, 0, 32192, 32192, 0, 0, 32192, 32192, 0, 0, 32182, 32182, 32182, 32182, 32182, 32182, 0, 0, 32153, 32153, 32153, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32202, 32202, 32202, 32202, 32202, 32202, 0, 0, 32192, 32192, 32192, 32192, 32192, 32192, 0, 0, 32182, 32182, 32182, 32182, 32182, 32182, 0, 0, 32153, 32153, 32153, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32193, 32193, 0, 0, 32193, 0, 0, 0, 32174, 32174, 0, 0, 32174, 0, 0, 0, 32194, 32194, 0, 0, 32194, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32202, 32202, 32202, 32202, 32202, 32202, 0, 0, 32192, 32192, 32192, 32192, 32192, 32192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32193, 32193, 32193, 32193, 32193, 0, 0, 0, 32174, 32174, 32174, 32174, 32174, 0, 0, 0, 32194, 32194, 32194, 32194, 32194, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32202, 32202, 32202, 32202, 32202, 32202, 0, 0, 32192, 32192, 32192, 32192, 32192, 32192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32193, 32193, 32193, 32193, 32193, 0, 0, 0, 32174, 32174, 32174, 32174, 32174, 0, 0, 0, 32194, 32194, 32194, 32194, 32194, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32202, 32202, 32202, 32202, 32202, 32202, 0, 0, 32192, 32192, 32192, 32192, 32192, 32192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32193, 32193, 32193, 32193, 32193, 0, 0, 0, 32174, 32174, 32174, 32174, 32174, 0, 0, 0, 32194, 32194, 32194, 32194, 32194, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32202, 32202, 32202, 32202, 32202, 32202, 0, 0, 32192, 32192, 32192, 32192, 32192, 32192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32193, 32193, 32193, 32193, 32193, 0, 0, 0, 32174, 32174, 32174, 32174, 32174, 0, 0, 0, 32194, 32194, 32194, 32194, 32194, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32202, 32202, 32202, 32202, 32202, 32202, 0, 0, 32192, 32192, 32192, 32192, 32192, 32192, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32193, 32193, 32193, 32193, 32193, 0, 0, 0, 32174, 32174, 32174, 32174, 32174, 0, 0, 0, 32194, 32194, 32194, 32194, 32194, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32193, 32193, 32193, 32193, 32193, 0, 0, 0, 32174, 32174, 32174, 32174, 32174, 0, 0, 0, 32194, 32194, 32194, 32194, 32194, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32193, 32193, 32193, 32193, 32193, 0, 0, 0, 32174, 32174, 32174, 32174, 32174, 0, 0, 0, 32194, 32194, 32194, 32194, 32194, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32225, 32225, 32225, 32225, 32225, 32225, 32235, 0, 0, 32245, 32245, 32245, 0, 0, 0, 0, 0, 0, 32206, 32206, 32206, 32206, 32206, 32206, 32216, 0, 0, 32226, 32226, 32226, 0, 0, 0, 0, 0, 0, 32266, 32266, 32266, 32266, 32266, 32266, 32276, 0, 0, 32207, 32207, 32207, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32203, 32203, 0, 0, 0, 0, 0, 0, 32184, 32184, 0, 0, 0, 0, 0, 0, 32204, 32204, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32225, 32225, 32225, 32225, 32225, 32225, 32235, 0, 0, 32245, 32245, 32245, 0, 0, 0, 0, 0, 0, 32206, 32206, 32206, 32206, 32206, 32206, 32216, 0, 0, 32226, 32226, 32226, 0, 0, 0, 0, 0, 0, 32266, 32266, 32266, 32266, 32266, 32266, 32276, 0, 0, 32207, 32207, 32207, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32203, 32203, 32203, 32203, 32203, 0, 0, 0, 32184, 32184, 32184, 32184, 32184, 0, 0, 0, 32204, 32204, 32204, 32204, 32204, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32225, 32225, 32225, 32225, 32225, 32225, 32235, 0, 0, 32245, 32245, 32245, 0, 0, 0, 0, 0, 0, 32206, 32206, 32206, 32206, 32206, 32206, 32216, 0, 0, 32226, 32226, 32226, 0, 0, 0, 0, 0, 0, 32266, 32266, 32266, 32266, 32266, 32266, 32276, 0, 0, 32207, 32207, 32207, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32203, 32203, 32203, 32203, 32203, 0, 0, 0, 32184, 32184, 32184, 32184, 32184, 0, 0, 0, 32204, 32204, 32204, 32204, 32204, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32225, 32225, 32225, 32225, 32225, 32225, 32235, 0, 0, 32245, 32245, 32245, 0, 0, 0, 0, 0, 0, 32206, 32206, 32206, 32206, 32206, 32206, 32216, 0, 0, 32226, 32226, 32226, 0, 0, 0, 0, 0, 0, 32266, 32266, 32266, 32266, 32266, 32266, 32276, 0, 0, 32207, 32207, 32207, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32203, 32203, 32203, 32203, 32203, 0, 0, 0, 32184, 32184, 32184, 32184, 32184, 0, 0, 0, 32204, 32204, 32204, 32204, 32204, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32225, 32225, 32225, 32225, 32225, 32225, 32235, 32235, 0, 0, 32245, 32245, 0, 0, 0, 0, 0, 0, 32206, 32206, 32206, 32206, 32206, 32206, 32216, 32216, 0, 0, 32226, 32226, 0, 0, 0, 0, 0, 0, 32266, 32266, 32266, 32266, 32266, 32266, 32276, 32276, 0, 0, 32207, 32207, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32203, 32203, 32203, 32203, 32203, 0, 0, 0, 32184, 32184, 32184, 32184, 32184, 0, 0, 0, 32204, 32204, 32204, 32204, 32204, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32225, 32225, 32225, 32225, 32225, 32225, 32225, 32225, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32206, 32206, 32206, 32206, 32206, 32206, 32206, 32206, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32266, 32266, 32266, 32266, 32266, 32266, 32266, 32266, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32225, 32225, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32205, 32205, 32205, 32205, 32205, 0, 32215, 32215, 32215, 32215, 32215, 32215, 0, 0, 0, 0, 0, 0, 32265, 32265, 32265, 32265, 32265, 0, 32275, 32275, 32275, 32275, 32275, 32275, 0, 0, 0, 0, 0, 0, 32246, 32246, 32246, 32246, 32246, 0, 32256, 32256, 32256, 32256, 32256, 32256, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32205, 32205, 32205, 32205, 32205, 0, 32215, 32215, 32215, 32215, 32215, 32215, 0, 0, 0, 0, 0, 0, 32265, 32265, 32265, 32265, 32265, 0, 32275, 32275, 32275, 32275, 32275, 32275, 0, 0, 0, 0, 0, 0, 32246, 32246, 32246, 32246, 32246, 0, 32256, 32256, 32256, 32256, 32256, 32256, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32205, 32205, 32205, 32205, 32205, 0, 32215, 32215, 32215, 32215, 32215, 32215, 0, 0, 0, 0, 0, 0, 32265, 32265, 32265, 32265, 32265, 0, 32275, 32275, 32275, 32275, 32275, 32275, 0, 0, 0, 0, 0, 0, 32246, 32246, 32246, 32246, 32246, 0, 32256, 32256, 32256, 32256, 32256, 32256, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 0, 0, 0, 0, 0, 0, 0, 0, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 0, 0, 0, 0, 0, 0, 0, 0, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 0, 0, 0, 0, 0, 0, 0, 0, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 0, 0, 0, 0, 0, 0, 0, 0, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 0, 0, 0, 0, 0, 0, 0, 0, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 0, 0, 0, 0, 0, 0, 0, 0, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 0, 0, 0, 0, 0, 0, 0, 0, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 0, 0, 0, 0, 0, 0, 0, 0, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 32255, 0, 0, 0, 0, 0, 0, 0, 0, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 32236, 0, 0, 0, 0, 0, 0, 0, 0, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 32217, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/maze/collision_maze.csv b/tests/data/environment/stanford_town/the_ville/matrix/maze/collision_maze.csv new file mode 100644 index 000000000..40329a8c4 --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/maze/collision_maze.csv @@ -0,0 +1 @@ +0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 32125, 0, 32125, 0, 0, 32125, 0, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 32125, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 32125, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 0, 32125, 0, 0, 0, 0, 32125, 32125, 0, 32125, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 32125, 0, 32125, 32125, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 32125, 32125, 0, 32125, 32125, 0, 32125, 32125, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 32125, 32125, 32125, 32125, 0, 0, 32125, 32125, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 32125, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 32125, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/maze/game_object_maze.csv b/tests/data/environment/stanford_town/the_ville/matrix/maze/game_object_maze.csv new file mode 100644 index 000000000..9c97dc7bd --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/maze/game_object_maze.csv @@ -0,0 +1 @@ +0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32208, 32208, 32277, 32277, 32218, 0, 0, 32208, 32208, 32277, 32277, 32218, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32227, 32227, 32238, 32247, 32257, 32257, 0, 32208, 32208, 32218, 0, 0, 32208, 32208, 32277, 32277, 32218, 0, 0, 32227, 32227, 0, 32238, 32247, 32247, 32257, 32257, 0, 32208, 32208, 32218, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32209, 0, 0, 0, 0, 0, 0, 0, 32277, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32237, 0, 0, 0, 0, 0, 0, 0, 0, 32277, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32237, 0, 0, 0, 0, 0, 0, 32277, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32227, 0, 0, 0, 32208, 32208, 32218, 0, 0, 32227, 32227, 0, 0, 0, 32208, 32208, 32218, 0, 0, 0, 32227, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32227, 32227, 0, 0, 32238, 0, 0, 32227, 32227, 0, 0, 32238, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32227, 32227, 0, 0, 32238, 0, 0, 32238, 32238, 0, 32258, 32258, 32228, 32249, 32249, 32249, 32249, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32237, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32238, 32238, 32239, 32239, 32239, 32228, 32258, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32279, 32279, 32279, 32279, 0, 0, 0, 0, 0, 0, 32240, 32240, 0, 32250, 32250, 32250, 32250, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32257, 0, 0, 0, 0, 32277, 0, 0, 0, 0, 0, 0, 32209, 0, 0, 32277, 0, 0, 0, 0, 0, 0, 32277, 32277, 32218, 32208, 32208, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32259, 0, 32259, 0, 32259, 0, 0, 0, 0, 32237, 0, 32228, 0, 0, 0, 0, 32237, 0, 32228, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32270, 0, 0, 32270, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32257, 0, 0, 0, 0, 32277, 0, 0, 0, 32267, 32267, 0, 0, 0, 0, 32277, 0, 0, 0, 0, 32237, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32228, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32260, 0, 0, 32260, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32229, 0, 32229, 0, 32229, 0, 32229, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32259, 0, 32259, 0, 32259, 0, 0, 32259, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32280, 0, 0, 0, 32270, 0, 0, 32270, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32220, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32279, 32279, 0, 32247, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32260, 0, 0, 32260, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32238, 32238, 0, 0, 0, 32248, 32228, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32257, 0, 0, 0, 0, 0, 0, 32219, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 32259, 0, 32259, 0, 32259, 0, 0, 32259, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32270, 0, 0, 32270, 0, 0, 0, 0, 0, 0, 0, 32250, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32257, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32269, 32269, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32250, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32250, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32257, 0, 0, 0, 0, 0, 0, 32218, 0, 0, 0, 32267, 32267, 0, 0, 0, 0, 32218, 0, 0, 0, 32268, 0, 0, 32268, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 32247, 0, 0, 0, 32277, 0, 0, 0, 32247, 32247, 0, 0, 0, 0, 32277, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32277, 0, 0, 0, 0, 32237, 0, 0, 0, 0, 32277, 0, 0, 0, 32278, 32268, 0, 0, 32268, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32267, 32267, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32278, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32227, 0, 0, 0, 32208, 32208, 0, 0, 0, 0, 32227, 0, 0, 0, 32208, 32208, 0, 0, 0, 32278, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 0, 0, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32278, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 0, 0, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 0, 0, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 0, 0, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 0, 0, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 32222, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32252, 32252, 32252, 0, 0, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 32252, 32252, 32252, 0, 0, 32252, 32252, 32252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32271, 32271, 32271, 32271, 32271, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32281, 32281, 32281, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32221, 32221, 32221, 32221, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32231, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32248, 32228, 32238, 32238, 0, 0, 0, 0, 0, 0, 0, 32247, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32252, 32252, 32252, 0, 0, 32252, 32252, 32252, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32212, 32212, 32212, 0, 0, 0, 0, 0, 0, 32231, 0, 0, 32231, 0, 0, 0, 0, 0, 0, 0, 0, 32261, 32261, 32261, 32261, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 32227, 32227, 32210, 32210, 0, 32237, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32231, 0, 0, 32231, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32268, 0, 0, 32268, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32252, 32252, 0, 32252, 32252, 0, 32252, 32252, 0, 0, 0, 0, 32252, 32252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32271, 32271, 32271, 32271, 32271, 32271, 32271, 32271, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32231, 0, 0, 32231, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32231, 0, 0, 0, 0, 32241, 32241, 32241, 32241, 32241, 32241, 0, 32241, 32241, 32241, 32241, 32241, 32241, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32268, 0, 0, 32268, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 32252, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32208, 32208, 0, 32277, 32277, 32218, 0, 0, 32278, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32211, 0, 0, 32251, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32278, 0, 0, 0, 32282, 32282, 32282, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32271, 32271, 32271, 32271, 32271, 32271, 32271, 32271, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32211, 0, 0, 32251, 0, 0, 32241, 32241, 32241, 32241, 32241, 32241, 0, 0, 0, 32241, 32241, 32241, 32241, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32278, 0, 0, 32282, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32211, 0, 0, 32251, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32282, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32218, 32277, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32282, 0, 0, 0, 0, 32247, 32247, 0, 0, 32230, 32230, 0, 0, 0, 32277, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32237, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32279, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32237, 0, 0, 0, 0, 0, 0, 32257, 32257, 0, 0, 32237, 0, 0, 0, 0, 32227, 0, 0, 0, 32279, 0, 0, 32208, 32208, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32272, 32272, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 32237, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 32247, 0, 32247, 0, 0, 0, 0, 32247, 32247, 0, 0, 0, 0, 0, 0, 32247, 32247, 0, 0, 32278, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 32237, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32210, 32210, 0, 0, 0, 0, 32247, 0, 0, 32227, 32227, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32237, 0, 32228, 0, 0, 0, 0, 0, 32257, 0, 32228, 0, 0, 0, 0, 0, 0, 0, 32228, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32227, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32257, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32227, 32227, 0, 32238, 0, 0, 0, 0, 32227, 32227, 0, 32238, 0, 0, 0, 0, 32227, 32227, 0, 32238, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32257, 32257, 32257, 0, 0, 0, 32228, 0, 0, 32218, 32277, 32277, 0, 0, 0, 0, 0, 0, 32257, 32257, 0, 0, 0, 0, 32228, 0, 0, 32218, 32277, 32277, 0, 0, 0, 0, 0, 0, 32257, 32257, 32257, 0, 0, 0, 32228, 0, 0, 32218, 32277, 32277, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32262, 32262, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32258, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32268, 0, 0, 32268, 0, 0, 0, 0, 32208, 32208, 0, 0, 0, 0, 0, 0, 0, 0, 32268, 0, 0, 32268, 0, 0, 0, 0, 32208, 32208, 0, 0, 0, 0, 0, 0, 0, 0, 32268, 0, 0, 32268, 0, 0, 0, 0, 32208, 32208, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32277, 32277, 0, 0, 32218, 0, 0, 0, 32277, 32277, 0, 0, 32218, 0, 0, 0, 32277, 32277, 0, 0, 32218, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32238, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32238, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32238, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32268, 0, 0, 32268, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32268, 0, 0, 32268, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32268, 0, 0, 32268, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32208, 32208, 0, 0, 0, 0, 0, 0, 32208, 32208, 0, 0, 0, 0, 0, 0, 32208, 32208, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32237, 32227, 32227, 0, 32227, 32227, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32237, 32227, 32227, 0, 32227, 32227, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32237, 32227, 32227, 0, 32227, 32227, 32237, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32209, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32209, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32247, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 32232, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 0, 0, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32242, 0, 0, 0, 32232, 0, 0, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 0, 0, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 0, 0, 0, 0, 32232, 32232, 0, 0, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 0, 0, 32232, 0, 0, 32232, 0, 0, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 0, 0, 0, 0, 32232, 32232, 0, 0, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 32232, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/maze/sector_maze.csv b/tests/data/environment/stanford_town/the_ville/matrix/maze/sector_maze.csv new file mode 100644 index 000000000..165b7b03a --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/maze/sector_maze.csv @@ -0,0 +1 @@ +0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32165, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32145, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32165, 32165, 32136, 32136, 32136, 32136, 32136, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32195, 32195, 32195, 32195, 32195, 32145, 32145, 32195, 32195, 32195, 32195, 32195, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32165, 32165, 32136, 32136, 32136, 32136, 32136, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32195, 32195, 32195, 32195, 32195, 32145, 32145, 32195, 32195, 32195, 32195, 32195, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32175, 32175, 32175, 32175, 32175, 32175, 32175, 32185, 32185, 32185, 32185, 32185, 32185, 32185, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32195, 32155, 32155, 32155, 32155, 32155, 32155, 32155, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 32136, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 32146, 0, 0, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32135, 32135, 32135, 32135, 32135, 32135, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 32156, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 32166, 0, 0, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 32176, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 0, 0, 32186, 32186, 0, 32196, 32196, 32196, 0, 0, 32196, 32196, 0, 32137, 32137, 32137, 0, 0, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 32177, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32186, 32186, 32186, 32186, 32186, 32186, 32186, 0, 32196, 32196, 32196, 32196, 32196, 32196, 32196, 0, 32137, 32137, 32137, 32137, 32137, 32137, 32137, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 32147, 0, 0, 0, 0, 0, 0, 0, 0, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 32157, 0, 0, 0, 0, 0, 0, 0, 0, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 32167, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/maze/spawning_location_maze.csv b/tests/data/environment/stanford_town/the_ville/matrix/maze/spawning_location_maze.csv new file mode 100644 index 000000000..6d7ca5727 --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/maze/spawning_location_maze.csv @@ -0,0 +1 @@ +0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32306, 32316, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32307, 32317, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32285, 0, 0, 0, 0, 0, 0, 0, 0, 32295, 32305, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32315, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32288, 32298, 0, 0, 0, 0, 0, 32308, 32318, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32287, 32297, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32286, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32296, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32313, 32323, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32294, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32304, 32314, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32324, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32289, 32299, 0, 0, 0, 0, 0, 0, 32309, 32319, 0, 0, 0, 0, 0, 0, 32290, 32300, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32310, 32320, 0, 32291, 32301, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32311, 32321, 0, 32292, 32302, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32312, 32322, 0, 32293, 32303, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/maze_meta_info.json b/tests/data/environment/stanford_town/the_ville/matrix/maze_meta_info.json new file mode 100644 index 000000000..32a15fbb6 --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/maze_meta_info.json @@ -0,0 +1,5 @@ +{"world_name": "the ville", + "maze_width": 140, + "maze_height": 100, + "sq_tile_size": 32, + "special_constraint": ""} \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/arena_blocks.csv b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/arena_blocks.csv new file mode 100644 index 000000000..c92e0c6ab --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/arena_blocks.csv @@ -0,0 +1,63 @@ +32138, the Ville, artist's co-living space, Latoya Williams's room +32148, the Ville, artist's co-living space, Latoya Williams's bathroom +32158, the Ville, artist's co-living space, Rajiv Patel's room +32168, the Ville, artist's co-living space, Rajiv Patel's bathroom +32178, the Ville, artist's co-living space, Abigail Chen's room +32188, the Ville, artist's co-living space, Abigail Chen's bathroom +32198, the Ville, artist's co-living space, Francisco Lopez's room +32139, the Ville, artist's co-living space, Francisco Lopez's bathroom +32149, the Ville, artist's co-living space, Hailey Johnson's room +32159, the Ville, artist's co-living space, Hailey Johnson's bathroom +32179, the Ville, artist's co-living space, common room +32189, the Ville, artist's co-living space, kitchen +32199, the Ville, Arthur Burton's apartment, main room +32140, the Ville, Arthur Burton's apartment, bathroom +32150, the Ville, Ryan Park's apartment, main room +32160, the Ville, Ryan Park's apartment, bathroom +32170, the Ville, Isabella Rodriguez's apartment, main room +32180, the Ville, Isabella Rodriguez's apartment, bathroom +32190, the Ville, Giorgio Rossi's apartment, main room +32200, the Ville, Giorgio Rossi's apartment, bathroom +32141, the Ville, Carlos Gomez's apartment, main room +32151, the Ville, Carlos Gomez's apartment, bathroom +32161, the Ville, The Rose and Crown Pub, pub +32171, the Ville, Hobbs Cafe, cafe +32181, the Ville, Oak Hill College, classroom +32191, the Ville, Oak Hill College, library +32201, the Ville, Oak Hill College, hallway +32142, the Ville, Johnson Park, park +32152, the Ville, Harvey Oak Supply Store, supply store +32162, the Ville, The Willows Market and Pharmacy, store +32193, the Ville, Adam Smith's house, main room +32203, the Ville, Adam Smith's house, bathroom +32174, the Ville, Yuriko Yamamoto's house, main room +32184, the Ville, Yuriko Yamamoto's house, bathroom +32194, the Ville, Moore family's house, main room +32204, the Ville, Moore family's house, bathroom +32172, the Ville, Dorm for Oak Hill College, Klaus Mueller's room +32182, the Ville, Dorm for Oak Hill College, Maria Lopez's room +32192, the Ville, Dorm for Oak Hill College, Ayesha Khan's room +32202, the Ville, Dorm for Oak Hill College, Wolfgang Schulz's room +32143, the Ville, Dorm for Oak Hill College, man's bathroom +32153, the Ville, Dorm for Oak Hill College, woman's bathroom +32163, the Ville, Dorm for Oak Hill College, common room +32173, the Ville, Dorm for Oak Hill College, kitchen +32183, the Ville, Dorm for Oak Hill College, garden +32205, the Ville, Tamara Taylor and Carmen Ortiz's house, Tamara Taylor's room +32215, the Ville, Tamara Taylor and Carmen Ortiz's house, Carmen Ortiz's room +32225, the Ville, Tamara Taylor and Carmen Ortiz's house, common room +32235, the Ville, Tamara Taylor and Carmen Ortiz's house, kitchen +32245, the Ville, Tamara Taylor and Carmen Ortiz's house, bathroom +32255, the Ville, Tamara Taylor and Carmen Ortiz's house, garden +32265, the Ville, Moreno family's house, Tom and Jane Moreno's bedroom +32275, the Ville, Moreno family's house, empty bedroom +32206, the Ville, Moreno family's house, common room +32216, the Ville, Moreno family's house, kitchen +32226, the Ville, Moreno family's house, bathroom +32236, the Ville, Moreno family's house, garden +32246, the Ville, Lin family's house, Mei and John Lin's bedroom +32256, the Ville, Lin family's house, Eddy Lin's bedroom +32266, the Ville, Lin family's house, common room +32276, the Ville, Lin family's house, kitchen +32207, the Ville, Lin family's house, bathroom +32217, the Ville, Lin family's house, garden \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/game_object_blocks.csv b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/game_object_blocks.csv new file mode 100644 index 000000000..4afc74b7a --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/game_object_blocks.csv @@ -0,0 +1,46 @@ +32227, the Ville, , bed +32237, the Ville, , desk +32247, the Ville, , closet +32257, the Ville, , shelf +32267, the Ville, , easel +32277, the Ville, , bathroom sink +32208, the Ville, , shower +32218, the Ville, , toilet +32228, the Ville, , kitchen sink +32238, the Ville, , refrigerator +32248, the Ville, , toaster +32258, the Ville, , cooking area +32268, the Ville, , common room table +32278, the Ville, , common room sofa +32209, the Ville, , guitar +32219, the Ville, , microphone +32229, the Ville, , bar customer seating +32239, the Ville, , behind the bar counter +32249, the Ville, , behind the cafe counter +32259, the Ville, , cafe customer seating +32269, the Ville, , piano +32279, the Ville, , blackboard +32210, the Ville, , game console +32220, the Ville, , computer desk +32230, the Ville, , computer +32240, the Ville, , library sofa +32250, the Ville, , bookshelf +32260, the Ville, , library table +32270, the Ville, , classroom student seating +32280, the Ville, , classroom podium +32211, the Ville, , behind the pharmacy counter +32221, the Ville, , behind the grocery counter +32231, the Ville, , pharmacy store shelf +32241, the Ville, , grocery store shelf +32251, the Ville, , pharmacy store counter +32261, the Ville, , grocery store counter +32271, the Ville, , supply store product shelf +32281, the Ville, , behind the supply store counter +32212, the Ville, , supply store counter +32222, the Ville, , dorm garden +32232, the Ville, , house garden +32242, the Ville, , garden chair +32252, the Ville, , park garden +32262, the Ville, , harp +32272, the Ville, , lifting weight +32282, the Ville, , pool table \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/sector_blocks.csv b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/sector_blocks.csv new file mode 100644 index 000000000..ba09c4c35 --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/sector_blocks.csv @@ -0,0 +1,19 @@ +32135, the Ville, artist's co-living space +32145, the Ville, Arthur Burton's apartment +32155, the Ville, Ryan Park's apartment +32165, the Ville, Isabella Rodriguez's apartment +32175, the Ville, Giorgio Rossi's apartment +32185, the Ville, Carlos Gomez's apartment +32195, the Ville, The Rose and Crown Pub +32136, the Ville, Hobbs Cafe +32146, the Ville, Oak Hill College +32156, the Ville, Johnson Park +32166, the Ville, Harvey Oak Supply Store +32176, the Ville, The Willows Market and Pharmacy +32186, the Ville, Adam Smith's house +32196, the Ville, Yuriko Yamamoto's house +32137, the Ville, Moore family's house +32147, the Ville, Tamara Taylor and Carmen Ortiz's house +32157, the Ville, Moreno family's house +32167, the Ville, Lin family's house +32177, the Ville, Dorm for Oak Hill College \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/spawning_location_blocks.csv b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/spawning_location_blocks.csv new file mode 100644 index 000000000..564fc76b4 --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/spawning_location_blocks.csv @@ -0,0 +1,40 @@ +32285, the Ville, artist's co-living space, Latoya Williams's room, sp-A +32295, the Ville, artist's co-living space, Rajiv Patel's room, sp-A +32305, the Ville, artist's co-living space, Rajiv Patel's room, sp-B +32315, the Ville, artist's co-living space, Abigail Chen's room, sp-A +32286, the Ville, artist's co-living space, Francisco Lopez's room, sp-A +32296, the Ville, artist's co-living space, Hailey Johnson's room, sp-A +32306, the Ville, Arthur Burton's apartment, main room, sp-A +32316, the Ville, Arthur Burton's apartment, main room, sp-B +32287, the Ville, Ryan Park's apartment, main room, sp-A +32297, the Ville, Ryan Park's apartment, main room, sp-B +32307, the Ville, Isabella Rodriguez's apartment, main room, sp-A +32317, the Ville, Isabella Rodriguez's apartment, main room, sp-B +32288, the Ville, Giorgio Rossi's apartment, main room, sp-A +32298, the Ville, Giorgio Rossi's apartment, main room, sp-B +32308, the Ville, Carlos Gomez's apartment, main room, sp-A +32318, the Ville, Carlos Gomez's apartment, main room, sp-B +32289, the Ville, Adam Smith's house, main room, sp-A +32299, the Ville, Adam Smith's house, main room, sp-B +32309, the Ville, Yuriko Yamamoto's house, main room, sp-A +32319, the Ville, Yuriko Yamamoto's house, main room, sp-B +32290, the Ville, Moore family's house, main room, sp-A +32300, the Ville, Moore family's house, main room, sp-B +32310, the Ville, Tamara Taylor and Carmen Ortiz's house, Tamara Taylor's room, sp-A +32320, the Ville, Tamara Taylor and Carmen Ortiz's house, Tamara Taylor's room, sp-B +32291, the Ville, Tamara Taylor and Carmen Ortiz's house, Carmen Ortiz's room, sp-A +32301, the Ville, Tamara Taylor and Carmen Ortiz's house, Carmen Ortiz's room, sp-B +32311, the Ville, Moreno family's house, Tom and Jane Moreno's bedroom, sp-A +32321, the Ville, Moreno family's house, Tom and Jane Moreno's bedroom, sp-B +32292, the Ville, Moreno family's house, empty bedroom, sp-A +32302, the Ville, Moreno family's house, empty bedroom, sp-B +32312, the Ville, Lin family's house, Mei and John Lin's bedroom, sp-A +32322, the Ville, Lin family's house, Mei and John Lin's bedroom, sp-B +32293, the Ville, Lin family's house, Eddy Lin's bedroom, sp-A +32303, the Ville, Lin family's house, Eddy Lin's bedroom, sp-B +32313, the Ville, Dorm for Oak Hill College, Klaus Mueller's room, sp-A +32323, the Ville, Dorm for Oak Hill College, Klaus Mueller's room, sp-B +32294, the Ville, Dorm for Oak Hill College, Maria Lopez's room, sp-A +32304, the Ville, Dorm for Oak Hill College, Ayesha Khan's room, sp-A +32314, the Ville, Dorm for Oak Hill College, Ayesha Khan's room, sp-B +32324, the Ville, Dorm for Oak Hill College, Wolfgang Schulz's room, sp-A \ No newline at end of file diff --git a/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/world_blocks.csv b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/world_blocks.csv new file mode 100644 index 000000000..b04d990bb --- /dev/null +++ b/tests/data/environment/stanford_town/the_ville/matrix/special_blocks/world_blocks.csv @@ -0,0 +1 @@ +32134, the Ville \ No newline at end of file From 3648669bd248bf780e70ad26dc383cf2191bba9a Mon Sep 17 00:00:00 2001 From: yzlin Date: Sun, 4 Feb 2024 10:22:23 +0800 Subject: [PATCH 594/637] fix test_ut_writer failure due to aask_code mock --- tests/data/rsp_cache.json | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index 6ee41e076..2257878e2 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -383,5 +383,11 @@ "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nSolve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\n[user: Tested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not correctly sort the array based on the number of ones in their binary representation followed by their decimal value. The error in the output indicates that the numbers 3 and 4 are not in the expected positions. The binary representations of 3 and 4 are '11' and '100', respectively, which means 3 has more ones than 4 and should come before 4 in the sorted array. To fix this issue, we need to ensure that the sorting is stable so that when two numbers have the same number of ones, they retain their original order if their decimal values are the same. We can achieve this by using the 'stable' sorting algorithm provided by Python's sorted function. Additionally, we need to handle negative numbers correctly by taking the absolute value before counting the ones in their binary representation.", "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(abs(n)).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n # Using 'stable' sorting algorithm to maintain the original order when necessary\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x), reverse=False)\n \n return sorted_arr" + }, + "[{\"role\": \"user\", \"content\": \"Interface definition:\\n```text\\nInterface Name: Element Tagging\\nInterface Path: /projects/{project_key}/node-tags\\nMethod: POST\\n\\nRequest parameters:\\nPath parameters:\\nproject_key\\n\\nBody parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nnodes\\tarray\\tYes\\t\\tNodes\\n\\tnode_key\\tstring\\tNo\\t\\tNode key\\n\\ttags\\tarray\\tNo\\t\\tOriginal node tag list\\n\\tnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\noperations\\tarray\\tYes\\t\\t\\n\\ttags\\tarray\\tNo\\t\\tOperation tag list\\n\\tmode\\tstring\\tNo\\t\\tOperation type ADD / DELETE\\n\\nReturn data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tinteger\\tYes\\t\\tStatus code\\nmsg\\tstring\\tYes\\t\\tPrompt message\\ndata\\tobject\\tYes\\t\\tReturned data\\nlist\\tarray\\tNo\\t\\tNode list true / false\\nnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\nnode_key\\tstring\\tNo\\t\\tNode key\\n```\\n\\nUnit test:\\n```python\\n@pytest.mark.parametrize(\\n\\\"project_key, nodes, operations, expected_msg\\\",\\n[\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"success\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_002\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"tag1\\\"], \\\"mode\\\": \\\"DELETE\\\"}], \\\"success\\\"),\\n(\\\"\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Missing the required parameter project_key\\\"),\\n(123, [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Incorrect parameter type\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"a\\\"*201, \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Request parameter exceeds field boundary\\\")\\n]\\n)\\ndef test_node_tags(project_key, nodes, operations, expected_msg):\\n pass\\n\\n# The above is an interface definition and a unit test example.\\n# Next, please play the role of an expert test manager with 20 years of experience at Google. When I give the interface definition, \\n# reply to me with a unit test. There are several requirements:\\n# 1. Only output one `@pytest.mark.parametrize` and the corresponding test_ function (inside pass, do not implement).\\n# -- The function parameter contains expected_msg for result verification.\\n# 2. The generated test cases use shorter text or numbers and are as compact as possible.\\n# 3. If comments are needed, use Chinese.\\n\\n# If you understand, please wait for me to give the interface definition and just answer \\\"Understood\\\" to save tokens.\\n\"}, {\"role\": \"user\", \"content\": \"Refer to the test types: such as SQL injection, cross-site scripting (XSS), unauthorized access and privilege escalation, \\nauthentication and authorization, parameter verification, exception handling, file upload and download.\\nPlease output 10 test cases within one `@pytest.mark.parametrize` scope.\\n```text\\nAPI Name: 获取 model 详情(job专用-后续开放给sdk)\\nAPI Path: /v1/projects/{project_key}/jobs/{job_id}/models/{model_key}\\nMethod: GET\\n\\nRequest Parameters:\\nPath Parameters:\\nproject_key \\njob_id \\nmodel_key \\n\\nBody Parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nproject_key\\tstring\\tYes\\t\\t\\njob_id\\tstring\\tYes\\t\\t\\nmodel_key\\tstring\\tYes\\t\\t\\n\\nResponse Data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tnumber\\tYes\\t\\t0成功,非0失败\\nmsg\\tstring\\tYes\\t\\t如果失败,这里有错误信息\\ndata\\tobject\\tYes\\t\\tdata信息\\n\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\tname\\tstring\\tNo\\t\\t用户可修改的name\\n\\tmodel\\tobject\\tNo\\t\\tmodel信息\\n\\t\\ttype\\tstring\\tNo\\t\\tdataset type\\n\\t\\tmanaged\\tboolean\\tNo\\t\\t为false时是第一类dataset,数据不可删除\\n\\t\\tname\\tstring\\tNo\\t\\t用户可修改的name\\n\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\tformat_type\\tstring\\tNo\\t\\t文件类型的dataset才有这项。“csv”\\n\\t\\tflow_options\\tobject\\tNo\\t\\t创建dataset时的高级设置\\n\\t\\t\\tvirtualizable\\tboolean\\tNo\\t\\t高级设置里的参数。缺省false\\n\\t\\t\\trebuild_behavior\\tstring\\tNo\\t\\t高级设置里的参数。缺省NORMAL\\n\\t\\t\\tcross_project_build_behavior\\tstring\\tNo\\t\\t高级设置里的参数。缺省DEFAULT\\n\\t\\tformat_params\\tobject\\tNo\\t\\t文件类型的dataset才有\\n\\t\\t\\tstyle\\tstring\\tNo\\t\\t\\n\\t\\t\\tcharset\\tstring\\tNo\\t\\t\\n\\t\\t\\tseparator\\tstring\\tNo\\t\\t\\n\\t\\t\\tquote_char\\tstring\\tNo\\t\\t\\n\\t\\t\\tescape_char\\tstring\\tNo\\t\\t\\n\\t\\t\\tdate_serialization_format\\tstring\\tNo\\t\\t\\n\\t\\t\\tarray_map_format\\tstring\\tNo\\t\\t\\n\\t\\t\\thive_separators\\tarray\\tNo\\t\\t\\n\\t\\t\\tskip_rows_before_header\\tnumber\\tNo\\t\\t\\n\\t\\t\\tparse_header_row\\tboolean\\tNo\\t\\t\\n\\t\\t\\tskip_rows_after_header\\tnumber\\tNo\\t\\t\\n\\t\\t\\tprobable_number_of_records\\tnumber\\tNo\\t\\t\\n\\t\\t\\tnormalize_booleans\\tboolean\\tNo\\t\\t\\n\\t\\t\\tnormalize_doubles\\tboolean\\tNo\\t\\t\\n\\t\\ttags\\tarray\\tNo\\t\\t标签tags\\n\\t\\tparams\\tobject\\tNo\\t\\t必有这项,但不同类型的dataset里面的key有差别\\n\\t\\t\\tconnection\\tstring\\tNo\\t\\tconnection id,到db查其他参数\\n\\t\\t\\tpath\\tstring\\tNo\\t\\t文件类connection才有这项\\n\\t\\t\\ttable\\tstring\\tNo\\t\\tdb表名,DB类connection才有这项\\n\\t\\t\\tmode\\tstring\\tNo\\t\\t存储类型,比如“table\\\",DB类connection才有这项\\n\\t\\t\\tbucket\\tstring\\tNo\\t\\tS3类型的connection才有这项\\n\\t\\t\\tkey_name\\tstring\\tNo\\t\\tredis才有,key name\\n\\t\\t\\tkey_type\\tstring\\tNo\\t\\tredis才有,key type\\n\\t\\t\\tcollection\\tstring\\tNo\\t\\t非关系型数据库才有,collection name\\n\\t\\t\\tindex\\tstring\\tNo\\t\\t索引类型的才有这项\\n\\t\\t\\tnot_ready_if_empty\\tboolean\\tNo\\t\\t数据非空才认为是data ready\\n\\t\\t\\tfiles_selection_rules\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tmode\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\texclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tinclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\texplicit_files\\tarray\\tNo\\t\\t\\n\\t\\tschema\\tobject\\tNo\\t\\tcolumns信息在这里\\n\\t\\t\\tcolumns\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tname\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\ttype\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\torigin_type\\tstring\\tNo\\t\\t\\n\\t\\t\\tuser_modified\\tboolean\\tNo\\t\\t\\n\\t\\tcustom_fields\\tobject\\tNo\\t\\t自定义fields\\n\\t\\tlast_build\\tobject\\tNo\\t\\t最后一次构建的信息\\n\\t\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\t\\tid\\tstring\\tNo\\t\\tactivity id\\n\\t\\t\\tjob_id\\tstring\\tNo\\t\\tjob id\\n\\t\\t\\tjob_project_key\\tstring\\tNo\\t\\t\\n\\t\\t\\tbuild_start_time\\tnumber\\tNo\\t\\t构建开始时间\\n\\t\\t\\tbuild_end_time\\tnumber\\tNo\\t\\t构建结束时间\\n\\t\\t\\tbuild_success\\tstring\\tNo\\t\\tsuccess或failed\\n\\t\\tobject_key\\tstring\\tNo\\t\\tdataset_key,后台用的id,用户不可见不可改\\n\\t\\tcache\\tobject\\tNo\\t\\t下载缓存数据链接\\n\\t\\t\\ts3_path\\tstring\\tNo\\t\\t\\n\\tstatus\\tobject\\tNo\\t\\t数据状态\\n\\t\\tsize\\tobject\\tNo\\t\\t数据大小信息\\n\\t\\t\\ttotal_value\\tnumber\\tNo\\t\\t占多少字节磁盘\\n\\t\\t\\tlast_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\tfirst_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\thas_data\\tboolean\\tNo\\t\\t是否有数据,这个影响前端的图标显示\\n\\t\\t\\tincomplete\\tboolean\\tNo\\t\\t\\n\\t\\trecords\\tobject\\tNo\\t\\t\\n\\t\\t\\ttotal_value\\tnumber\\tNo\\t\\t\\n\\t\\t\\tlast_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\tfirst_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\thas_data\\tboolean\\tNo\\t\\t是否有数据,这个影响前端的图标显示\\n\\t\\t\\tincomplete\\tboolean\\tNo\\t\\t\\n\\t\\tpartitions_last_compute\\tnumber\\tNo\\t\\t\\n\\t\\tpartitions\\tnumber\\tNo\\t\\t\\n\\tbuildable\\tboolean\\tNo\\t\\t有recipe时为true\\n\\theaders\\tarray\\tNo\\t\\t\\n\\t\\tdataset_schema\\tobject\\tNo\\t\\t\\n\\t\\t\\tname\\tstring\\tNo\\t字段名称\\t\\n\\t\\t\\ttype\\tstring\\tNo\\t字段类型\\t\\n\\t\\tnormal_rate\\tobject\\tNo\\t缺失值统计信息\\t\\n\\n```\"}]": { + "code": "import string\nimport random\n\ndef random_string(length=10):\n return ''.join(random.choice(string.ascii_lowercase) for i in range(length))" + }, + "[{\"role\": \"user\", \"content\": \"Interface definition:\\n```text\\nInterface Name: Element Tagging\\nInterface Path: /projects/{project_key}/node-tags\\nMethod: POST\\n\\nRequest parameters:\\nPath parameters:\\nproject_key\\n\\nBody parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nnodes\\tarray\\tYes\\t\\tNodes\\n\\tnode_key\\tstring\\tNo\\t\\tNode key\\n\\ttags\\tarray\\tNo\\t\\tOriginal node tag list\\n\\tnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\noperations\\tarray\\tYes\\t\\t\\n\\ttags\\tarray\\tNo\\t\\tOperation tag list\\n\\tmode\\tstring\\tNo\\t\\tOperation type ADD / DELETE\\n\\nReturn data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tinteger\\tYes\\t\\tStatus code\\nmsg\\tstring\\tYes\\t\\tPrompt message\\ndata\\tobject\\tYes\\t\\tReturned data\\nlist\\tarray\\tNo\\t\\tNode list true / false\\nnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\nnode_key\\tstring\\tNo\\t\\tNode key\\n```\\n\\nUnit test:\\n```python\\n@pytest.mark.parametrize(\\n\\\"project_key, nodes, operations, expected_msg\\\",\\n[\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"success\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_002\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"tag1\\\"], \\\"mode\\\": \\\"DELETE\\\"}], \\\"success\\\"),\\n(\\\"\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Missing the required parameter project_key\\\"),\\n(123, [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Incorrect parameter type\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"a\\\"*201, \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Request parameter exceeds field boundary\\\")\\n]\\n)\\ndef test_node_tags(project_key, nodes, operations, expected_msg):\\n pass\\n\\n# The above is an interface definition and a unit test example.\\n# Next, please play the role of an expert test manager with 20 years of experience at Google. When I give the interface definition, \\n# reply to me with a unit test. There are several requirements:\\n# 1. Only output one `@pytest.mark.parametrize` and the corresponding test_ function (inside pass, do not implement).\\n# -- The function parameter contains expected_msg for result verification.\\n# 2. The generated test cases use shorter text or numbers and are as compact as possible.\\n# 3. If comments are needed, use Chinese.\\n\\n# If you understand, please wait for me to give the interface definition and just answer \\\"Understood\\\" to save tokens.\\n\"}, {\"role\": \"user\", \"content\": \"Refer to the test types: such as SQL injection, cross-site scripting (XSS), unauthorized access and privilege escalation, \\nauthentication and authorization, parameter verification, exception handling, file upload and download.\\nPlease output 10 test cases within one `@pytest.mark.parametrize` scope.\\n```text\\nAPI Name: 获取managed folder详情(job专用)\\nAPI Path: /v1/projects/{project_key}/jobs/{job_id}/folders/{folder_key}\\nMethod: GET\\n\\nRequest Parameters:\\nPath Parameters:\\nproject_key \\njob_id \\nfolder_key \\n\\nBody Parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nproject_key\\tstring\\tYes\\t\\t\\njob_id\\tstring\\tYes\\t\\t\\nfolder_key\\tstring\\tYes\\t\\t\\n\\nResponse Data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tnumber\\tYes\\t\\t0成功,非0失败\\nmsg\\tstring\\tYes\\t\\t失败时这里有错误信息\\ndata\\tobject\\tYes\\t\\t\\n\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\tfolder\\tobject\\tNo\\t\\tfolder配置在这里\\n\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\tobject_key\\tstring\\tNo\\t\\tobject key\\n\\t\\tname\\tstring\\tNo\\t\\t用户可编辑的那个name\\n\\t\\ttype\\tstring\\tNo\\t\\tfolder类型,与connection有关\\n\\t\\tparams\\tobject\\tNo\\t\\t数据读写相关配置在这里\\n\\t\\t\\tconnection\\tstring\\tNo\\t\\tconnection id\\n\\t\\t\\tpath\\tstring\\tNo\\t\\t文件夹内容存放的相对路径\\n\\t\\t\\tnot_ready_if_empty\\tboolean\\tNo\\t\\treserved\\n\\t\\t\\tfiles_selection_rules\\tobject\\tNo\\t\\t文件过滤规则\\n\\t\\t\\t\\tmode\\tstring\\tNo\\t\\tALL\\n\\t\\t\\t\\texclude_rules\\tarray\\tNo\\t\\t排除规则\\n\\t\\t\\t\\tinclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\texplicit_files\\tarray\\tNo\\t\\t\\n\\t\\tflow_options\\tobject\\tNo\\t\\tflow参数\\n\\t\\t\\tvirtualizable\\tboolean\\tNo\\t\\t\\n\\t\\t\\trebuild_behavior\\tstring\\tNo\\t\\t构建方式\\n\\t\\t\\tcross_project_build_behavior\\tstring\\tNo\\t\\t\\n\\t\\tmetrics\\tobject\\tNo\\t\\t\\n\\t\\t\\tprobes\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\ttype\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\tcompute_on_build_mode\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tmeta\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tname\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\tlevel\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\tconfiguration\\tobject\\tNo\\t\\t\\n\\t\\t\\tengine_config\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpad_runs_with_metrics\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\thive\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\textra_conf\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tbasic\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tdss\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\tselection\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tuse_mem_table\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tfilter\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tdistinct\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tpartition_selection_method\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tlatest_partitions_n\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tordering\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\trules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tsampling_method\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tmax_records\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\ttarget_ratio\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\twithin_first_n\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tmax_read_uncompressed_bytes\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\tsql\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\timpala\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\tspark\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\textra_conf\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tpython\\tobject\\tNo\\t\\t\\n\\t\\t\\tdisplayed_state\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpartition\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tcolumns\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tmetrics\\tarray\\tNo\\t\\t\\n\\t\\tchecks\\tobject\\tNo\\t\\t\\n\\t\\t\\trun_on_build\\tboolean\\tNo\\t\\t\\n\\t\\t\\tchecks\\tarray\\tNo\\t\\t\\n\\t\\t\\tdisplayed_state\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpartition\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tchecks\\tarray\\tNo\\t\\t\\n\\t\\tversion_tag\\tobject\\tNo\\t\\t配置版本信息\\n\\t\\t\\tversion_number\\tnumber\\tNo\\t\\t\\n\\t\\t\\tlast_modified_by\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tlogin\\tstring\\tNo\\t\\t\\n\\t\\t\\tlast_modified_on\\tnumber\\tNo\\t\\t修改时间unix time ms\\n\\t\\tcreation_tag\\tobject\\tNo\\t\\t配置创建时间\\n\\t\\t\\tversion_number\\tnumber\\tNo\\t\\t1\\n\\t\\t\\tlast_modified_by\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tlogin\\tstring\\tNo\\t\\t\\n\\t\\t\\tlast_modified_on\\tnumber\\tNo\\t\\t创建时间unix time ms\\n\\t\\ttags\\tarray\\tNo\\t\\t文件夹标签\\n\\t\\tcustom_fields\\tobject\\tNo\\t\\t\\n\\t\\tchecklists\\tobject\\tNo\\t\\t\\n\\t\\t\\tchecklists\\tarray\\tNo\\t\\t\\n\\n```\"}]": { + "code": "import string\nimport random\n\ndef random_string(length=10):\n return ''.join(random.choice(string.ascii_lowercase) for i in range(length))" } } \ No newline at end of file From 24d2c5c8e62226bc8fde45ae96362ee985ec6e28 Mon Sep 17 00:00:00 2001 From: yzlin Date: Sun, 4 Feb 2024 10:45:02 +0800 Subject: [PATCH 595/637] isolate codes to be restructured in the future --- metagpt/actions/__init__.py | 6 +++--- metagpt/actions/{ => ci}/ask_review.py | 0 metagpt/actions/{ => ci}/debug_code.py | 2 +- metagpt/actions/{ => ci}/execute_nb_code.py | 0 metagpt/actions/{ => ci}/ml_action.py | 6 +++--- metagpt/actions/{ => ci}/write_analysis_code.py | 2 +- metagpt/actions/{ => ci}/write_plan.py | 2 +- metagpt/plan/planner.py | 4 ++-- metagpt/prompts/{ => ci}/ml_action.py | 0 metagpt/prompts/{ => ci}/write_analysis_code.py | 0 metagpt/roles/code_interpreter.py | 9 ++++++--- metagpt/roles/ml_engineer.py | 6 +++--- tests/metagpt/actions/{ => ci}/test_ask_review.py | 2 +- tests/metagpt/actions/{ => ci}/test_debug_code.py | 2 +- tests/metagpt/actions/{ => ci}/test_execute_nb_code.py | 2 +- tests/metagpt/actions/{ => ci}/test_ml_action.py | 2 +- .../metagpt/actions/{ => ci}/test_write_analysis_code.py | 7 +++++-- tests/metagpt/actions/{ => ci}/test_write_plan.py | 2 +- tests/metagpt/roles/run_code_interpreter.py | 2 +- tests/metagpt/roles/test_code_interpreter.py | 2 +- tests/metagpt/roles/test_ml_engineer.py | 4 ++-- tests/metagpt/utils/test_save_code.py | 2 +- 22 files changed, 35 insertions(+), 29 deletions(-) rename metagpt/actions/{ => ci}/ask_review.py (100%) rename metagpt/actions/{ => ci}/debug_code.py (97%) rename metagpt/actions/{ => ci}/execute_nb_code.py (100%) rename metagpt/actions/{ => ci}/ml_action.py (93%) rename metagpt/actions/{ => ci}/write_analysis_code.py (99%) rename metagpt/actions/{ => ci}/write_plan.py (98%) rename metagpt/prompts/{ => ci}/ml_action.py (100%) rename metagpt/prompts/{ => ci}/write_analysis_code.py (100%) rename tests/metagpt/actions/{ => ci}/test_ask_review.py (84%) rename tests/metagpt/actions/{ => ci}/test_debug_code.py (96%) rename tests/metagpt/actions/{ => ci}/test_execute_nb_code.py (97%) rename tests/metagpt/actions/{ => ci}/test_ml_action.py (95%) rename tests/metagpt/actions/{ => ci}/test_write_analysis_code.py (98%) rename tests/metagpt/actions/{ => ci}/test_write_plan.py (95%) diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index 3f88fbcf3..6c0a2addc 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -22,9 +22,9 @@ from metagpt.actions.write_code_review import WriteCodeReview from metagpt.actions.write_prd import WritePRD from metagpt.actions.write_prd_review import WritePRDReview from metagpt.actions.write_test import WriteTest -from metagpt.actions.execute_nb_code import ExecuteNbCode -from metagpt.actions.write_analysis_code import WriteCodeByGenerate -from metagpt.actions.write_plan import WritePlan +from metagpt.actions.ci.execute_nb_code import ExecuteNbCode +from metagpt.actions.ci.write_analysis_code import WriteCodeByGenerate +from metagpt.actions.ci.write_plan import WritePlan class ActionType(Enum): diff --git a/metagpt/actions/ask_review.py b/metagpt/actions/ci/ask_review.py similarity index 100% rename from metagpt/actions/ask_review.py rename to metagpt/actions/ci/ask_review.py diff --git a/metagpt/actions/debug_code.py b/metagpt/actions/ci/debug_code.py similarity index 97% rename from metagpt/actions/debug_code.py rename to metagpt/actions/ci/debug_code.py index 34dac0147..f6b86b8bf 100644 --- a/metagpt/actions/debug_code.py +++ b/metagpt/actions/ci/debug_code.py @@ -1,6 +1,6 @@ from typing import List -from metagpt.actions.write_analysis_code import BaseWriteAnalysisCode +from metagpt.actions.ci.write_analysis_code import BaseWriteAnalysisCode from metagpt.logs import logger from metagpt.schema import Message from metagpt.utils.common import create_func_call_config diff --git a/metagpt/actions/execute_nb_code.py b/metagpt/actions/ci/execute_nb_code.py similarity index 100% rename from metagpt/actions/execute_nb_code.py rename to metagpt/actions/ci/execute_nb_code.py diff --git a/metagpt/actions/ml_action.py b/metagpt/actions/ci/ml_action.py similarity index 93% rename from metagpt/actions/ml_action.py rename to metagpt/actions/ci/ml_action.py index 88476707c..6fecae898 100644 --- a/metagpt/actions/ml_action.py +++ b/metagpt/actions/ci/ml_action.py @@ -1,14 +1,14 @@ from typing import List, Tuple from metagpt.actions import Action -from metagpt.actions.write_analysis_code import WriteCodeWithTools -from metagpt.prompts.ml_action import ( +from metagpt.actions.ci.write_analysis_code import WriteCodeWithTools +from metagpt.prompts.ci.ml_action import ( GENERATE_CODE_PROMPT, ML_TOOL_USAGE_PROMPT, PRINT_DATA_COLUMNS, UPDATE_DATA_COLUMNS, ) -from metagpt.prompts.write_analysis_code import CODE_GENERATOR_WITH_TOOLS +from metagpt.prompts.ci.write_analysis_code import CODE_GENERATOR_WITH_TOOLS from metagpt.schema import Message, Plan from metagpt.utils.common import create_func_call_config, remove_comments diff --git a/metagpt/actions/write_analysis_code.py b/metagpt/actions/ci/write_analysis_code.py similarity index 99% rename from metagpt/actions/write_analysis_code.py rename to metagpt/actions/ci/write_analysis_code.py index c4ac44f20..4e4ea7953 100644 --- a/metagpt/actions/write_analysis_code.py +++ b/metagpt/actions/ci/write_analysis_code.py @@ -8,7 +8,7 @@ from typing import Tuple from metagpt.actions import Action from metagpt.logs import logger -from metagpt.prompts.write_analysis_code import ( +from metagpt.prompts.ci.write_analysis_code import ( CODE_GENERATOR_WITH_TOOLS, SELECT_FUNCTION_TOOLS, TOOL_RECOMMENDATION_PROMPT, diff --git a/metagpt/actions/write_plan.py b/metagpt/actions/ci/write_plan.py similarity index 98% rename from metagpt/actions/write_plan.py rename to metagpt/actions/ci/write_plan.py index 77b52b78e..885611c68 100644 --- a/metagpt/actions/write_plan.py +++ b/metagpt/actions/ci/write_plan.py @@ -10,7 +10,7 @@ from typing import Dict, List, Tuple from metagpt.actions import Action from metagpt.logs import logger -from metagpt.prompts.write_analysis_code import ( +from metagpt.prompts.ci.write_analysis_code import ( ASSIGN_TASK_TYPE_CONFIG, ASSIGN_TASK_TYPE_PROMPT, ) diff --git a/metagpt/plan/planner.py b/metagpt/plan/planner.py index 0b3a05199..1b3971b7d 100644 --- a/metagpt/plan/planner.py +++ b/metagpt/plan/planner.py @@ -2,8 +2,8 @@ import json from pydantic import BaseModel, Field -from metagpt.actions.ask_review import AskReview, ReviewConst -from metagpt.actions.write_plan import ( +from metagpt.actions.ci.ask_review import AskReview, ReviewConst +from metagpt.actions.ci.write_plan import ( WritePlan, precheck_update_plan_from_rsp, update_plan_from_rsp, diff --git a/metagpt/prompts/ml_action.py b/metagpt/prompts/ci/ml_action.py similarity index 100% rename from metagpt/prompts/ml_action.py rename to metagpt/prompts/ci/ml_action.py diff --git a/metagpt/prompts/write_analysis_code.py b/metagpt/prompts/ci/write_analysis_code.py similarity index 100% rename from metagpt/prompts/write_analysis_code.py rename to metagpt/prompts/ci/write_analysis_code.py diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/code_interpreter.py index 1cae17ca0..f8d00bb91 100644 --- a/metagpt/roles/code_interpreter.py +++ b/metagpt/roles/code_interpreter.py @@ -1,8 +1,11 @@ from pydantic import Field -from metagpt.actions.ask_review import ReviewConst -from metagpt.actions.execute_nb_code import ExecuteNbCode -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.ci.ask_review import ReviewConst +from metagpt.actions.ci.execute_nb_code import ExecuteNbCode +from metagpt.actions.ci.write_analysis_code import ( + WriteCodeByGenerate, + WriteCodeWithTools, +) from metagpt.logs import logger from metagpt.roles import Role from metagpt.schema import Message, Task, TaskResult diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ml_engineer.py index 9d222b0bf..c7702771d 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ml_engineer.py @@ -1,6 +1,6 @@ -from metagpt.actions.debug_code import DebugCode -from metagpt.actions.execute_nb_code import ExecuteNbCode -from metagpt.actions.ml_action import UpdateDataColumns, WriteCodeWithToolsML +from metagpt.actions.ci.debug_code import DebugCode +from metagpt.actions.ci.execute_nb_code import ExecuteNbCode +from metagpt.actions.ci.ml_action import UpdateDataColumns, WriteCodeWithToolsML from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter from metagpt.tools.tool_types import ToolTypes diff --git a/tests/metagpt/actions/test_ask_review.py b/tests/metagpt/actions/ci/test_ask_review.py similarity index 84% rename from tests/metagpt/actions/test_ask_review.py rename to tests/metagpt/actions/ci/test_ask_review.py index 00001fad6..4f02fe10b 100644 --- a/tests/metagpt/actions/test_ask_review.py +++ b/tests/metagpt/actions/ci/test_ask_review.py @@ -1,6 +1,6 @@ import pytest -from metagpt.actions.ask_review import AskReview +from metagpt.actions.ci.ask_review import AskReview @pytest.mark.asyncio diff --git a/tests/metagpt/actions/test_debug_code.py b/tests/metagpt/actions/ci/test_debug_code.py similarity index 96% rename from tests/metagpt/actions/test_debug_code.py rename to tests/metagpt/actions/ci/test_debug_code.py index 32a4914f4..0307ac17e 100644 --- a/tests/metagpt/actions/test_debug_code.py +++ b/tests/metagpt/actions/ci/test_debug_code.py @@ -5,7 +5,7 @@ import pytest -from metagpt.actions.debug_code import DebugCode +from metagpt.actions.ci.debug_code import DebugCode from metagpt.schema import Message ErrorStr = """Tested passed: diff --git a/tests/metagpt/actions/test_execute_nb_code.py b/tests/metagpt/actions/ci/test_execute_nb_code.py similarity index 97% rename from tests/metagpt/actions/test_execute_nb_code.py rename to tests/metagpt/actions/ci/test_execute_nb_code.py index d1b40c350..6402cb883 100644 --- a/tests/metagpt/actions/test_execute_nb_code.py +++ b/tests/metagpt/actions/ci/test_execute_nb_code.py @@ -1,6 +1,6 @@ import pytest -from metagpt.actions.execute_nb_code import ExecuteNbCode, truncate +from metagpt.actions.ci.execute_nb_code import ExecuteNbCode, truncate @pytest.mark.asyncio diff --git a/tests/metagpt/actions/test_ml_action.py b/tests/metagpt/actions/ci/test_ml_action.py similarity index 95% rename from tests/metagpt/actions/test_ml_action.py rename to tests/metagpt/actions/ci/test_ml_action.py index 2c8d34da8..5d9507094 100644 --- a/tests/metagpt/actions/test_ml_action.py +++ b/tests/metagpt/actions/ci/test_ml_action.py @@ -1,6 +1,6 @@ import pytest -from metagpt.actions.ml_action import WriteCodeWithToolsML +from metagpt.actions.ci.ml_action import WriteCodeWithToolsML from metagpt.schema import Plan, Task diff --git a/tests/metagpt/actions/test_write_analysis_code.py b/tests/metagpt/actions/ci/test_write_analysis_code.py similarity index 98% rename from tests/metagpt/actions/test_write_analysis_code.py rename to tests/metagpt/actions/ci/test_write_analysis_code.py index eec3d3e38..72071fa35 100644 --- a/tests/metagpt/actions/test_write_analysis_code.py +++ b/tests/metagpt/actions/ci/test_write_analysis_code.py @@ -2,8 +2,11 @@ import asyncio import pytest -from metagpt.actions.execute_nb_code import ExecuteNbCode -from metagpt.actions.write_analysis_code import WriteCodeByGenerate, WriteCodeWithTools +from metagpt.actions.ci.execute_nb_code import ExecuteNbCode +from metagpt.actions.ci.write_analysis_code import ( + WriteCodeByGenerate, + WriteCodeWithTools, +) from metagpt.logs import logger from metagpt.plan.planner import STRUCTURAL_CONTEXT from metagpt.schema import Message, Plan, Task diff --git a/tests/metagpt/actions/test_write_plan.py b/tests/metagpt/actions/ci/test_write_plan.py similarity index 95% rename from tests/metagpt/actions/test_write_plan.py rename to tests/metagpt/actions/ci/test_write_plan.py index f36527711..3eb80ca3e 100644 --- a/tests/metagpt/actions/test_write_plan.py +++ b/tests/metagpt/actions/ci/test_write_plan.py @@ -1,6 +1,6 @@ import pytest -from metagpt.actions.write_plan import ( +from metagpt.actions.ci.write_plan import ( Plan, Task, WritePlan, diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py index e5e2b8df5..f0fcdb200 100644 --- a/tests/metagpt/roles/run_code_interpreter.py +++ b/tests/metagpt/roles/run_code_interpreter.py @@ -1,6 +1,6 @@ import fire -from metagpt.actions.execute_nb_code import ExecuteNbCode +from metagpt.actions.ci.execute_nb_code import ExecuteNbCode from metagpt.const import DATA_PATH from metagpt.logs import logger from metagpt.roles.code_interpreter import CodeInterpreter diff --git a/tests/metagpt/roles/test_code_interpreter.py b/tests/metagpt/roles/test_code_interpreter.py index 2263b2a4a..2d71fcbb0 100644 --- a/tests/metagpt/roles/test_code_interpreter.py +++ b/tests/metagpt/roles/test_code_interpreter.py @@ -7,7 +7,7 @@ from metagpt.roles.code_interpreter import CodeInterpreter @pytest.mark.asyncio @pytest.mark.parametrize("auto_run", [(True), (False)]) async def test_code_interpreter(mocker, auto_run): - mocker.patch("metagpt.actions.execute_nb_code.ExecuteNbCode.run", return_value=("a successful run", True)) + mocker.patch("metagpt.actions.ci.execute_nb_code.ExecuteNbCode.run", return_value=("a successful run", True)) mocker.patch("builtins.input", return_value="confirm") requirement = "Run data analysis on sklearn Iris dataset, include a plot" diff --git a/tests/metagpt/roles/test_ml_engineer.py b/tests/metagpt/roles/test_ml_engineer.py index c00481019..2728c6411 100644 --- a/tests/metagpt/roles/test_ml_engineer.py +++ b/tests/metagpt/roles/test_ml_engineer.py @@ -1,11 +1,11 @@ import pytest -from metagpt.actions.execute_nb_code import ExecuteNbCode +from metagpt.actions.ci.execute_nb_code import ExecuteNbCode from metagpt.logs import logger from metagpt.roles.ml_engineer import MLEngineer from metagpt.schema import Message, Plan, Task from metagpt.tools.tool_types import ToolTypes -from tests.metagpt.actions.test_debug_code import CODE, DebugContext, ErrorStr +from tests.metagpt.actions.ci.test_debug_code import CODE, DebugContext, ErrorStr def test_mle_init(): diff --git a/tests/metagpt/utils/test_save_code.py b/tests/metagpt/utils/test_save_code.py index 62724dde5..5ab08c454 100644 --- a/tests/metagpt/utils/test_save_code.py +++ b/tests/metagpt/utils/test_save_code.py @@ -6,7 +6,7 @@ import nbformat import pytest -from metagpt.actions.execute_nb_code import ExecuteNbCode +from metagpt.actions.ci.execute_nb_code import ExecuteNbCode from metagpt.utils.common import read_json_file from metagpt.utils.save_code import DATA_PATH, save_code_file From b7d0379faecc8645734175b695592e06e7d3c96c Mon Sep 17 00:00:00 2001 From: yzlin Date: Sun, 4 Feb 2024 13:10:43 +0800 Subject: [PATCH 596/637] rm experimental code --- tests/metagpt/roles/run_code_interpreter.py | 82 --------------------- 1 file changed, 82 deletions(-) delete mode 100644 tests/metagpt/roles/run_code_interpreter.py diff --git a/tests/metagpt/roles/run_code_interpreter.py b/tests/metagpt/roles/run_code_interpreter.py deleted file mode 100644 index f0fcdb200..000000000 --- a/tests/metagpt/roles/run_code_interpreter.py +++ /dev/null @@ -1,82 +0,0 @@ -import fire - -from metagpt.actions.ci.execute_nb_code import ExecuteNbCode -from metagpt.const import DATA_PATH -from metagpt.logs import logger -from metagpt.roles.code_interpreter import CodeInterpreter -from metagpt.roles.ml_engineer import MLEngineer -from metagpt.schema import Plan -from metagpt.utils.recovery_util import load_history, save_history - - -async def run_code_interpreter(role_class, requirement, auto_run, use_tools, save_dir, tools): - """ - The main function to run the MLEngineer with optional history loading. - - Args: - requirement (str): The requirement for the MLEngineer. - auto_run (bool): Whether to auto-run the MLEngineer. - save_dir (str): The directory from which to load the history or to save the new history. - - Raises: - Exception: If an error occurs during execution, log the error and save the history. - """ - - if role_class == "ci": - role = CodeInterpreter(auto_run=auto_run, use_tools=use_tools, tools=tools) - else: - role = MLEngineer( - auto_run=auto_run, - use_tools=use_tools, - tools=tools, - ) - - if save_dir: - logger.info("Resuming from history trajectory") - plan, nb = load_history(save_dir) - role.planner.plan = Plan(**plan) - role.execute_code = ExecuteNbCode(nb) - - else: - logger.info("Run from scratch") - - try: - await role.run(requirement) - except Exception as e: - logger.exception(f"An error occurred: {e}, save trajectory here: {save_path}") - - save_history(role, save_dir) - - -if __name__ == "__main__": - # requirement = "Run data analysis on sklearn Iris dataset, include a plot" - # requirement = "Run data analysis on sklearn Wine recognition dataset, include a plot, and train a model to predict wine class (20% as validation), and show validation accuracy" - data_path = f"{DATA_PATH}/titanic" - requirement = f"This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - # data_path = f"{DATA_PATH}/icr-identify-age-related-conditions" - # requirement = f"This is a medical dataset with over fifty anonymized health characteristics linked to three age-related conditions. Your goal is to predict whether a subject has or has not been diagnosed with one of these conditions.The target column is Class. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report f1 score on the eval data. Train data path: {data_path}/split_train.csv, eval data path: {data_path}/split_eval.csv." - # data_path = f"{DATA_PATH}/santander-customer-transaction-prediction" - # requirement = f"This is a customers financial dataset. Your goal is to predict which customers will make a specific transaction in the future. The target column is target. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report AUC Score on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv' ." - # data_path = f"{DATA_PATH}/house-prices-advanced-regression-techniques" - # requirement = f"This is a house price dataset, your goal is to predict the sale price of a property based on its features. The target column is SalePrice. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report RMSE between the logarithm of the predicted value and the logarithm of the observed sales price on the eval data. Train data path: '{data_path}/split_train.csv', eval data path: '{data_path}/split_eval.csv'." - - save_dir = "" - - # role_class = "ci" - role_class = "mle" - auto_run = True - use_tools = True - tools = [] - # tools = ["FillMissingValue", "CatCross", "non_existing_test"] - - async def main( - role_class: str = role_class, - requirement: str = requirement, - auto_run: bool = auto_run, - use_tools: bool = use_tools, - save_dir: str = save_dir, - tools=tools, - ): - await run_code_interpreter(role_class, requirement, auto_run, use_tools, save_dir, tools) - - fire.Fire(main) From d1deb0ff7ccaa9d4f74eeb632b579cd080944c67 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Sun, 4 Feb 2024 18:03:27 +0800 Subject: [PATCH 597/637] Remove _parse_arguments function and comment out handle_exception decorator on get_choice_function_arguments. --- metagpt/provider/openai_api.py | 41 +++++----------------------------- 1 file changed, 6 insertions(+), 35 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 206701efd..3ab25c276 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -8,7 +8,6 @@ """ import json -import re from typing import AsyncIterator, Optional, Union from openai import APIConnectionError, AsyncOpenAI, AsyncStream @@ -195,31 +194,7 @@ class OpenAILLM(BaseLLM): rsp = await self._achat_completion_function(messages, **kwargs) return self.get_choice_function_arguments(rsp) - def _parse_arguments(self, arguments: str) -> dict: - """parse arguments in openai function call""" - if "langugae" not in arguments and "code" not in arguments: - logger.warning(f"Not found `code`, `language`, We assume it is pure code:\n {arguments}\n. ") - return {"language": "python", "code": arguments} - - # 匹配language - language_pattern = re.compile(r'[\"\']?language[\"\']?\s*:\s*["\']([^"\']+?)["\']', re.DOTALL) - language_match = language_pattern.search(arguments) - language_value = language_match.group(1) if language_match else "python" - - # 匹配code - code_pattern = r'(["\'`]{3}|["\'`])([\s\S]*?)\1' - try: - code_value = re.findall(code_pattern, arguments)[-1][-1] - except Exception as e: - logger.error(f"{e}, when re.findall({code_pattern}, {arguments})") - code_value = None - - if code_value is None: - raise ValueError(f"Parse code error for {arguments}") - # arguments只有code的情况 - return {"language": language_value, "code": code_value} - - @handle_exception + # @handle_exception def get_choice_function_arguments(self, rsp: ChatCompletion) -> dict: """Required to provide the first function arguments of choice. @@ -237,19 +212,15 @@ class OpenAILLM(BaseLLM): try: return json.loads(message.tool_calls[0].function.arguments, strict=False) except json.decoder.JSONDecodeError as e: - logger.warning( - "\n".join( - [ - (f"Got JSONDecodeError for \n{'--'*40} \n{message.tool_calls[0].function.arguments}"), - (f"{'--'*40}\nwe will use RegExp to parse code. JSONDecodeError is: {e}"), - ] - ) + error_msg = ( + f"Got JSONDecodeError for \n{'--'*40} \n{message.tool_calls[0].function.arguments}, {str(e)}" ) - return self._parse_arguments(message.tool_calls[0].function.arguments) + logger.error(error_msg) + raise json.decoder.JSONDecodeError(error_msg, e.doc, e.pos) elif message.tool_calls is None and message.content is not None: # reponse is code, fix openai tools_call respond bug, # The response content is `code``, but it appears in the content instead of the arguments. - code_formats = ("```", '"""', "'''") + code_formats = "```" if message.content.startswith(code_formats) and message.content.endswith(code_formats): code = CodeParser.parse_code(None, message.content) return {"language": "python", "code": code} From 4caa1ece816737c696438de00c3b51578ce25a70 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Sun, 4 Feb 2024 18:07:16 +0800 Subject: [PATCH 598/637] Revert CodeParser.parse_code function to version 0.6.6. --- metagpt/utils/common.py | 24 ++++++++++-------------- 1 file changed, 10 insertions(+), 14 deletions(-) diff --git a/metagpt/utils/common.py b/metagpt/utils/common.py index 9d6a6bb24..d7eef5bd9 100644 --- a/metagpt/utils/common.py +++ b/metagpt/utils/common.py @@ -8,7 +8,6 @@ Add generic class-to-string and object-to-string conversion functionality. @Modified By: mashenquan, 2023/11/27. Bug fix: `parse_recipient` failed to parse the recipient in certain GPT-3.5 responses. -@Modified By: liubangbang, 2024/01/23. Update: support [```, ''', \"\"\" ] codes in CodeParser.parse_code. """ from __future__ import annotations @@ -268,19 +267,16 @@ class CodeParser: def parse_code(cls, block: str, text: str, lang: str = "") -> str: if block: text = cls.parse_block(block, text) - start_ends = ["```", "'''", '"""'] - patterns = [] - for start_end in start_ends: - pattern = rf"{start_end}{lang}.*?\s+(.*?){start_end}" - match = re.search(pattern, text, re.DOTALL) - if match: - code = match.group(1) - return code - patterns.append(pattern) - logger.error(f"{patterns} not match following text:") - logger.error(text) - # raise Exception - return text # just assume original text is code + pattern = rf"```{lang}.*?\s+(.*?)```" + match = re.search(pattern, text, re.DOTALL) + if match: + code = match.group(1) + else: + logger.error(f"{pattern} not match following text:") + logger.error(text) + # raise Exception + return text # just assume original text is code + return code @classmethod def parse_str(cls, block: str, text: str, lang: str = ""): From 4b912cc527ec6567eff896b8a2891f33b5fbcc98 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Sun, 4 Feb 2024 18:07:43 +0800 Subject: [PATCH 599/637] update test. --- tests/metagpt/provider/test_openai.py | 31 +++++++++++++++------------ 1 file changed, 17 insertions(+), 14 deletions(-) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index a49d7e85b..a48e27432 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -1,3 +1,5 @@ +import json + import pytest from openai.types.chat import ( ChatCompletion, @@ -40,16 +42,6 @@ async def test_speech_to_text(): def tool_calls_rsp(): function_rsps = [ Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"}', name="execute"), - Function(arguments='{\n"language": "python",\n"code": \'print("hello world")\'}', name="execute"), - Function(arguments='{\n"language": \'python\',\n"code": "print(\'hello world\')"}', name="execute"), - Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"}', name="execute"), - Function(arguments='{\n"language": "python",\n"code": ```print("hello world")```}', name="execute"), - Function(arguments='{\n"language": "python",\n"code": """print("hello world")"""}', name="execute"), - Function(arguments='\nprint("hello world")\\n', name="execute"), - # only `{` in arguments - Function(arguments='{\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), - # no `{`, `}` in arguments - Function(arguments='\n"language": "python",\n"code": "print(\'hello world\')"', name="execute"), ] tool_calls = [ ChatCompletionMessageToolCall(type="function", id=f"call_{i}", function=f) for i, f in enumerate(function_rsps) @@ -63,10 +55,6 @@ def tool_calls_rsp(): messages.extend( [ ChatCompletionMessage(content="```python\nprint('hello world')```", role="assistant", tool_calls=None), - ChatCompletionMessage(content="'''python\nprint('hello world')'''", role="assistant", tool_calls=None), - ChatCompletionMessage(content='"""python\nprint(\'hello world\')"""', role="assistant", tool_calls=None), - ChatCompletionMessage(content="'''python\nprint(\"hello world\")'''", role="assistant", tool_calls=None), - ChatCompletionMessage(content="```python\nprint('hello world')```", role="assistant", tool_calls=None), ] ) choices = [ @@ -78,6 +66,15 @@ def tool_calls_rsp(): ] +@pytest.fixture +def json_decode_error(): + function_rsp = Function(arguments='{\n"language": \'python\',\n"code": "print(\'hello world\')"}', name="execute") + tool_calls = [ChatCompletionMessageToolCall(type="function", id=f"call_{0}", function=function_rsp)] + message = ChatCompletionMessage(content=None, role="assistant", tool_calls=tool_calls) + choices = [Choice(finish_reason="tool_calls", logprobs=None, index=0, message=message)] + return ChatCompletion(id="0", choices=choices, created=0, model="gpt-4", object="chat.completion") + + class TestOpenAI: def test_make_client_kwargs_without_proxy(self): instance = OpenAILLM(mock_llm_config) @@ -105,3 +102,9 @@ class TestOpenAI: code["language"] == "markdown" else: code["language"] == "python" + + def test_aask_code_JSONDecodeError(self, json_decode_error): + instance = OpenAILLM(mock_llm_config) + with pytest.raises(json.decoder.JSONDecodeError) as e: + instance.get_choice_function_arguments(json_decode_error) + assert "JSONDecodeError" in str(e) From b4d032c8bffecf06fcf5f1869e620e62cbd5eb96 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=88=98=E6=A3=92=E6=A3=92?= Date: Sun, 4 Feb 2024 18:22:33 +0800 Subject: [PATCH 600/637] chore. --- tests/metagpt/provider/test_openai.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/metagpt/provider/test_openai.py b/tests/metagpt/provider/test_openai.py index a48e27432..3883aab2e 100644 --- a/tests/metagpt/provider/test_openai.py +++ b/tests/metagpt/provider/test_openai.py @@ -103,7 +103,7 @@ class TestOpenAI: else: code["language"] == "python" - def test_aask_code_JSONDecodeError(self, json_decode_error): + def test_aask_code_json_decode_error(self, json_decode_error): instance = OpenAILLM(mock_llm_config) with pytest.raises(json.decoder.JSONDecodeError) as e: instance.get_choice_function_arguments(json_decode_error) From 321a4c0d75c4d522edd37edcef3e26efe59007f9 Mon Sep 17 00:00:00 2001 From: yzlin Date: Sun, 4 Feb 2024 20:25:49 +0800 Subject: [PATCH 601/637] rm redundant function and docstring in libs --- metagpt/tools/__init__.py | 3 +- metagpt/tools/libs/data_preprocess.py | 318 ++++++---------------- metagpt/tools/libs/feature_engineering.py | 165 ----------- metagpt/tools/tool_convert.py | 15 +- metagpt/tools/tool_registry.py | 28 +- metagpt/utils/parse_docstring.py | 2 +- tests/metagpt/tools/test_tool_convert.py | 88 +++--- tests/metagpt/tools/test_tool_registry.py | 61 +++-- tests/metagpt/utils/test_save_code.py | 4 +- 9 files changed, 176 insertions(+), 508 deletions(-) diff --git a/metagpt/tools/__init__.py b/metagpt/tools/__init__.py index bb87f1b62..c1f604df9 100644 --- a/metagpt/tools/__init__.py +++ b/metagpt/tools/__init__.py @@ -7,11 +7,10 @@ """ from enum import Enum -from metagpt.tools import tool_types # this registers all tool types from metagpt.tools import libs # this registers all tools from metagpt.tools.tool_registry import TOOL_REGISTRY -_ = tool_types, libs, TOOL_REGISTRY # Avoid pre-commit error +_ = libs, TOOL_REGISTRY # Avoid pre-commit error class SearchEngineType(Enum): diff --git a/metagpt/tools/libs/data_preprocess.py b/metagpt/tools/libs/data_preprocess.py index 307a6bc5b..9c571ad6b 100644 --- a/metagpt/tools/libs/data_preprocess.py +++ b/metagpt/tools/libs/data_preprocess.py @@ -19,14 +19,29 @@ from metagpt.tools.tool_types import ToolTypes TOOL_TYPE = ToolTypes.DATA_PREPROCESS.type_name -class MLProcess(object): - def fit(self, df): +class MLProcess: + def fit(self, df: pd.DataFrame): + """ + Fit a model to be used in subsequent transform. + + Args: + df (pd.DataFrame): The input DataFrame. + """ raise NotImplementedError - def transform(self, df): + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ raise NotImplementedError - def fit_transform(self, df) -> pd.DataFrame: + def fit_transform(self, df: pd.DataFrame) -> pd.DataFrame: """ Fit and transform the input DataFrame. @@ -40,6 +55,49 @@ class MLProcess(object): return self.transform(df) +class DataPreprocessTool(MLProcess): + """ + Completing a data preprocessing operation. + """ + + def __init__(self, features: list): + """ + Initialize self. + + Args: + features (list): Columns to be processed. + """ + self.features = features + self.model = None # to be filled by specific subclass Tool + + def fit(self, df: pd.DataFrame): + """ + Fit a model to be used in subsequent transform. + + Args: + df (pd.DataFrame): The input DataFrame. + """ + if len(self.features) == 0: + return + self.model.fit(df[self.features]) + + def transform(self, df: pd.DataFrame) -> pd.DataFrame: + """ + Transform the input DataFrame with the fitted model. + + Args: + df (pd.DataFrame): The input DataFrame. + + Returns: + pd.DataFrame: The transformed DataFrame. + """ + if len(self.features) == 0: + return df + new_df = df.copy() + new_df[self.features] = self.model.transform(new_df[self.features]) + return new_df + + @register_tool(tool_type=TOOL_TYPE) class FillMissingValue(MLProcess): """ @@ -58,282 +116,77 @@ class FillMissingValue(MLProcess): Defaults to None. """ self.features = features - self.strategy = strategy - self.fill_value = fill_value - self.si = None - - def fit(self, df: pd.DataFrame): - """ - Fit the FillMissingValue model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ - if len(self.features) == 0: - return - self.si = SimpleImputer(strategy=self.strategy, fill_value=self.fill_value) - self.si.fit(df[self.features]) - - def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ - if len(self.features) == 0: - return df - new_df = df.copy() - new_df[self.features] = self.si.transform(new_df[self.features]) - return new_df + self.model = SimpleImputer(strategy=strategy, fill_value=fill_value) @register_tool(tool_type=TOOL_TYPE) -class MinMaxScale(MLProcess): +class MinMaxScale(DataPreprocessTool): """ Transform features by scaling each feature to a range, which is (0, 1). """ def __init__(self, features: list): - """ - Initialize self. - - Args: - features (list): Columns to be processed. - """ self.features = features - self.mms = None - - def fit(self, df: pd.DataFrame): - """ - Fit the MinMaxScale model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ - self.mms = MinMaxScaler() - self.mms.fit(df[self.features]) - - def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ - new_df = df.copy() - new_df[self.features] = self.mms.transform(new_df[self.features]) - return new_df + self.model = MinMaxScaler() @register_tool(tool_type=TOOL_TYPE) -class StandardScale(MLProcess): +class StandardScale(DataPreprocessTool): """ Standardize features by removing the mean and scaling to unit variance. """ def __init__(self, features: list): - """ - Initialize self. - - Args: - features (list): Columns to be processed. - """ self.features = features - self.ss = None - - def fit(self, df: pd.DataFrame): - """ - Fit the StandardScale model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ - self.ss = StandardScaler() - self.ss.fit(df[self.features]) - - def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ - new_df = df.copy() - new_df[self.features] = self.ss.transform(new_df[self.features]) - return new_df + self.model = StandardScaler() @register_tool(tool_type=TOOL_TYPE) -class MaxAbsScale(MLProcess): +class MaxAbsScale(DataPreprocessTool): """ Scale each feature by its maximum absolute value. """ def __init__(self, features: list): - """ - Initialize self. - - Args: - features (list): Columns to be processed. - """ self.features = features - self.mas = None - - def fit(self, df: pd.DataFrame): - """ - Fit the MaxAbsScale model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ - self.mas = MaxAbsScaler() - self.mas.fit(df[self.features]) - - def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ - new_df = df.copy() - new_df[self.features] = self.mas.transform(new_df[self.features]) - return new_df + self.model = MaxAbsScaler() @register_tool(tool_type=TOOL_TYPE) -class RobustScale(MLProcess): +class RobustScale(DataPreprocessTool): """ Apply the RobustScaler to scale features using statistics that are robust to outliers. """ def __init__(self, features: list): - """ - Initialize the RobustScale instance with feature names. - - Args: - features (list): List of feature names to be scaled. - """ self.features = features - self.rs = None - - def fit(self, df: pd.DataFrame): - """ - Compute the median and IQR for scaling. - - Args: - df (pd.DataFrame): Dataframe containing the features. - """ - self.rs = RobustScaler() - self.rs.fit(df[self.features]) - - def transform(self, df: pd.DataFrame): - """ - Scale features using the previously computed median and IQR. - - Args: - df (pd.DataFrame): Dataframe containing the features to be scaled. - - Returns: - pd.DataFrame: A new dataframe with scaled features. - """ - new_df = df.copy() - new_df[self.features] = self.rs.transform(new_df[self.features]) - return new_df + self.model = RobustScaler() @register_tool(tool_type=TOOL_TYPE) -class OrdinalEncode(MLProcess): +class OrdinalEncode(DataPreprocessTool): """ Encode categorical features as ordinal integers. """ def __init__(self, features: list): - """ - Initialize the OrdinalEncode instance with feature names. - - Args: - features (list): List of categorical feature names to be encoded. - """ self.features = features - self.oe = None - - def fit(self, df: pd.DataFrame): - """ - Learn the ordinal encodings for the features. - - Args: - df (pd.DataFrame): Dataframe containing the categorical features. - """ - self.oe = OrdinalEncoder() - self.oe.fit(df[self.features]) - - def transform(self, df: pd.DataFrame): - """ - Convert the categorical features to ordinal integers. - - Args: - df (pd.DataFrame): Dataframe containing the categorical features to be encoded. - - Returns: - pd.DataFrame: A new dataframe with the encoded features. - """ - new_df = df.copy() - new_df[self.features] = self.oe.transform(new_df[self.features]) - return new_df + self.model = OrdinalEncoder() @register_tool(tool_type=TOOL_TYPE) -class OneHotEncode(MLProcess): +class OneHotEncode(DataPreprocessTool): """ Apply one-hot encoding to specified categorical columns, the original columns will be dropped. """ def __init__(self, features: list): - """ - Initialize self. - - Args: - features (list): Categorical columns to be one-hot encoded and dropped. - """ self.features = features - self.ohe = None - - def fit(self, df: pd.DataFrame): - """ - Fit the OneHotEncoding model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ - self.ohe = OneHotEncoder(handle_unknown="ignore", sparse=False) - self.ohe.fit(df[self.features]) + self.model = OneHotEncoder(handle_unknown="ignore", sparse=False) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ - ts_data = self.ohe.transform(df[self.features]) - new_columns = self.ohe.get_feature_names_out(self.features) + ts_data = self.model.transform(df[self.features]) + new_columns = self.model.get_feature_names_out(self.features) ts_data = pd.DataFrame(ts_data, columns=new_columns, index=df.index) new_df = df.drop(self.features, axis=1) new_df = pd.concat([new_df, ts_data], axis=1) @@ -341,7 +194,7 @@ class OneHotEncode(MLProcess): @register_tool(tool_type=TOOL_TYPE) -class LabelEncode(MLProcess): +class LabelEncode(DataPreprocessTool): """ Apply label encoding to specified categorical columns in-place. """ @@ -357,12 +210,6 @@ class LabelEncode(MLProcess): self.le_encoders = [] def fit(self, df: pd.DataFrame): - """ - Fit the LabelEncode model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ if len(self.features) == 0: return for col in self.features: @@ -370,15 +217,6 @@ class LabelEncode(MLProcess): self.le_encoders.append(le) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ if len(self.features) == 0: return df new_df = df.copy() diff --git a/metagpt/tools/libs/feature_engineering.py b/metagpt/tools/libs/feature_engineering.py index 6de5696d4..bbd16b681 100644 --- a/metagpt/tools/libs/feature_engineering.py +++ b/metagpt/tools/libs/feature_engineering.py @@ -45,12 +45,6 @@ class PolynomialExpansion(MLProcess): self.poly = PolynomialFeatures(degree=degree, include_bias=False) def fit(self, df: pd.DataFrame): - """ - Fit the PolynomialExpansion model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ if len(self.cols) == 0: return if len(self.cols) > 10: @@ -61,15 +55,6 @@ class PolynomialExpansion(MLProcess): self.poly.fit(df[self.cols].fillna(0)) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame without duplicated columns. - """ if len(self.cols) == 0: return df ts_data = self.poly.transform(df[self.cols].fillna(0)) @@ -97,24 +82,9 @@ class CatCount(MLProcess): self.encoder_dict = None def fit(self, df: pd.DataFrame): - """ - Fit the CatCount model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ self.encoder_dict = df[self.col].value_counts().to_dict() def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ new_df = df.copy() new_df[f"{self.col}_cnt"] = new_df[self.col].map(self.encoder_dict) return new_df @@ -139,24 +109,9 @@ class TargetMeanEncoder(MLProcess): self.encoder_dict = None def fit(self, df: pd.DataFrame): - """ - Fit the TargetMeanEncoder model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ self.encoder_dict = df.groupby(self.col)[self.label].mean().to_dict() def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ new_df = df.copy() new_df[f"{self.col}_target_mean"] = new_df[self.col].map(self.encoder_dict) return new_df @@ -185,12 +140,6 @@ class KFoldTargetMeanEncoder(MLProcess): self.encoder_dict = None def fit(self, df: pd.DataFrame): - """ - Fit the KFoldTargetMeanEncoder model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ tmp = df.copy() kf = KFold(n_splits=self.n_splits, shuffle=True, random_state=self.random_state) @@ -203,15 +152,6 @@ class KFoldTargetMeanEncoder(MLProcess): self.encoder_dict = tmp.groupby(self.col)[col_name].mean().to_dict() def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ new_df = df.copy() new_df[f"{self.col}_kf_target_mean"] = new_df[self.col].map(self.encoder_dict) return new_df @@ -255,12 +195,6 @@ class CatCross(MLProcess): return new_col, comb_map def fit(self, df: pd.DataFrame): - """ - Fit the CatCross model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ for col in self.cols: if df[col].nunique() > self.max_cat_num: self.cols.remove(col) @@ -269,15 +203,6 @@ class CatCross(MLProcess): self.combs_map = dict(res) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ new_df = df.copy() for comb in self.combs: new_col = f"{comb[0]}_{comb[1]}" @@ -310,12 +235,6 @@ class GroupStat(MLProcess): self.group_df = None def fit(self, df: pd.DataFrame): - """ - Fit the GroupStat model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ group_df = df.groupby(self.group_col)[self.agg_col].agg(self.agg_funcs).reset_index() group_df.columns = [self.group_col] + [ f"{self.agg_col}_{agg_func}_by_{self.group_col}" for agg_func in self.agg_funcs @@ -323,15 +242,6 @@ class GroupStat(MLProcess): self.group_df = group_df def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ new_df = df.merge(self.group_df, on=self.group_col, how="left") return new_df @@ -355,25 +265,10 @@ class SplitBins(MLProcess): self.encoder = None def fit(self, df: pd.DataFrame): - """ - Fit the SplitBins model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ self.encoder = KBinsDiscretizer(strategy=self.strategy, encode="ordinal") self.encoder.fit(df[self.cols].fillna(0)) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ new_df = df.copy() new_df[self.cols] = self.encoder.transform(new_df[self.cols].fillna(0)) return new_df @@ -397,24 +292,9 @@ class ExtractTimeComps(MLProcess): self.time_comps = time_comps def fit(self, df: pd.DataFrame): - """ - Fit the ExtractTimeComps model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ pass def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ time_s = pd.to_datetime(df[self.time_col], errors="coerce") time_comps_df = pd.DataFrame() @@ -445,12 +325,6 @@ class GeneralSelection(MLProcess): self.feats = [] def fit(self, df: pd.DataFrame): - """ - Fit the GeneralSelection model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ feats = [f for f in df.columns if f != self.label_col] for col in df.columns: if df[col].isnull().sum() / df.shape[0] == 1: @@ -468,15 +342,6 @@ class GeneralSelection(MLProcess): self.feats = feats def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame contain label_col. - """ new_df = df[self.feats + [self.label_col]] return new_df @@ -501,12 +366,6 @@ class TreeBasedSelection(MLProcess): self.feats = None def fit(self, df: pd.DataFrame): - """ - Fit the TreeBasedSelection model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ params = { "boosting_type": "gbdt", "objective": "binary", @@ -538,15 +397,6 @@ class TreeBasedSelection(MLProcess): self.feats.append(self.label_col) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame contain label_col. - """ new_df = df[self.feats] return new_df @@ -571,12 +421,6 @@ class VarianceBasedSelection(MLProcess): self.selector = VarianceThreshold(threshold=self.threshold) def fit(self, df: pd.DataFrame): - """ - Fit the VarianceBasedSelection model. - - Args: - df (pd.DataFrame): The input DataFrame. - """ num_cols = df.select_dtypes(include=np.number).columns.tolist() cols = [f for f in num_cols if f not in [self.label_col]] @@ -585,14 +429,5 @@ class VarianceBasedSelection(MLProcess): self.feats.append(self.label_col) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame contain label_col. - """ new_df = df[self.feats] return new_df diff --git a/metagpt/tools/tool_convert.py b/metagpt/tools/tool_convert.py index b8377e67a..417a938e1 100644 --- a/metagpt/tools/tool_convert.py +++ b/metagpt/tools/tool_convert.py @@ -12,7 +12,8 @@ def convert_code_to_tool_schema(obj, include: list[str] = []): for name, method in inspect.getmembers(obj, inspect.isfunction): if include and name not in include: continue - method_doc = inspect.getdoc(method) + # method_doc = inspect.getdoc(method) + method_doc = get_class_method_docstring(obj, name) if method_doc: schema["methods"][name] = docstring_to_schema(method_doc) @@ -22,8 +23,6 @@ def convert_code_to_tool_schema(obj, include: list[str] = []): **docstring_to_schema(docstring), } - schema = {obj.__name__: schema} - return schema @@ -70,3 +69,13 @@ def docstring_to_schema(docstring: str): schema["returns"] = [{"type": ret[0], "description": remove_spaces(ret[1])} for ret in returns] return schema + + +def get_class_method_docstring(cls, method_name): + """Retrieve a method's docstring, searching the class hierarchy if necessary.""" + for base_class in cls.__mro__: + if method_name in base_class.__dict__: + method = base_class.__dict__[method_name] + if method.__doc__: + return method.__doc__ + return None # No docstring found in the class hierarchy diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index 5922e7f69..299d62ca3 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -39,7 +39,6 @@ class ToolRegistry(BaseModel): tool_type="other", tool_source_object=None, include_functions=[], - make_schema_if_not_exists=True, verbose=False, ): if self.has_tool(tool_name): @@ -57,19 +56,11 @@ class ToolRegistry(BaseModel): schema_path = schema_path or TOOL_SCHEMA_PATH / tool_type / f"{tool_name}.yml" - if not os.path.exists(schema_path): - if make_schema_if_not_exists: - logger.warning(f"no schema found, will make schema at {schema_path}") - schema_dict = make_schema(tool_source_object, include_functions, schema_path) - else: - logger.warning(f"no schema found at assumed schema_path {schema_path}, skip registering {tool_name}") - return - else: - with open(schema_path, "r", encoding="utf-8") as f: - schema_dict = yaml.safe_load(f) - if not schema_dict: + schemas = make_schema(tool_source_object, include_functions, schema_path) + + if not schemas: return - schemas = schema_dict.get(tool_name) or list(schema_dict.values())[0] + schemas["tool_path"] = tool_path # corresponding code file path of the tool try: ToolSchema(**schemas) # validation @@ -78,11 +69,13 @@ class ToolRegistry(BaseModel): # logger.warning( # f"{tool_name} schema not conforms to required format, but will be used anyway. Mismatch: {e}" # ) + tool = Tool(name=tool_name, path=tool_path, schemas=schemas, code=tool_code) self.tools[tool_name] = tool self.tools_by_types[tool_type][tool_name] = tool if verbose: logger.info(f"{tool_name} registered") + logger.info(f"schema made at {str(schema_path)}, can be used for checking") def has_tool(self, key: str) -> Tool: return key in self.tools @@ -107,12 +100,10 @@ class ToolRegistry(BaseModel): TOOL_REGISTRY = ToolRegistry(tool_types=ToolTypes) -def register_tool(tool_name: str = "", tool_type: str = "other", schema_path: str = "", **kwargs): +def register_tool(tool_type: str = "other", schema_path: str = "", **kwargs): """register a tool to registry""" - def decorator(cls, tool_name=tool_name): - tool_name = tool_name or cls.__name__ - + def decorator(cls): # Get the file path where the function / class is defined and the source code file_path = inspect.getfile(cls) if "metagpt" in file_path: @@ -120,7 +111,7 @@ def register_tool(tool_name: str = "", tool_type: str = "other", schema_path: st source_code = inspect.getsource(cls) TOOL_REGISTRY.register_tool( - tool_name=tool_name, + tool_name=cls.__name__, tool_path=file_path, schema_path=schema_path, tool_code=source_code, @@ -142,7 +133,6 @@ def make_schema(tool_source_object, include, path): # import json # with open(str(path).replace("yml", "json"), "w", encoding="utf-8") as f: # json.dump(schema, f, ensure_ascii=False, indent=4) - logger.info(f"schema made at {path}") except Exception as e: schema = {} logger.error(f"Fail to make schema: {e}") diff --git a/metagpt/utils/parse_docstring.py b/metagpt/utils/parse_docstring.py index 8a017e1f7..e91be8e75 100644 --- a/metagpt/utils/parse_docstring.py +++ b/metagpt/utils/parse_docstring.py @@ -5,7 +5,7 @@ from pydantic import BaseModel def remove_spaces(text): - return re.sub(r"\s+", " ", text) + return re.sub(r"\s+", " ", text).strip() class DocstringParser(BaseModel): diff --git a/tests/metagpt/tools/test_tool_convert.py b/tests/metagpt/tools/test_tool_convert.py index 1dad997bd..2ae2ea000 100644 --- a/tests/metagpt/tools/test_tool_convert.py +++ b/tests/metagpt/tools/test_tool_convert.py @@ -17,7 +17,7 @@ def test_docstring_to_schema(): pd.DataFrame: The transformed DataFrame. """ expected = { - "description": " Some test desc. ", + "description": "Some test desc.", "parameters": { "properties": { "features": {"type": "list", "description": "Columns to be processed."}, @@ -97,47 +97,45 @@ def dummy_fn(df: pd.DataFrame) -> dict: def test_convert_code_to_tool_schema_class(): expected = { - "DummyClass": { - "type": "class", - "description": "Completing missing values with simple strategies.", - "methods": { - "__init__": { - "description": "Initialize self. ", - "parameters": { - "properties": { - "features": {"type": "list", "description": "Columns to be processed."}, - "strategy": { - "type": "str", - "description": "The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.", - "default": "'mean'", - "enum": ["'mean'", "'median'", "'most_frequent'", "'constant'"], - }, - "fill_value": { - "type": "int", - "description": "Fill_value is used to replace all occurrences of missing_values. Defaults to None.", - "default": "None", - }, + "type": "class", + "description": "Completing missing values with simple strategies.", + "methods": { + "__init__": { + "description": "Initialize self.", + "parameters": { + "properties": { + "features": {"type": "list", "description": "Columns to be processed."}, + "strategy": { + "type": "str", + "description": "The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.", + "default": "'mean'", + "enum": ["'mean'", "'median'", "'most_frequent'", "'constant'"], + }, + "fill_value": { + "type": "int", + "description": "Fill_value is used to replace all occurrences of missing_values. Defaults to None.", + "default": "None", }, - "required": ["features"], }, - }, - "fit": { - "description": "Fit the FillMissingValue model. ", - "parameters": { - "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, - "required": ["df"], - }, - }, - "transform": { - "description": "Transform the input DataFrame with the fitted model. ", - "parameters": { - "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, - "required": ["df"], - }, - "returns": [{"type": "pd.DataFrame", "description": "The transformed DataFrame."}], + "required": ["features"], }, }, - } + "fit": { + "description": "Fit the FillMissingValue model.", + "parameters": { + "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, + "required": ["df"], + }, + }, + "transform": { + "description": "Transform the input DataFrame with the fitted model.", + "parameters": { + "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, + "required": ["df"], + }, + "returns": [{"type": "pd.DataFrame", "description": "The transformed DataFrame."}], + }, + }, } schema = convert_code_to_tool_schema(DummyClass) assert schema == expected @@ -145,14 +143,12 @@ def test_convert_code_to_tool_schema_class(): def test_convert_code_to_tool_schema_function(): expected = { - "dummy_fn": { - "type": "function", - "description": "Analyzes a DataFrame and categorizes its columns based on data types. ", - "parameters": { - "properties": {"df": {"type": "pd.DataFrame", "description": "The DataFrame to be analyzed."}}, - "required": ["df"], - }, - } + "type": "function", + "description": "Analyzes a DataFrame and categorizes its columns based on data types.", + "parameters": { + "properties": {"df": {"type": "pd.DataFrame", "description": "The DataFrame to be analyzed."}}, + "required": ["df"], + }, } schema = convert_code_to_tool_schema(dummy_fn) assert schema == expected diff --git a/tests/metagpt/tools/test_tool_registry.py b/tests/metagpt/tools/test_tool_registry.py index bb5d7a0bd..e41ddfa79 100644 --- a/tests/metagpt/tools/test_tool_registry.py +++ b/tests/metagpt/tools/test_tool_registry.py @@ -14,18 +14,6 @@ def tool_registry_full(): return ToolRegistry(tool_types=ToolTypes) -@pytest.fixture -def schema_yaml(mocker): - mock_yaml_content = """ - tool_name: - key1: value1 - key2: value2 - """ - mocker.patch("os.path.exists", return_value=True) - mocker.patch("builtins.open", mocker.mock_open(read_data=mock_yaml_content)) - return mocker - - # Test Initialization def test_initialization(tool_registry): assert isinstance(tool_registry, ToolRegistry) @@ -42,33 +30,46 @@ def test_initialize_with_tool_types(tool_registry_full): assert "data_preprocess" in tool_registry_full.tool_types -# Test Tool Registration -def test_register_tool(tool_registry, schema_yaml): - tool_registry.register_tool("TestTool", "/path/to/tool") - assert "TestTool" in tool_registry.tools +class TestClassTool: + """test class""" + + def test_class_fn(self): + """test class fn""" + pass -# Test Tool Registration with Non-existing Schema -def test_register_tool_no_schema(tool_registry, mocker): - mocker.patch("os.path.exists", return_value=False) - tool_registry.register_tool("TestTool", "/path/to/tool") - assert "TestTool" not in tool_registry.tools +def test_fn(): + """test function""" + pass + + +# Test Tool Registration Class +def test_register_tool_class(tool_registry): + tool_registry.register_tool("TestClassTool", "/path/to/tool", tool_source_object=TestClassTool) + assert "TestClassTool" in tool_registry.tools + + +# Test Tool Registration Function +def test_register_tool_fn(tool_registry): + tool_registry.register_tool("test_fn", "/path/to/tool", tool_source_object=test_fn) + assert "test_fn" in tool_registry.tools # Test Tool Existence Checks -def test_has_tool(tool_registry, schema_yaml): - tool_registry.register_tool("TestTool", "/path/to/tool") - assert tool_registry.has_tool("TestTool") +def test_has_tool(tool_registry): + tool_registry.register_tool("TestClassTool", "/path/to/tool", tool_source_object=TestClassTool) + assert tool_registry.has_tool("TestClassTool") assert not tool_registry.has_tool("NonexistentTool") # Test Tool Retrieval -def test_get_tool(tool_registry, schema_yaml): - tool_registry.register_tool("TestTool", "/path/to/tool") - tool = tool_registry.get_tool("TestTool") +def test_get_tool(tool_registry): + tool_registry.register_tool("TestClassTool", "/path/to/tool", tool_source_object=TestClassTool) + tool = tool_registry.get_tool("TestClassTool") assert tool is not None - assert tool.name == "TestTool" + assert tool.name == "TestClassTool" assert tool.path == "/path/to/tool" + assert "description" in tool.schemas # Similar tests for has_tool_type, get_tool_type, get_tools_by_type @@ -83,12 +84,12 @@ def test_get_tool_type(tool_registry_full): assert retrieved_type.name == "data_preprocess" -def test_get_tools_by_type(tool_registry, schema_yaml): +def test_get_tools_by_type(tool_registry): tool_type_name = "TestType" tool_name = "TestTool" tool_path = "/path/to/tool" - tool_registry.register_tool(tool_name, tool_path, tool_type=tool_type_name) + tool_registry.register_tool(tool_name, tool_path, tool_type=tool_type_name, tool_source_object=TestClassTool) tools_by_type = tool_registry.get_tools_by_type(tool_type_name) assert tools_by_type is not None diff --git a/tests/metagpt/utils/test_save_code.py b/tests/metagpt/utils/test_save_code.py index 5ab08c454..57a19049b 100644 --- a/tests/metagpt/utils/test_save_code.py +++ b/tests/metagpt/utils/test_save_code.py @@ -14,7 +14,7 @@ from metagpt.utils.save_code import DATA_PATH, save_code_file def test_save_code_file_python(): save_code_file("example", "print('Hello, World!')") file_path = DATA_PATH / "output" / "example" / "code.py" - assert file_path.exists, f"File does not exist: {file_path}" + assert file_path.exists(), f"File does not exist: {file_path}" content = file_path.read_text() assert "print('Hello, World!')" in content, "File content does not match" @@ -35,7 +35,7 @@ async def test_save_code_file_notebook(): # Save as a Notebook file save_code_file("example_nb", executor.nb, file_format="ipynb") file_path = DATA_PATH / "output" / "example_nb" / "code.ipynb" - assert file_path.exists, f"Notebook file does not exist: {file_path}" + assert file_path.exists(), f"Notebook file does not exist: {file_path}" # Additional checks specific to notebook format notebook = nbformat.read(file_path, as_version=4) From 55dac10146cc67e877ac7d91358b0c2a07e999ff Mon Sep 17 00:00:00 2001 From: yzlin Date: Sun, 4 Feb 2024 20:34:46 +0800 Subject: [PATCH 602/637] fix bug and update cache --- metagpt/tools/libs/data_preprocess.py | 2 +- tests/data/rsp_cache.json | 6 ++++++ 2 files changed, 7 insertions(+), 1 deletion(-) diff --git a/metagpt/tools/libs/data_preprocess.py b/metagpt/tools/libs/data_preprocess.py index 9c571ad6b..66f579f66 100644 --- a/metagpt/tools/libs/data_preprocess.py +++ b/metagpt/tools/libs/data_preprocess.py @@ -99,7 +99,7 @@ class DataPreprocessTool(MLProcess): @register_tool(tool_type=TOOL_TYPE) -class FillMissingValue(MLProcess): +class FillMissingValue(DataPreprocessTool): """ Completing missing values with simple strategies. """ diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index 2257878e2..f92fb42c0 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -389,5 +389,11 @@ }, "[{\"role\": \"user\", \"content\": \"Interface definition:\\n```text\\nInterface Name: Element Tagging\\nInterface Path: /projects/{project_key}/node-tags\\nMethod: POST\\n\\nRequest parameters:\\nPath parameters:\\nproject_key\\n\\nBody parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nnodes\\tarray\\tYes\\t\\tNodes\\n\\tnode_key\\tstring\\tNo\\t\\tNode key\\n\\ttags\\tarray\\tNo\\t\\tOriginal node tag list\\n\\tnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\noperations\\tarray\\tYes\\t\\t\\n\\ttags\\tarray\\tNo\\t\\tOperation tag list\\n\\tmode\\tstring\\tNo\\t\\tOperation type ADD / DELETE\\n\\nReturn data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tinteger\\tYes\\t\\tStatus code\\nmsg\\tstring\\tYes\\t\\tPrompt message\\ndata\\tobject\\tYes\\t\\tReturned data\\nlist\\tarray\\tNo\\t\\tNode list true / false\\nnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\nnode_key\\tstring\\tNo\\t\\tNode key\\n```\\n\\nUnit test:\\n```python\\n@pytest.mark.parametrize(\\n\\\"project_key, nodes, operations, expected_msg\\\",\\n[\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"success\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_002\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"tag1\\\"], \\\"mode\\\": \\\"DELETE\\\"}], \\\"success\\\"),\\n(\\\"\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Missing the required parameter project_key\\\"),\\n(123, [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Incorrect parameter type\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"a\\\"*201, \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Request parameter exceeds field boundary\\\")\\n]\\n)\\ndef test_node_tags(project_key, nodes, operations, expected_msg):\\n pass\\n\\n# The above is an interface definition and a unit test example.\\n# Next, please play the role of an expert test manager with 20 years of experience at Google. When I give the interface definition, \\n# reply to me with a unit test. There are several requirements:\\n# 1. Only output one `@pytest.mark.parametrize` and the corresponding test_ function (inside pass, do not implement).\\n# -- The function parameter contains expected_msg for result verification.\\n# 2. The generated test cases use shorter text or numbers and are as compact as possible.\\n# 3. If comments are needed, use Chinese.\\n\\n# If you understand, please wait for me to give the interface definition and just answer \\\"Understood\\\" to save tokens.\\n\"}, {\"role\": \"user\", \"content\": \"Refer to the test types: such as SQL injection, cross-site scripting (XSS), unauthorized access and privilege escalation, \\nauthentication and authorization, parameter verification, exception handling, file upload and download.\\nPlease output 10 test cases within one `@pytest.mark.parametrize` scope.\\n```text\\nAPI Name: 获取managed folder详情(job专用)\\nAPI Path: /v1/projects/{project_key}/jobs/{job_id}/folders/{folder_key}\\nMethod: GET\\n\\nRequest Parameters:\\nPath Parameters:\\nproject_key \\njob_id \\nfolder_key \\n\\nBody Parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nproject_key\\tstring\\tYes\\t\\t\\njob_id\\tstring\\tYes\\t\\t\\nfolder_key\\tstring\\tYes\\t\\t\\n\\nResponse Data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tnumber\\tYes\\t\\t0成功,非0失败\\nmsg\\tstring\\tYes\\t\\t失败时这里有错误信息\\ndata\\tobject\\tYes\\t\\t\\n\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\tfolder\\tobject\\tNo\\t\\tfolder配置在这里\\n\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\tobject_key\\tstring\\tNo\\t\\tobject key\\n\\t\\tname\\tstring\\tNo\\t\\t用户可编辑的那个name\\n\\t\\ttype\\tstring\\tNo\\t\\tfolder类型,与connection有关\\n\\t\\tparams\\tobject\\tNo\\t\\t数据读写相关配置在这里\\n\\t\\t\\tconnection\\tstring\\tNo\\t\\tconnection id\\n\\t\\t\\tpath\\tstring\\tNo\\t\\t文件夹内容存放的相对路径\\n\\t\\t\\tnot_ready_if_empty\\tboolean\\tNo\\t\\treserved\\n\\t\\t\\tfiles_selection_rules\\tobject\\tNo\\t\\t文件过滤规则\\n\\t\\t\\t\\tmode\\tstring\\tNo\\t\\tALL\\n\\t\\t\\t\\texclude_rules\\tarray\\tNo\\t\\t排除规则\\n\\t\\t\\t\\tinclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\texplicit_files\\tarray\\tNo\\t\\t\\n\\t\\tflow_options\\tobject\\tNo\\t\\tflow参数\\n\\t\\t\\tvirtualizable\\tboolean\\tNo\\t\\t\\n\\t\\t\\trebuild_behavior\\tstring\\tNo\\t\\t构建方式\\n\\t\\t\\tcross_project_build_behavior\\tstring\\tNo\\t\\t\\n\\t\\tmetrics\\tobject\\tNo\\t\\t\\n\\t\\t\\tprobes\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\ttype\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\tcompute_on_build_mode\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tmeta\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tname\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\tlevel\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\tconfiguration\\tobject\\tNo\\t\\t\\n\\t\\t\\tengine_config\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpad_runs_with_metrics\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\thive\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\textra_conf\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tbasic\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tdss\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\tselection\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tuse_mem_table\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tfilter\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tdistinct\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tpartition_selection_method\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tlatest_partitions_n\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tordering\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\trules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tsampling_method\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tmax_records\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\ttarget_ratio\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\twithin_first_n\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tmax_read_uncompressed_bytes\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\tsql\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\timpala\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\tspark\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\textra_conf\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tpython\\tobject\\tNo\\t\\t\\n\\t\\t\\tdisplayed_state\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpartition\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tcolumns\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tmetrics\\tarray\\tNo\\t\\t\\n\\t\\tchecks\\tobject\\tNo\\t\\t\\n\\t\\t\\trun_on_build\\tboolean\\tNo\\t\\t\\n\\t\\t\\tchecks\\tarray\\tNo\\t\\t\\n\\t\\t\\tdisplayed_state\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpartition\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tchecks\\tarray\\tNo\\t\\t\\n\\t\\tversion_tag\\tobject\\tNo\\t\\t配置版本信息\\n\\t\\t\\tversion_number\\tnumber\\tNo\\t\\t\\n\\t\\t\\tlast_modified_by\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tlogin\\tstring\\tNo\\t\\t\\n\\t\\t\\tlast_modified_on\\tnumber\\tNo\\t\\t修改时间unix time ms\\n\\t\\tcreation_tag\\tobject\\tNo\\t\\t配置创建时间\\n\\t\\t\\tversion_number\\tnumber\\tNo\\t\\t1\\n\\t\\t\\tlast_modified_by\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tlogin\\tstring\\tNo\\t\\t\\n\\t\\t\\tlast_modified_on\\tnumber\\tNo\\t\\t创建时间unix time ms\\n\\t\\ttags\\tarray\\tNo\\t\\t文件夹标签\\n\\t\\tcustom_fields\\tobject\\tNo\\t\\t\\n\\t\\tchecklists\\tobject\\tNo\\t\\t\\n\\t\\t\\tchecklists\\tarray\\tNo\\t\\t\\n\\n```\"}]": { "code": "import string\nimport random\n\ndef random_string(length=10):\n return ''.join(random.choice(string.ascii_lowercase) for i in range(length))" + }, + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { + "code": "# Data preprocessing\n\n# Step 1: Fill missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create a copy of the dataframe to avoid modifying the original one\ndf_processed = df.copy()\n\n# Identify numeric columns to fill missing values\nnumeric_cols = df_processed.select_dtypes(include='number').columns.tolist()\n\n# Initialize the FillMissingValue tool for numeric columns\nfill_missing_value = FillMissingValue(features=numeric_cols, strategy='mean')\n\n# Fit and transform the dataframe\ndf_processed = fill_missing_value.fit_transform(df_processed)\n\n# Step 2: Scale numeric columns\nfrom metagpt.tools.libs.data_preprocess import MinMaxScale\n\n# Initialize the MinMaxScale tool for numeric columns\nminmax_scale = MinMaxScale(features=numeric_cols)\n\n# Fit and transform the dataframe\ndf_processed = minmax_scale.fit_transform(df_processed)\n\n# Display the first few rows of the processed dataframe\ndf_processed.head()" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code_steps='', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code_steps='', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Instantiate the FillMissingValue class for column 'b'\nfill_missing_b = FillMissingValue(features=['b'], strategy='mean')\n\n# Fit and transform the DataFrame for column 'b'\ndf = fill_missing_b.fit_transform(df)" } } \ No newline at end of file From 4cfee3ba279253f99fa7f3cad8e4f4b786a56b49 Mon Sep 17 00:00:00 2001 From: yzlin Date: Sun, 4 Feb 2024 23:23:18 +0800 Subject: [PATCH 603/637] rm code steps, mv planner, mv ci mle --- examples/crawl_webpage.py | 2 +- examples/imitate_webpage.py | 2 +- examples/sd_tool_usage.py | 2 +- metagpt/actions/ci/ml_action.py | 3 - metagpt/actions/ci/write_analysis_code.py | 6 +- metagpt/plan/__init__.py | 0 metagpt/prompts/ci/ml_action.py | 17 +---- metagpt/prompts/ci/write_analysis_code.py | 2 - metagpt/prompts/tool_types.py | 4 +- metagpt/roles/{ => ci}/code_interpreter.py | 0 metagpt/roles/{ => ci}/ml_engineer.py | 2 +- metagpt/roles/role.py | 2 +- metagpt/schema.py | 3 - metagpt/{plan => strategy}/planner.py | 5 -- tests/data/rsp_cache.json | 65 +++++++++---------- .../actions/ci/test_write_analysis_code.py | 5 +- .../roles/{ => ci}/test_code_interpreter.py | 2 +- .../roles/{ => ci}/test_ml_engineer.py | 4 +- 18 files changed, 44 insertions(+), 82 deletions(-) delete mode 100644 metagpt/plan/__init__.py rename metagpt/roles/{ => ci}/code_interpreter.py (100%) rename metagpt/roles/{ => ci}/ml_engineer.py (97%) rename metagpt/{plan => strategy}/planner.py (94%) rename tests/metagpt/roles/{ => ci}/test_code_interpreter.py (90%) rename tests/metagpt/roles/{ => ci}/test_ml_engineer.py (96%) diff --git a/examples/crawl_webpage.py b/examples/crawl_webpage.py index 35413d2ff..7dcbf7993 100644 --- a/examples/crawl_webpage.py +++ b/examples/crawl_webpage.py @@ -5,7 +5,7 @@ @File : crawl_webpage.py """ -from metagpt.roles.code_interpreter import CodeInterpreter +from metagpt.roles.ci.code_interpreter import CodeInterpreter async def main(): diff --git a/examples/imitate_webpage.py b/examples/imitate_webpage.py index b69101861..5075e1e39 100644 --- a/examples/imitate_webpage.py +++ b/examples/imitate_webpage.py @@ -5,7 +5,7 @@ @Author : mannaandpoem @File : imitate_webpage.py """ -from metagpt.roles.code_interpreter import CodeInterpreter +from metagpt.roles.ci.code_interpreter import CodeInterpreter async def main(): diff --git a/examples/sd_tool_usage.py b/examples/sd_tool_usage.py index 92f4cd5b0..b4642af23 100644 --- a/examples/sd_tool_usage.py +++ b/examples/sd_tool_usage.py @@ -4,7 +4,7 @@ # @Desc : import asyncio -from metagpt.roles.code_interpreter import CodeInterpreter +from metagpt.roles.ci.code_interpreter import CodeInterpreter async def main(requirement: str = ""): diff --git a/metagpt/actions/ci/ml_action.py b/metagpt/actions/ci/ml_action.py index 6fecae898..9640a7918 100644 --- a/metagpt/actions/ci/ml_action.py +++ b/metagpt/actions/ci/ml_action.py @@ -25,7 +25,6 @@ class WriteCodeWithToolsML(WriteCodeWithTools): tool_schemas, tool_type_usage_prompt = await self._prepare_tools(plan=plan) # ML-specific variables to be used in prompt - code_steps = plan.current_task.code_steps finished_tasks = plan.get_finished_tasks() code_context = [remove_comments(task.code) for task in finished_tasks] code_context = "\n\n".join(code_context) @@ -38,7 +37,6 @@ class WriteCodeWithToolsML(WriteCodeWithTools): current_task=plan.current_task.instruction, column_info=column_info, tool_type_usage_prompt=tool_type_usage_prompt, - code_steps=code_steps, tool_schemas=tool_schemas, ) @@ -49,7 +47,6 @@ class WriteCodeWithToolsML(WriteCodeWithTools): current_task=plan.current_task.instruction, column_info=column_info, tool_type_usage_prompt=tool_type_usage_prompt, - code_steps=code_steps, ) tool_config = create_func_call_config(CODE_GENERATOR_WITH_TOOLS) rsp = await self.llm.aask_code(prompt, **tool_config) diff --git a/metagpt/actions/ci/write_analysis_code.py b/metagpt/actions/ci/write_analysis_code.py index 4e4ea7953..38fe107fd 100644 --- a/metagpt/actions/ci/write_analysis_code.py +++ b/metagpt/actions/ci/write_analysis_code.py @@ -79,7 +79,6 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): async def _recommend_tool( self, task: str, - code_steps: str, available_tools: dict, ) -> list: """ @@ -87,7 +86,6 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): Args: task (str): the task to recommend tools for - code_steps (str): the code steps to generate the full code for the task available_tools (dict): the available tools description Returns: @@ -95,7 +93,6 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): """ prompt = TOOL_RECOMMENDATION_PROMPT.format( current_task=task, - code_steps=code_steps, available_tools=available_tools, ) tool_config = create_func_call_config(SELECT_FUNCTION_TOOLS) @@ -132,8 +129,7 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): available_tools = self._get_tools_by_type(tool_type) if available_tools: available_tools = {tool_name: tool.schemas["description"] for tool_name, tool in available_tools.items()} - code_steps = plan.current_task.code_steps - tool_schemas = await self._recommend_tool(plan.current_task.instruction, code_steps, available_tools) + tool_schemas = await self._recommend_tool(plan.current_task.instruction, available_tools) return tool_schemas, tool_type_usage_prompt diff --git a/metagpt/plan/__init__.py b/metagpt/plan/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/metagpt/prompts/ci/ml_action.py b/metagpt/prompts/ci/ml_action.py index 582b01146..46d419dfb 100644 --- a/metagpt/prompts/ci/ml_action.py +++ b/metagpt/prompts/ci/ml_action.py @@ -84,15 +84,11 @@ Latest data info after previous tasks: Write complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc. Specifically, {tool_type_usage_prompt} -# Code Steps: -Strictly follow steps below when you writing code if it's convenient. -{code_steps} - # Output Example: -when current task is "train a lightgbm model on training data", and their are two steps in 'Code Steps', the code be like: +when current task is "train a lightgbm model on training data", the code can be like: ```python # Step 1: check data type and convert to numeric -ojb_cols = train.select_dtypes(include='object').columns.tolist() +obj_cols = train.select_dtypes(include='object').columns.tolist() for col in obj_cols: encoder = LabelEncoder() @@ -107,7 +103,6 @@ model.fit(train, y_train) # Constraints: - Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. -- The output code should contain all steps implemented in 'Code Steps'. """ ML_TOOL_USAGE_PROMPT = """ @@ -130,10 +125,6 @@ Latest data info after previous tasks: Write complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc. Specifically, {tool_type_usage_prompt} -# Code Steps: -Strictly follow steps below when you writing code if it's convenient. -{code_steps} - # Capabilities - You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class. - You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. @@ -143,7 +134,7 @@ Each Class tool is described in JSON format. When you call a tool, import the to {tool_schemas} # Output Example: -when current task is "do data preprocess, like fill missing value, handle outliers, etc.", and their are two steps in 'Code Steps', the code be like: +when current task is "do data preprocess, like fill missing value, handle outliers, etc.", the code can be like: ```python # Step 1: fill missing value # Tools used: ['FillMissingValue'] @@ -170,6 +161,4 @@ for col in num_cols: - Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. - Always prioritize using pre-defined tools for the same functionality. - Always copy the DataFrame before processing it and use the copy to process. -- The output code should contain all steps implemented correctly in 'Code Steps'. """ -# - If 'Code Steps' contains step done in 'Done Tasks', such as reading data, don't repeat it. diff --git a/metagpt/prompts/ci/write_analysis_code.py b/metagpt/prompts/ci/write_analysis_code.py index 4c8a5081e..15d8b1443 100644 --- a/metagpt/prompts/ci/write_analysis_code.py +++ b/metagpt/prompts/ci/write_analysis_code.py @@ -30,8 +30,6 @@ TOOL_RECOMMENDATION_PROMPT = """ ## Task Recommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. -This is a detailed code steps for current task. You can refer to it when recommending tools. -{code_steps} ## Available Tools: {available_tools} diff --git a/metagpt/prompts/tool_types.py b/metagpt/prompts/tool_types.py index 381fb25ad..f27fbea99 100644 --- a/metagpt/prompts/tool_types.py +++ b/metagpt/prompts/tool_types.py @@ -14,10 +14,10 @@ The current task is about data preprocessing, please note the following: FEATURE_ENGINEERING_PROMPT = """ The current task is about feature engineering. when performing it, please adhere to the following principles: - Generate as diverse features as possible to improve the model's performance step-by-step. -- If potential impactful features are not included in 'Code Steps', add new steps to generate them. +- Use available feature engineering tools if they are potential impactful. - Avoid creating redundant or excessively numerous features in one step. - Exclude ID columns from feature generation and remove them. -- Each step do feature engineering to train, must do same for test separately at the same time. +- Each feature engineering operation performed on the train set must also applies to the test separately at the same time. - Avoid using the label column to create features, except for cat encoding. - Use the data from previous task result if exist, do not mock or reload data yourself. """ diff --git a/metagpt/roles/code_interpreter.py b/metagpt/roles/ci/code_interpreter.py similarity index 100% rename from metagpt/roles/code_interpreter.py rename to metagpt/roles/ci/code_interpreter.py diff --git a/metagpt/roles/ml_engineer.py b/metagpt/roles/ci/ml_engineer.py similarity index 97% rename from metagpt/roles/ml_engineer.py rename to metagpt/roles/ci/ml_engineer.py index c7702771d..6fa6fe7b2 100644 --- a/metagpt/roles/ml_engineer.py +++ b/metagpt/roles/ci/ml_engineer.py @@ -2,7 +2,7 @@ from metagpt.actions.ci.debug_code import DebugCode from metagpt.actions.ci.execute_nb_code import ExecuteNbCode from metagpt.actions.ci.ml_action import UpdateDataColumns, WriteCodeWithToolsML from metagpt.logs import logger -from metagpt.roles.code_interpreter import CodeInterpreter +from metagpt.roles.ci.code_interpreter import CodeInterpreter from metagpt.tools.tool_types import ToolTypes from metagpt.utils.common import any_to_str diff --git a/metagpt/roles/role.py b/metagpt/roles/role.py index bcfec708c..3938664ba 100644 --- a/metagpt/roles/role.py +++ b/metagpt/roles/role.py @@ -33,9 +33,9 @@ from metagpt.actions.add_requirement import UserRequirement from metagpt.context_mixin import ContextMixin from metagpt.logs import logger from metagpt.memory import Memory -from metagpt.plan.planner import Planner from metagpt.provider import HumanProvider from metagpt.schema import Message, MessageQueue, SerializationMixin +from metagpt.strategy.planner import Planner from metagpt.utils.common import any_to_name, any_to_str, role_raise_decorator from metagpt.utils.project_repo import ProjectRepo from metagpt.utils.repair_llm_raw_output import extract_state_value_from_output diff --git a/metagpt/schema.py b/metagpt/schema.py index 1b0be279c..15854f676 100644 --- a/metagpt/schema.py +++ b/metagpt/schema.py @@ -335,7 +335,6 @@ class Task(BaseModel): dependent_task_ids: list[str] = [] # Tasks prerequisite to this Task instruction: str = "" task_type: str = "" - code_steps: str = "" code: str = "" result: str = "" is_success: bool = False @@ -348,7 +347,6 @@ class Task(BaseModel): self.is_finished = False def update_task_result(self, task_result: TaskResult): - self.code_steps = task_result.code_steps self.code = task_result.code self.result = task_result.result self.is_success = task_result.is_success @@ -357,7 +355,6 @@ class Task(BaseModel): class TaskResult(BaseModel): """Result of taking a task, with result and is_success required to be filled""" - code_steps: str = "" code: str = "" result: str is_success: bool diff --git a/metagpt/plan/planner.py b/metagpt/strategy/planner.py similarity index 94% rename from metagpt/plan/planner.py rename to metagpt/strategy/planner.py index 1b3971b7d..bcb0bda9b 100644 --- a/metagpt/plan/planner.py +++ b/metagpt/strategy/planner.py @@ -124,11 +124,6 @@ class Planner(BaseModel): def get_useful_memories(self, task_exclude_field=None) -> list[Message]: """find useful memories only to reduce context length and improve performance""" - # TODO dataset description , code steps - if task_exclude_field is None: - # Shorten the context as we don't need code steps after we get the codes. - # This doesn't affect current_task below, which should hold the code steps - task_exclude_field = {"code_steps"} user_requirement = self.plan.goal context = self.plan.context tasks = [task.dict(exclude=task_exclude_field) for task in self.plan.tasks] diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index f92fb42c0..c5f2e9643 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -316,11 +316,17 @@ "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.txt\n## Development Code\n```python\nresult = 'helloworld'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\n\n## Running Output\nstandard output: \n```text\nhelloworld\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nNo errors detected. The development code runs successfully and outputs the expected result without any errors.\n## File To Rewrite:\nNone\n## Status:\nPASS\n## Send To:\nNoOne\n---", "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.sh\n## Development Code\n```python\necho 'Hello World'\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\necho Hello World\n## Running Output\nstandard output: \n```text\nHello World\n\n```\nstandard errors: \n```text\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nNo errors detected in the running result. The development code executed successfully, and the output matches the expected result. Since there is no test code provided, no test execution was performed. Therefore, no specific instructions are needed for correction.\n## File To Rewrite:\nNone\n## Status:\nPASS\n## Send To:\nNoOne\n---", "\nRole: You are a senior development and qa engineer, your role is summarize the code running result.\nIf the running result does not include an error, you should explicitly approve the result.\nOn the other hand, if the running result indicates some error, you should point out which part, the development code or the test code, produces the error,\nand give specific instructions on fixing the errors. Here is the code info:\n\n## Development Code File Name\na.py\n## Development Code\n```python\npython -c \"print(1/0)\"\n```\n## Test File Name\n\n## Test Code\n```python\nNone\n```\n## Running Command\npython -c print(1/0)\n## Running Output\nstandard output: \n```text\n\n```\nstandard errors: \n```text\nTraceback (most recent call last):\n File \"\", line 1, in \nZeroDivisionError: division by zero\n\n```\n\nNow you should begin your analysis\n---\n## instruction:\nPlease summarize the cause of the errors and give correction instruction\n## File To Rewrite:\nDetermine the ONE file to rewrite in order to fix the error, for example, xyz.py, or test_xyz.py\n## Status:\nDetermine if all of the code works fine, if so write PASS, else FAIL,\nWRITE ONLY ONE WORD, PASS OR FAIL, IN THIS SECTION\n## Send To:\nPlease write NoOne if there are no errors, Engineer if the errors are due to problematic development codes, else QaEngineer,\nWRITE ONLY ONE WORD, NoOne OR Engineer OR QaEngineer, IN THIS SECTION.\n---\nYou should fill in necessary instruction, status, send to, and finally return all content between the --- segment line.\n": "---\n## instruction:\nThe error is caused by attempting to divide by zero in the development code. To fix this error, you should modify the division operation to avoid division by zero. For example, you can add a condition to check if the denominator is zero before performing the division.\n\n## File To Rewrite:\na.py\n\n## Status:\nFAIL\n\n## Send To:\nEngineer\n---", - "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nuser: Solve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\nTested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { - "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not handle negative numbers correctly. The binary representation of negative numbers in Python uses two's complement, which is not what we want for this sorting task. To fix this issue, we should modify the `count_ones` function to only work with non-negative numbers and ignore negative numbers. We should also ensure that negative numbers are sorted based on their decimal value, as mentioned in the context.", - "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n # Only count ones for non-negative integers\n return bin(n).count('1') if n >= 0 else 0\n \n # Separate the array into non-negative and negative numbers\n non_negatives = [x for x in arr if x >= 0]\n negatives = [x for x in arr if x < 0]\n \n # Sort non-negative numbers using a custom key function\n sorted_non_negatives = sorted(non_negatives, key=lambda x: (count_ones(x), x))\n \n # Sort negative numbers based on their decimal value\n sorted_negatives = sorted(negatives)\n \n # Combine the sorted non-negative and negative numbers\n sorted_arr = sorted_non_negatives + sorted_negatives\n \n return sorted_arr" + "[{\"role\": \"user\", \"content\": \"Interface definition:\\n```text\\nInterface Name: Element Tagging\\nInterface Path: /projects/{project_key}/node-tags\\nMethod: POST\\n\\nRequest parameters:\\nPath parameters:\\nproject_key\\n\\nBody parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nnodes\\tarray\\tYes\\t\\tNodes\\n\\tnode_key\\tstring\\tNo\\t\\tNode key\\n\\ttags\\tarray\\tNo\\t\\tOriginal node tag list\\n\\tnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\noperations\\tarray\\tYes\\t\\t\\n\\ttags\\tarray\\tNo\\t\\tOperation tag list\\n\\tmode\\tstring\\tNo\\t\\tOperation type ADD / DELETE\\n\\nReturn data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tinteger\\tYes\\t\\tStatus code\\nmsg\\tstring\\tYes\\t\\tPrompt message\\ndata\\tobject\\tYes\\t\\tReturned data\\nlist\\tarray\\tNo\\t\\tNode list true / false\\nnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\nnode_key\\tstring\\tNo\\t\\tNode key\\n```\\n\\nUnit test:\\n```python\\n@pytest.mark.parametrize(\\n\\\"project_key, nodes, operations, expected_msg\\\",\\n[\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"success\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_002\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"tag1\\\"], \\\"mode\\\": \\\"DELETE\\\"}], \\\"success\\\"),\\n(\\\"\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Missing the required parameter project_key\\\"),\\n(123, [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Incorrect parameter type\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"a\\\"*201, \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Request parameter exceeds field boundary\\\")\\n]\\n)\\ndef test_node_tags(project_key, nodes, operations, expected_msg):\\n pass\\n\\n# The above is an interface definition and a unit test example.\\n# Next, please play the role of an expert test manager with 20 years of experience at Google. When I give the interface definition, \\n# reply to me with a unit test. There are several requirements:\\n# 1. Only output one `@pytest.mark.parametrize` and the corresponding test_ function (inside pass, do not implement).\\n# -- The function parameter contains expected_msg for result verification.\\n# 2. The generated test cases use shorter text or numbers and are as compact as possible.\\n# 3. If comments are needed, use Chinese.\\n\\n# If you understand, please wait for me to give the interface definition and just answer \\\"Understood\\\" to save tokens.\\n\"}, {\"role\": \"user\", \"content\": \"Refer to the test types: such as SQL injection, cross-site scripting (XSS), unauthorized access and privilege escalation, \\nauthentication and authorization, parameter verification, exception handling, file upload and download.\\nPlease output 10 test cases within one `@pytest.mark.parametrize` scope.\\n```text\\nAPI Name: 获取 model 详情(job专用-后续开放给sdk)\\nAPI Path: /v1/projects/{project_key}/jobs/{job_id}/models/{model_key}\\nMethod: GET\\n\\nRequest Parameters:\\nPath Parameters:\\nproject_key \\njob_id \\nmodel_key \\n\\nBody Parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nproject_key\\tstring\\tYes\\t\\t\\njob_id\\tstring\\tYes\\t\\t\\nmodel_key\\tstring\\tYes\\t\\t\\n\\nResponse Data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tnumber\\tYes\\t\\t0成功,非0失败\\nmsg\\tstring\\tYes\\t\\t如果失败,这里有错误信息\\ndata\\tobject\\tYes\\t\\tdata信息\\n\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\tname\\tstring\\tNo\\t\\t用户可修改的name\\n\\tmodel\\tobject\\tNo\\t\\tmodel信息\\n\\t\\ttype\\tstring\\tNo\\t\\tdataset type\\n\\t\\tmanaged\\tboolean\\tNo\\t\\t为false时是第一类dataset,数据不可删除\\n\\t\\tname\\tstring\\tNo\\t\\t用户可修改的name\\n\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\tformat_type\\tstring\\tNo\\t\\t文件类型的dataset才有这项。“csv”\\n\\t\\tflow_options\\tobject\\tNo\\t\\t创建dataset时的高级设置\\n\\t\\t\\tvirtualizable\\tboolean\\tNo\\t\\t高级设置里的参数。缺省false\\n\\t\\t\\trebuild_behavior\\tstring\\tNo\\t\\t高级设置里的参数。缺省NORMAL\\n\\t\\t\\tcross_project_build_behavior\\tstring\\tNo\\t\\t高级设置里的参数。缺省DEFAULT\\n\\t\\tformat_params\\tobject\\tNo\\t\\t文件类型的dataset才有\\n\\t\\t\\tstyle\\tstring\\tNo\\t\\t\\n\\t\\t\\tcharset\\tstring\\tNo\\t\\t\\n\\t\\t\\tseparator\\tstring\\tNo\\t\\t\\n\\t\\t\\tquote_char\\tstring\\tNo\\t\\t\\n\\t\\t\\tescape_char\\tstring\\tNo\\t\\t\\n\\t\\t\\tdate_serialization_format\\tstring\\tNo\\t\\t\\n\\t\\t\\tarray_map_format\\tstring\\tNo\\t\\t\\n\\t\\t\\thive_separators\\tarray\\tNo\\t\\t\\n\\t\\t\\tskip_rows_before_header\\tnumber\\tNo\\t\\t\\n\\t\\t\\tparse_header_row\\tboolean\\tNo\\t\\t\\n\\t\\t\\tskip_rows_after_header\\tnumber\\tNo\\t\\t\\n\\t\\t\\tprobable_number_of_records\\tnumber\\tNo\\t\\t\\n\\t\\t\\tnormalize_booleans\\tboolean\\tNo\\t\\t\\n\\t\\t\\tnormalize_doubles\\tboolean\\tNo\\t\\t\\n\\t\\ttags\\tarray\\tNo\\t\\t标签tags\\n\\t\\tparams\\tobject\\tNo\\t\\t必有这项,但不同类型的dataset里面的key有差别\\n\\t\\t\\tconnection\\tstring\\tNo\\t\\tconnection id,到db查其他参数\\n\\t\\t\\tpath\\tstring\\tNo\\t\\t文件类connection才有这项\\n\\t\\t\\ttable\\tstring\\tNo\\t\\tdb表名,DB类connection才有这项\\n\\t\\t\\tmode\\tstring\\tNo\\t\\t存储类型,比如“table\\\",DB类connection才有这项\\n\\t\\t\\tbucket\\tstring\\tNo\\t\\tS3类型的connection才有这项\\n\\t\\t\\tkey_name\\tstring\\tNo\\t\\tredis才有,key name\\n\\t\\t\\tkey_type\\tstring\\tNo\\t\\tredis才有,key type\\n\\t\\t\\tcollection\\tstring\\tNo\\t\\t非关系型数据库才有,collection name\\n\\t\\t\\tindex\\tstring\\tNo\\t\\t索引类型的才有这项\\n\\t\\t\\tnot_ready_if_empty\\tboolean\\tNo\\t\\t数据非空才认为是data ready\\n\\t\\t\\tfiles_selection_rules\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tmode\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\texclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tinclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\texplicit_files\\tarray\\tNo\\t\\t\\n\\t\\tschema\\tobject\\tNo\\t\\tcolumns信息在这里\\n\\t\\t\\tcolumns\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tname\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\ttype\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\torigin_type\\tstring\\tNo\\t\\t\\n\\t\\t\\tuser_modified\\tboolean\\tNo\\t\\t\\n\\t\\tcustom_fields\\tobject\\tNo\\t\\t自定义fields\\n\\t\\tlast_build\\tobject\\tNo\\t\\t最后一次构建的信息\\n\\t\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\t\\tid\\tstring\\tNo\\t\\tactivity id\\n\\t\\t\\tjob_id\\tstring\\tNo\\t\\tjob id\\n\\t\\t\\tjob_project_key\\tstring\\tNo\\t\\t\\n\\t\\t\\tbuild_start_time\\tnumber\\tNo\\t\\t构建开始时间\\n\\t\\t\\tbuild_end_time\\tnumber\\tNo\\t\\t构建结束时间\\n\\t\\t\\tbuild_success\\tstring\\tNo\\t\\tsuccess或failed\\n\\t\\tobject_key\\tstring\\tNo\\t\\tdataset_key,后台用的id,用户不可见不可改\\n\\t\\tcache\\tobject\\tNo\\t\\t下载缓存数据链接\\n\\t\\t\\ts3_path\\tstring\\tNo\\t\\t\\n\\tstatus\\tobject\\tNo\\t\\t数据状态\\n\\t\\tsize\\tobject\\tNo\\t\\t数据大小信息\\n\\t\\t\\ttotal_value\\tnumber\\tNo\\t\\t占多少字节磁盘\\n\\t\\t\\tlast_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\tfirst_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\thas_data\\tboolean\\tNo\\t\\t是否有数据,这个影响前端的图标显示\\n\\t\\t\\tincomplete\\tboolean\\tNo\\t\\t\\n\\t\\trecords\\tobject\\tNo\\t\\t\\n\\t\\t\\ttotal_value\\tnumber\\tNo\\t\\t\\n\\t\\t\\tlast_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\tfirst_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\thas_data\\tboolean\\tNo\\t\\t是否有数据,这个影响前端的图标显示\\n\\t\\t\\tincomplete\\tboolean\\tNo\\t\\t\\n\\t\\tpartitions_last_compute\\tnumber\\tNo\\t\\t\\n\\t\\tpartitions\\tnumber\\tNo\\t\\t\\n\\tbuildable\\tboolean\\tNo\\t\\t有recipe时为true\\n\\theaders\\tarray\\tNo\\t\\t\\n\\t\\tdataset_schema\\tobject\\tNo\\t\\t\\n\\t\\t\\tname\\tstring\\tNo\\t字段名称\\t\\n\\t\\t\\ttype\\tstring\\tNo\\t字段类型\\t\\n\\t\\tnormal_rate\\tobject\\tNo\\t缺失值统计信息\\t\\n\\n```\"}]": { + "code": "import string\nimport random\n\ndef random_string(length=10):\n return ''.join(random.choice(string.ascii_lowercase) for i in range(length))" }, - "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\n对数据集进行数据清洗\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Completing missing values with simple strategies.', 'MinMaxScale': 'Transform features by scaling each feature to a range, which is (0, 1).', 'StandardScale': 'Standardize features by removing the mean and scaling to unit variance.', 'MaxAbsScale': 'Scale each feature by its maximum absolute value.', 'RobustScale': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'OrdinalEncode': 'Encode categorical features as ordinal integers.', 'OneHotEncode': 'Apply one-hot encoding to specified categorical columns, the original columns will be dropped.', 'LabelEncode': 'Apply label encoding to specified categorical columns in-place.'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { + "[{\"role\": \"user\", \"content\": \"Interface definition:\\n```text\\nInterface Name: Element Tagging\\nInterface Path: /projects/{project_key}/node-tags\\nMethod: POST\\n\\nRequest parameters:\\nPath parameters:\\nproject_key\\n\\nBody parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nnodes\\tarray\\tYes\\t\\tNodes\\n\\tnode_key\\tstring\\tNo\\t\\tNode key\\n\\ttags\\tarray\\tNo\\t\\tOriginal node tag list\\n\\tnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\noperations\\tarray\\tYes\\t\\t\\n\\ttags\\tarray\\tNo\\t\\tOperation tag list\\n\\tmode\\tstring\\tNo\\t\\tOperation type ADD / DELETE\\n\\nReturn data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tinteger\\tYes\\t\\tStatus code\\nmsg\\tstring\\tYes\\t\\tPrompt message\\ndata\\tobject\\tYes\\t\\tReturned data\\nlist\\tarray\\tNo\\t\\tNode list true / false\\nnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\nnode_key\\tstring\\tNo\\t\\tNode key\\n```\\n\\nUnit test:\\n```python\\n@pytest.mark.parametrize(\\n\\\"project_key, nodes, operations, expected_msg\\\",\\n[\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"success\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_002\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"tag1\\\"], \\\"mode\\\": \\\"DELETE\\\"}], \\\"success\\\"),\\n(\\\"\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Missing the required parameter project_key\\\"),\\n(123, [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Incorrect parameter type\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"a\\\"*201, \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Request parameter exceeds field boundary\\\")\\n]\\n)\\ndef test_node_tags(project_key, nodes, operations, expected_msg):\\n pass\\n\\n# The above is an interface definition and a unit test example.\\n# Next, please play the role of an expert test manager with 20 years of experience at Google. When I give the interface definition, \\n# reply to me with a unit test. There are several requirements:\\n# 1. Only output one `@pytest.mark.parametrize` and the corresponding test_ function (inside pass, do not implement).\\n# -- The function parameter contains expected_msg for result verification.\\n# 2. The generated test cases use shorter text or numbers and are as compact as possible.\\n# 3. If comments are needed, use Chinese.\\n\\n# If you understand, please wait for me to give the interface definition and just answer \\\"Understood\\\" to save tokens.\\n\"}, {\"role\": \"user\", \"content\": \"Refer to the test types: such as SQL injection, cross-site scripting (XSS), unauthorized access and privilege escalation, \\nauthentication and authorization, parameter verification, exception handling, file upload and download.\\nPlease output 10 test cases within one `@pytest.mark.parametrize` scope.\\n```text\\nAPI Name: 获取managed folder详情(job专用)\\nAPI Path: /v1/projects/{project_key}/jobs/{job_id}/folders/{folder_key}\\nMethod: GET\\n\\nRequest Parameters:\\nPath Parameters:\\nproject_key \\njob_id \\nfolder_key \\n\\nBody Parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nproject_key\\tstring\\tYes\\t\\t\\njob_id\\tstring\\tYes\\t\\t\\nfolder_key\\tstring\\tYes\\t\\t\\n\\nResponse Data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tnumber\\tYes\\t\\t0成功,非0失败\\nmsg\\tstring\\tYes\\t\\t失败时这里有错误信息\\ndata\\tobject\\tYes\\t\\t\\n\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\tfolder\\tobject\\tNo\\t\\tfolder配置在这里\\n\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\tobject_key\\tstring\\tNo\\t\\tobject key\\n\\t\\tname\\tstring\\tNo\\t\\t用户可编辑的那个name\\n\\t\\ttype\\tstring\\tNo\\t\\tfolder类型,与connection有关\\n\\t\\tparams\\tobject\\tNo\\t\\t数据读写相关配置在这里\\n\\t\\t\\tconnection\\tstring\\tNo\\t\\tconnection id\\n\\t\\t\\tpath\\tstring\\tNo\\t\\t文件夹内容存放的相对路径\\n\\t\\t\\tnot_ready_if_empty\\tboolean\\tNo\\t\\treserved\\n\\t\\t\\tfiles_selection_rules\\tobject\\tNo\\t\\t文件过滤规则\\n\\t\\t\\t\\tmode\\tstring\\tNo\\t\\tALL\\n\\t\\t\\t\\texclude_rules\\tarray\\tNo\\t\\t排除规则\\n\\t\\t\\t\\tinclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\texplicit_files\\tarray\\tNo\\t\\t\\n\\t\\tflow_options\\tobject\\tNo\\t\\tflow参数\\n\\t\\t\\tvirtualizable\\tboolean\\tNo\\t\\t\\n\\t\\t\\trebuild_behavior\\tstring\\tNo\\t\\t构建方式\\n\\t\\t\\tcross_project_build_behavior\\tstring\\tNo\\t\\t\\n\\t\\tmetrics\\tobject\\tNo\\t\\t\\n\\t\\t\\tprobes\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\ttype\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\tcompute_on_build_mode\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tmeta\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tname\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\tlevel\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\tconfiguration\\tobject\\tNo\\t\\t\\n\\t\\t\\tengine_config\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpad_runs_with_metrics\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\thive\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\textra_conf\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tbasic\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tdss\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\tselection\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tuse_mem_table\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tfilter\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tdistinct\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tpartition_selection_method\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tlatest_partitions_n\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tordering\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\trules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tsampling_method\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tmax_records\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\ttarget_ratio\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\twithin_first_n\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tmax_read_uncompressed_bytes\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\tsql\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\timpala\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\tspark\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\textra_conf\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tpython\\tobject\\tNo\\t\\t\\n\\t\\t\\tdisplayed_state\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpartition\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tcolumns\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tmetrics\\tarray\\tNo\\t\\t\\n\\t\\tchecks\\tobject\\tNo\\t\\t\\n\\t\\t\\trun_on_build\\tboolean\\tNo\\t\\t\\n\\t\\t\\tchecks\\tarray\\tNo\\t\\t\\n\\t\\t\\tdisplayed_state\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpartition\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tchecks\\tarray\\tNo\\t\\t\\n\\t\\tversion_tag\\tobject\\tNo\\t\\t配置版本信息\\n\\t\\t\\tversion_number\\tnumber\\tNo\\t\\t\\n\\t\\t\\tlast_modified_by\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tlogin\\tstring\\tNo\\t\\t\\n\\t\\t\\tlast_modified_on\\tnumber\\tNo\\t\\t修改时间unix time ms\\n\\t\\tcreation_tag\\tobject\\tNo\\t\\t配置创建时间\\n\\t\\t\\tversion_number\\tnumber\\tNo\\t\\t1\\n\\t\\t\\tlast_modified_by\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tlogin\\tstring\\tNo\\t\\t\\n\\t\\t\\tlast_modified_on\\tnumber\\tNo\\t\\t创建时间unix time ms\\n\\t\\ttags\\tarray\\tNo\\t\\t文件夹标签\\n\\t\\tcustom_fields\\tobject\\tNo\\t\\t\\n\\t\\tchecklists\\tobject\\tNo\\t\\t\\n\\t\\t\\tchecklists\\tarray\\tNo\\t\\t\\n\\n```\"}]": { + "code": "import string\nimport random\n\ndef random_string(length=10):\n return ''.join(random.choice(string.ascii_lowercase) for i in range(length))" + }, + "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nuser: Solve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\nTested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { + "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not correctly sort the array based on the number of ones in their binary representation and then by their decimal value. The current implementation sorts the array primarily by the number of ones in the binary representation, but it does not handle the secondary sorting by decimal value correctly for negative numbers, as it does not account for the fact that the binary representation of negative numbers in Python is different due to two's complement representation. To fix this issue, we should modify the count_ones function to correctly handle negative numbers by taking the absolute value before converting to binary and counting ones. Additionally, we should ensure that the array is sorted by the absolute value when the number of ones is the same.", + "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(abs(n)).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, absolute value, original value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their absolute value\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), abs(x), x))\n \n return sorted_arr" + }, + "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\n对数据集进行数据清洗\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\n\\n## Available Tools:\\n{'FillMissingValue': 'Completing missing values with simple strategies.', 'MinMaxScale': 'Transform features by scaling each feature to a range, which is (0, 1).', 'StandardScale': 'Standardize features by removing the mean and scaling to unit variance.', 'MaxAbsScale': 'Scale each feature by its maximum absolute value.', 'RobustScale': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'OrdinalEncode': 'Encode categorical features as ordinal integers.', 'OneHotEncode': 'Apply one-hot encoding to specified categorical columns, the original columns will be dropped.', 'LabelEncode': 'Apply label encoding to specified categorical columns in-place.'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { "recommend_tools": [ "FillMissingValue", "MinMaxScale", @@ -329,16 +335,16 @@ "RobustScale" ] }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MinMaxScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the StandardScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MaxAbsScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize the RobustScale instance with feature names. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'List of feature names to be scaled.'}}, 'required': ['features']}}, 'fit': {'description': 'Compute the median and IQR for scaling. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Scale features using the previously computed median and IQR. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features to be scaled.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'A new dataframe with scaled features.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { - "code": "# Step 1: Fill missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create a copy of the dataframe to avoid modifying the original one\ndf_processed = df.copy()\n\n# Identify numeric columns to fill missing values with mean\nnumeric_cols = df_processed.select_dtypes(include='number').columns.tolist()\nfill_missing_value_numeric = FillMissingValue(features=numeric_cols, strategy='mean')\ndf_processed = fill_missing_value_numeric.fit_transform(df_processed)\n\n# Step 2: Scale numeric features\nfrom metagpt.tools.libs.data_preprocess import MinMaxScale\n\n# Initialize the MinMaxScaler and scale the numeric columns\nminmax_scaler = MinMaxScale(features=numeric_cols)\ndf_processed = minmax_scaler.fit_transform(df_processed)\n\n# Step 3: Encode categorical features\n# For simplicity, we'll use label encoding for this example\nfrom sklearn.preprocessing import LabelEncoder\n\nlabel_encoder = LabelEncoder()\ndf_processed['c'] = label_encoder.fit_transform(df_processed['c'])\n\n# The dataset is now preprocessed and ready for further analysis or modeling." + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", the code can be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n\"}]": { + "code": "# Step 1: Fill missing values for numerical columns\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create a copy of the DataFrame to avoid modifying the original one\ndf_processed = df.copy()\n\n# Identify numerical columns with missing values\nnum_cols_with_missing = df_processed.select_dtypes(include='number').columns[df_processed.isnull().any()].tolist()\n\n# Initialize the FillMissingValue tool for numerical columns\nfill_missing_num = FillMissingValue(features=num_cols_with_missing, strategy='mean')\nfill_missing_num.fit(df_processed)\ndf_processed = fill_missing_num.transform(df_processed)\n\n# Step 2: Encode categorical columns\n# For simplicity, we will replace categorical values with a numerical code\n# This is a simple form of encoding that can be useful for tree-based models\n# and doesn't increase dimensionality like one-hot encoding\ncat_cols = df_processed.select_dtypes(include='object').columns.tolist()\nfor col in cat_cols:\n df_processed[col] = df_processed[col].astype('category').cat.codes\n\n# Display the processed DataFrame\ndisplay(df_processed)" }, - "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nclean and preprocess the data\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\nThis is a detailed code steps for current task. You can refer to it when recommending tools.\\n\\n\\n## Available Tools:\\n{'FillMissingValue': 'Filling missing values', 'SplitBins': 'Bin continuous data into intervals and return the bin identifier encoded as an integer value'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { + "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nclean and preprocess the data\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\n\\n## Available Tools:\\n{'FillMissingValue': 'Filling missing values', 'SplitBins': 'Bin continuous data into intervals and return the bin identifier encoded as an integer value'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { "recommend_tools": [ "FillMissingValue" ] }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code_steps='', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code_steps='', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the FillMissingValue model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MinMaxScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the StandardScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit the MaxAbsScale model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize the RobustScale instance with feature names. ', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'List of feature names to be scaled.'}}, 'required': ['features']}}, 'fit': {'description': 'Compute the median and IQR for scaling. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Scale features using the previously computed median and IQR. ', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'Dataframe containing the features to be scaled.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'A new dataframe with scaled features.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Fill missing values in column 'b' with the mean value\nfill_missing = FillMissingValue(features=['b'], strategy='mean')\ndf = fill_missing.fit_transform(df)\n\n# Check the DataFrame after filling missing values\ndf" + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Instantiate the FillMissingValue class for column 'b'\nfill_missing_b = FillMissingValue(features=['b'], strategy='mean')\n\n# Fit and transform the DataFrame\ndf = fill_missing_b.fit_transform(df)" }, "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n ## User Requirement\\n read a dataset test.csv and print its head\\n ## Current Plan\\n [\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n }\\n ]\\n ## Current Task\\n {\\\"task_id\\\": \\\"1\\\", \\\"dependent_task_ids\\\": [], \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\", \\\"task_type\\\": \\\"\\\", \\\"code\\\": \\\"\\\", \\\"result\\\": \\\"\\\", \\\"is_finished\\\": false}\\n \"}, {\"role\": \"assistant\", \"content\": \"import pandas as pd\\ndata = pd.read_excel('test.csv')\\ndata\"}, {\"role\": \"user\", \"content\": \"\\n Traceback (most recent call last):\\n File \\\"\\\", line 2, in \\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 478, in read_excel\\n io = ExcelFile(io, storage_options=storage_options, engine=engine)\\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 1500, in __init__\\n raise ValueError(\\n ValueError: Excel file format cannot be determined, you must specify an engine manually.\\n \"}]": { "language": "python", @@ -348,52 +354,39 @@ "language": "python", "code": "print(data.head())" }, - "\n # Context:\n user: run analysis on sklearn iris dataset\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 5 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Import the Iris dataset from scikit-learn.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis to understand the dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Preprocess the data if necessary (e.g., scaling, encoding).\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Split the dataset into training and testing sets.\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Choose a suitable model and train it on the dataset.\"\n },\n {\n \"task_id\": \"6\",\n \"dependent_task_ids\": [\"5\"],\n \"instruction\": \"Evaluate the model's performance on the test set.\"\n },\n {\n \"task_id\": \"7\",\n \"dependent_task_ids\": [\"6\"],\n \"instruction\": \"Report the results of the analysis.\"\n }\n]\n```", - "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Import the Iris dataset from scikit-learn.\\nTask 2: Perform exploratory data analysis to understand the dataset.\\nTask 3: Preprocess the data if necessary (e.g., scaling, encoding).\\nTask 4: Split the dataset into training and testing sets.\\nTask 5: Choose a suitable model and train it on the dataset.\\nTask 6: Evaluate the model's performance on the test set.\\nTask 7: Report the results of the analysis.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { + "\n # Context:\n user: run analysis on sklearn iris dataset\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 5 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Import the Iris dataset from sklearn.datasets\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis to understand the dataset\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Preprocess the data to prepare it for modeling\"\n },\n {\n \"task_id\": \"4\",\n \"dependent_task_ids\": [\"3\"],\n \"instruction\": \"Split the dataset into training and testing sets\"\n },\n {\n \"task_id\": \"5\",\n \"dependent_task_ids\": [\"4\"],\n \"instruction\": \"Train a classifier using the training set and evaluate it using the test set\"\n }\n]\n```", + "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Import the Iris dataset from sklearn.datasets\\nTask 2: Perform exploratory data analysis to understand the dataset\\nTask 3: Preprocess the data to prepare it for modeling\\nTask 4: Split the dataset into training and testing sets\\nTask 5: Train a classifier using the training set and evaluate it using the test set\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { "task_type": [ "other", "eda", "data_preprocess", "data_preprocess", "model_train", - "model_evaluate", - "other" + "model_evaluate" ] }, - "\n # Context:\n user: \n## User Requirement\nRun data analysis on sklearn Iris dataset, include a plot\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Load the sklearn Iris dataset.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis on the Iris dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Create a plot visualizing the Iris dataset features.\"\n }\n]\n```", - "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Load the sklearn Iris dataset.\\nTask 2: Perform exploratory data analysis on the Iris dataset.\\nTask 3: Create a plot visualizing the Iris dataset features.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { + "\n # Context:\n user: \n## User Requirement\nRun data analysis on sklearn Iris dataset, include a plot\n## Context\n\n## Current Plan\n[]\n## Current Task\n{}\n\n # Task:\n Based on the context, write a plan or modify an existing plan of what you should do to achieve the goal. A plan consists of one to 3 tasks.\n If you are modifying an existing plan, carefully follow the instruction, don't make unnecessary changes. Give the whole plan unless instructed to modify only one task of the plan.\n If you encounter errors on the current task, revise and output the current single task only.\n Output a list of jsons following the format:\n ```json\n [\n {\n \"task_id\": str = \"unique identifier for a task in plan, can be an ordinal\",\n \"dependent_task_ids\": list[str] = \"ids of tasks prerequisite to this task\",\n \"instruction\": \"what you should do in this task, one short phrase or sentence\",\n },\n ...\n ]\n ```\n ": "```json\n[\n {\n \"task_id\": \"1\",\n \"dependent_task_ids\": [],\n \"instruction\": \"Load the sklearn Iris dataset.\"\n },\n {\n \"task_id\": \"2\",\n \"dependent_task_ids\": [\"1\"],\n \"instruction\": \"Perform exploratory data analysis on the Iris dataset.\"\n },\n {\n \"task_id\": \"3\",\n \"dependent_task_ids\": [\"2\"],\n \"instruction\": \"Create a plot visualizing the Iris dataset.\"\n }\n]\n```", + "[{\"role\": \"user\", \"content\": \"\\nPlease assign a task type to each task in the list below from the given categories:\\nTask 1: Load the sklearn Iris dataset.\\nTask 2: Perform exploratory data analysis on the Iris dataset.\\nTask 3: Create a plot visualizing the Iris dataset.\\n\\n## All Task Type:\\n- **eda**: For performing exploratory data analysis\\n- **data_preprocess**: Only for changing value inplace.\\n- **feature_engineering**: Only for creating new columns for input data.\\n- **model_train**: Only for training model.\\n- **model_evaluate**: Only for evaluating model.\\n- **stable_diffusion**: Related to text2image, image2image using stable diffusion model.\\n- **image2webpage**: For converting image into webpage code.\\n- **web_scraping**: For scraping data from web pages.\\n- **other**: Any tools not in the defined categories\\n\"}]": { "task_type": [ "other", "eda", "other" ] }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"1\\\",\\\"dependent_task_ids\\\":[],\\\"instruction\\\":\\\"Load the sklearn Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"1\\\",\\\"dependent_task_ids\\\":[],\\\"instruction\\\":\\\"Load the sklearn Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { "code": "from sklearn.datasets import load_iris\niris_data = load_iris()" }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "import pandas as pd\n\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\niris_df['target'] = iris_data.target\n\n# Display basic information about the dataset\niris_df_info = iris_df.info()\n\n# Display statistical summary of the dataset\niris_df_description = iris_df.describe()\n\n# Show the first few rows of the dataset\ndf_head = iris_df.head()\n\n# Output the results\ndisplay(iris_df_info, iris_df_description, df_head)" + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "import pandas as pd\nimport numpy as np\n\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\niris_df = pd.DataFrame(data=np.c_[iris_data['data'], iris_data['target']], columns=iris_data['feature_names'] + ['target'])\n\n# Show basic information about the dataset\ndataset_info = iris_df.info()\n\n# Show basic statistics of the dataset\ndataset_description = iris_df.describe()\n\n# Show the first few rows of the dataset\ndataset_head = iris_df.head()\n\n# Count the number of samples for each class\nclass_distribution = iris_df['target'].value_counts()\n\n# Output the results\ndataset_info, dataset_description, dataset_head, class_distribution" }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\n\\\\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\\\\niris_df['target'] = iris_data.target\\\\n\\\\n# Display basic information about the dataset\\\\niris_df_info = iris_df.info()\\\\n\\\\n# Display statistical summary of the dataset\\\\niris_df_description = iris_df.describe()\\\\n\\\\n# Show the first few rows of the dataset\\\\ndf_head = iris_df.head()\\\\n\\\\n# Output the results\\\\ndisplay(iris_df_info, iris_df_description, df_head)\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset features.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset features.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "import seaborn as sns\nimport matplotlib.pyplot as plt\n\n# Pairplot to visualize the relationships between features\nsns.pairplot(iris_df, hue='target', diag_kind='kde')\n\n# Show the plot\nplt.show()" + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\nimport numpy as np\\\\n\\\\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=np.c_[iris_data['data'], iris_data['target']], columns=iris_data['feature_names'] + ['target'])\\\\n\\\\n# Show basic information about the dataset\\\\ndataset_info = iris_df.info()\\\\n\\\\n# Show basic statistics of the dataset\\\\ndataset_description = iris_df.describe()\\\\n\\\\n# Show the first few rows of the dataset\\\\ndataset_head = iris_df.head()\\\\n\\\\n# Count the number of samples for each class\\\\nclass_distribution = iris_df['target'].value_counts()\\\\n\\\\n# Output the results\\\\ndataset_info, dataset_description, dataset_head, class_distribution\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "import matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Set the style of seaborn\nsns.set(style=\"whitegrid\")\n\n# Create a pairplot to visualize the relationships between features\ng = sns.pairplot(iris_df, hue='target', palette='husl', markers=['o', 's', 'D'])\n\n# Add titles and labels\ng.fig.suptitle('Iris Dataset Pairplot', y=1.02) # y=1.02 to make space for the title\nplt.show()" }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n\\n```end\\n\\n## Current Task\\nPerform exploratory data analysis on the train dataset to understand the features and target variable.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nojb_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- The output code should contain all steps implemented in 'Code Steps'.\\n\"}]": { - "code": "# Step 1: Load the training dataset\nimport pandas as pd\ntrain_data = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\n\n# Step 2: Display the first few rows of the dataset\ndisplay(train_data.head())\n\n# Step 3: Get a summary of the dataset\ntrain_data.info()\n\n# Step 4: Describe the numerical features\ntrain_data.describe()\n\n# Step 5: Describe the categorical features\ntrain_data.describe(include=['O'])\n\n# Step 6: Check for missing values\nmissing_values = train_data.isnull().sum()\nmissing_values[missing_values > 0]\n\n# Step 7: Visualize the distribution of the target variable 'Survived'\nimport seaborn as sns\nimport matplotlib.pyplot as plt\nsns.countplot(x='Survived', data=train_data)\nplt.title('Distribution of Survival')\nplt.show()\n\n# Step 8: Explore the relationship between features and the target variable\n# Visualize the survival rate by sex\nsns.barplot(x='Sex', y='Survived', data=train_data)\nplt.title('Survival Rate by Sex')\nplt.show()\n\n# Visualize the survival rate by class\nsns.barplot(x='Pclass', y='Survived', data=train_data)\nplt.title('Survival Rate by Class')\nplt.show()\n\n# Visualize the survival rate by embarkation port\nsns.barplot(x='Embarked', y='Survived', data=train_data)\nplt.title('Survival Rate by Embarkation Port')\nplt.show()" + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n\\n```end\\n\\n## Current Task\\nPerform exploratory data analysis on the train dataset to understand the features and target variable.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", the code can be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nobj_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n\"}]": { + "code": "# Perform exploratory data analysis on the train dataset\ndf_train = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\n\n# Display the first few rows of the dataset\ndisplay(df_train.head())\n\n# Summary statistics for numerical features\ndisplay(df_train.describe())\n\n# Summary information about the dataset including the data types and number of non-null values\ndisplay(df_train.info())\n\n# Distribution of the target variable 'Survived'\nsurvival_counts = df_train['Survived'].value_counts()\nprint(\"Survival counts:\\n\", survival_counts)\n\n# Visualizations\nimport matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Distribution of the target variable\nsns.countplot(x='Survived', data=df_train)\nplt.title('Distribution of Survival')\nplt.show()\n\n# Correlation matrix heatmap to understand the relationship between features\nplt.figure(figsize=(10, 8))\nsns.heatmap(df_train.corr(), annot=True, fmt='.2f')\nplt.title('Correlation Matrix')\nplt.show()\n\n# Pairplot to visualize the pairwise relationships between features\nsns.pairplot(df_train, hue='Survived')\nplt.title('Pairplot of Features')\nplt.show()" }, "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nSolve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\n[user: Tested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { - "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not correctly sort the array based on the number of ones in their binary representation followed by their decimal value. The error in the output indicates that the numbers 3 and 4 are not in the expected positions. The binary representations of 3 and 4 are '11' and '100', respectively, which means 3 has more ones than 4 and should come before 4 in the sorted array. To fix this issue, we need to ensure that the sorting is stable so that when two numbers have the same number of ones, they retain their original order if their decimal values are the same. We can achieve this by using the 'stable' sorting algorithm provided by Python's sorted function. Additionally, we need to handle negative numbers correctly by taking the absolute value before counting the ones in their binary representation.", - "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(abs(n)).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n # Using 'stable' sorting algorithm to maintain the original order when necessary\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x), reverse=False)\n \n return sorted_arr" - }, - "[{\"role\": \"user\", \"content\": \"Interface definition:\\n```text\\nInterface Name: Element Tagging\\nInterface Path: /projects/{project_key}/node-tags\\nMethod: POST\\n\\nRequest parameters:\\nPath parameters:\\nproject_key\\n\\nBody parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nnodes\\tarray\\tYes\\t\\tNodes\\n\\tnode_key\\tstring\\tNo\\t\\tNode key\\n\\ttags\\tarray\\tNo\\t\\tOriginal node tag list\\n\\tnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\noperations\\tarray\\tYes\\t\\t\\n\\ttags\\tarray\\tNo\\t\\tOperation tag list\\n\\tmode\\tstring\\tNo\\t\\tOperation type ADD / DELETE\\n\\nReturn data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tinteger\\tYes\\t\\tStatus code\\nmsg\\tstring\\tYes\\t\\tPrompt message\\ndata\\tobject\\tYes\\t\\tReturned data\\nlist\\tarray\\tNo\\t\\tNode list true / false\\nnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\nnode_key\\tstring\\tNo\\t\\tNode key\\n```\\n\\nUnit test:\\n```python\\n@pytest.mark.parametrize(\\n\\\"project_key, nodes, operations, expected_msg\\\",\\n[\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"success\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_002\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"tag1\\\"], \\\"mode\\\": \\\"DELETE\\\"}], \\\"success\\\"),\\n(\\\"\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Missing the required parameter project_key\\\"),\\n(123, [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Incorrect parameter type\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"a\\\"*201, \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Request parameter exceeds field boundary\\\")\\n]\\n)\\ndef test_node_tags(project_key, nodes, operations, expected_msg):\\n pass\\n\\n# The above is an interface definition and a unit test example.\\n# Next, please play the role of an expert test manager with 20 years of experience at Google. When I give the interface definition, \\n# reply to me with a unit test. There are several requirements:\\n# 1. Only output one `@pytest.mark.parametrize` and the corresponding test_ function (inside pass, do not implement).\\n# -- The function parameter contains expected_msg for result verification.\\n# 2. The generated test cases use shorter text or numbers and are as compact as possible.\\n# 3. If comments are needed, use Chinese.\\n\\n# If you understand, please wait for me to give the interface definition and just answer \\\"Understood\\\" to save tokens.\\n\"}, {\"role\": \"user\", \"content\": \"Refer to the test types: such as SQL injection, cross-site scripting (XSS), unauthorized access and privilege escalation, \\nauthentication and authorization, parameter verification, exception handling, file upload and download.\\nPlease output 10 test cases within one `@pytest.mark.parametrize` scope.\\n```text\\nAPI Name: 获取 model 详情(job专用-后续开放给sdk)\\nAPI Path: /v1/projects/{project_key}/jobs/{job_id}/models/{model_key}\\nMethod: GET\\n\\nRequest Parameters:\\nPath Parameters:\\nproject_key \\njob_id \\nmodel_key \\n\\nBody Parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nproject_key\\tstring\\tYes\\t\\t\\njob_id\\tstring\\tYes\\t\\t\\nmodel_key\\tstring\\tYes\\t\\t\\n\\nResponse Data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tnumber\\tYes\\t\\t0成功,非0失败\\nmsg\\tstring\\tYes\\t\\t如果失败,这里有错误信息\\ndata\\tobject\\tYes\\t\\tdata信息\\n\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\tname\\tstring\\tNo\\t\\t用户可修改的name\\n\\tmodel\\tobject\\tNo\\t\\tmodel信息\\n\\t\\ttype\\tstring\\tNo\\t\\tdataset type\\n\\t\\tmanaged\\tboolean\\tNo\\t\\t为false时是第一类dataset,数据不可删除\\n\\t\\tname\\tstring\\tNo\\t\\t用户可修改的name\\n\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\tformat_type\\tstring\\tNo\\t\\t文件类型的dataset才有这项。“csv”\\n\\t\\tflow_options\\tobject\\tNo\\t\\t创建dataset时的高级设置\\n\\t\\t\\tvirtualizable\\tboolean\\tNo\\t\\t高级设置里的参数。缺省false\\n\\t\\t\\trebuild_behavior\\tstring\\tNo\\t\\t高级设置里的参数。缺省NORMAL\\n\\t\\t\\tcross_project_build_behavior\\tstring\\tNo\\t\\t高级设置里的参数。缺省DEFAULT\\n\\t\\tformat_params\\tobject\\tNo\\t\\t文件类型的dataset才有\\n\\t\\t\\tstyle\\tstring\\tNo\\t\\t\\n\\t\\t\\tcharset\\tstring\\tNo\\t\\t\\n\\t\\t\\tseparator\\tstring\\tNo\\t\\t\\n\\t\\t\\tquote_char\\tstring\\tNo\\t\\t\\n\\t\\t\\tescape_char\\tstring\\tNo\\t\\t\\n\\t\\t\\tdate_serialization_format\\tstring\\tNo\\t\\t\\n\\t\\t\\tarray_map_format\\tstring\\tNo\\t\\t\\n\\t\\t\\thive_separators\\tarray\\tNo\\t\\t\\n\\t\\t\\tskip_rows_before_header\\tnumber\\tNo\\t\\t\\n\\t\\t\\tparse_header_row\\tboolean\\tNo\\t\\t\\n\\t\\t\\tskip_rows_after_header\\tnumber\\tNo\\t\\t\\n\\t\\t\\tprobable_number_of_records\\tnumber\\tNo\\t\\t\\n\\t\\t\\tnormalize_booleans\\tboolean\\tNo\\t\\t\\n\\t\\t\\tnormalize_doubles\\tboolean\\tNo\\t\\t\\n\\t\\ttags\\tarray\\tNo\\t\\t标签tags\\n\\t\\tparams\\tobject\\tNo\\t\\t必有这项,但不同类型的dataset里面的key有差别\\n\\t\\t\\tconnection\\tstring\\tNo\\t\\tconnection id,到db查其他参数\\n\\t\\t\\tpath\\tstring\\tNo\\t\\t文件类connection才有这项\\n\\t\\t\\ttable\\tstring\\tNo\\t\\tdb表名,DB类connection才有这项\\n\\t\\t\\tmode\\tstring\\tNo\\t\\t存储类型,比如“table\\\",DB类connection才有这项\\n\\t\\t\\tbucket\\tstring\\tNo\\t\\tS3类型的connection才有这项\\n\\t\\t\\tkey_name\\tstring\\tNo\\t\\tredis才有,key name\\n\\t\\t\\tkey_type\\tstring\\tNo\\t\\tredis才有,key type\\n\\t\\t\\tcollection\\tstring\\tNo\\t\\t非关系型数据库才有,collection name\\n\\t\\t\\tindex\\tstring\\tNo\\t\\t索引类型的才有这项\\n\\t\\t\\tnot_ready_if_empty\\tboolean\\tNo\\t\\t数据非空才认为是data ready\\n\\t\\t\\tfiles_selection_rules\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tmode\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\texclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tinclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\texplicit_files\\tarray\\tNo\\t\\t\\n\\t\\tschema\\tobject\\tNo\\t\\tcolumns信息在这里\\n\\t\\t\\tcolumns\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tname\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\ttype\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\torigin_type\\tstring\\tNo\\t\\t\\n\\t\\t\\tuser_modified\\tboolean\\tNo\\t\\t\\n\\t\\tcustom_fields\\tobject\\tNo\\t\\t自定义fields\\n\\t\\tlast_build\\tobject\\tNo\\t\\t最后一次构建的信息\\n\\t\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\t\\tid\\tstring\\tNo\\t\\tactivity id\\n\\t\\t\\tjob_id\\tstring\\tNo\\t\\tjob id\\n\\t\\t\\tjob_project_key\\tstring\\tNo\\t\\t\\n\\t\\t\\tbuild_start_time\\tnumber\\tNo\\t\\t构建开始时间\\n\\t\\t\\tbuild_end_time\\tnumber\\tNo\\t\\t构建结束时间\\n\\t\\t\\tbuild_success\\tstring\\tNo\\t\\tsuccess或failed\\n\\t\\tobject_key\\tstring\\tNo\\t\\tdataset_key,后台用的id,用户不可见不可改\\n\\t\\tcache\\tobject\\tNo\\t\\t下载缓存数据链接\\n\\t\\t\\ts3_path\\tstring\\tNo\\t\\t\\n\\tstatus\\tobject\\tNo\\t\\t数据状态\\n\\t\\tsize\\tobject\\tNo\\t\\t数据大小信息\\n\\t\\t\\ttotal_value\\tnumber\\tNo\\t\\t占多少字节磁盘\\n\\t\\t\\tlast_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\tfirst_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\thas_data\\tboolean\\tNo\\t\\t是否有数据,这个影响前端的图标显示\\n\\t\\t\\tincomplete\\tboolean\\tNo\\t\\t\\n\\t\\trecords\\tobject\\tNo\\t\\t\\n\\t\\t\\ttotal_value\\tnumber\\tNo\\t\\t\\n\\t\\t\\tlast_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\tfirst_computed\\tnumber\\tNo\\t\\t\\n\\t\\t\\thas_data\\tboolean\\tNo\\t\\t是否有数据,这个影响前端的图标显示\\n\\t\\t\\tincomplete\\tboolean\\tNo\\t\\t\\n\\t\\tpartitions_last_compute\\tnumber\\tNo\\t\\t\\n\\t\\tpartitions\\tnumber\\tNo\\t\\t\\n\\tbuildable\\tboolean\\tNo\\t\\t有recipe时为true\\n\\theaders\\tarray\\tNo\\t\\t\\n\\t\\tdataset_schema\\tobject\\tNo\\t\\t\\n\\t\\t\\tname\\tstring\\tNo\\t字段名称\\t\\n\\t\\t\\ttype\\tstring\\tNo\\t字段类型\\t\\n\\t\\tnormal_rate\\tobject\\tNo\\t缺失值统计信息\\t\\n\\n```\"}]": { - "code": "import string\nimport random\n\ndef random_string(length=10):\n return ''.join(random.choice(string.ascii_lowercase) for i in range(length))" - }, - "[{\"role\": \"user\", \"content\": \"Interface definition:\\n```text\\nInterface Name: Element Tagging\\nInterface Path: /projects/{project_key}/node-tags\\nMethod: POST\\n\\nRequest parameters:\\nPath parameters:\\nproject_key\\n\\nBody parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nnodes\\tarray\\tYes\\t\\tNodes\\n\\tnode_key\\tstring\\tNo\\t\\tNode key\\n\\ttags\\tarray\\tNo\\t\\tOriginal node tag list\\n\\tnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\noperations\\tarray\\tYes\\t\\t\\n\\ttags\\tarray\\tNo\\t\\tOperation tag list\\n\\tmode\\tstring\\tNo\\t\\tOperation type ADD / DELETE\\n\\nReturn data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tinteger\\tYes\\t\\tStatus code\\nmsg\\tstring\\tYes\\t\\tPrompt message\\ndata\\tobject\\tYes\\t\\tReturned data\\nlist\\tarray\\tNo\\t\\tNode list true / false\\nnode_type\\tstring\\tNo\\t\\tNode type DATASET / RECIPE\\nnode_key\\tstring\\tNo\\t\\tNode key\\n```\\n\\nUnit test:\\n```python\\n@pytest.mark.parametrize(\\n\\\"project_key, nodes, operations, expected_msg\\\",\\n[\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"success\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"dataset_002\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"tag1\\\"], \\\"mode\\\": \\\"DELETE\\\"}], \\\"success\\\"),\\n(\\\"\\\", [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Missing the required parameter project_key\\\"),\\n(123, [{\\\"node_key\\\": \\\"dataset_001\\\", \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Incorrect parameter type\\\"),\\n(\\\"project_key\\\", [{\\\"node_key\\\": \\\"a\\\"*201, \\\"tags\\\": [\\\"tag1\\\", \\\"tag2\\\"], \\\"node_type\\\": \\\"DATASET\\\"}], [{\\\"tags\\\": [\\\"new_tag1\\\"], \\\"mode\\\": \\\"ADD\\\"}], \\\"Request parameter exceeds field boundary\\\")\\n]\\n)\\ndef test_node_tags(project_key, nodes, operations, expected_msg):\\n pass\\n\\n# The above is an interface definition and a unit test example.\\n# Next, please play the role of an expert test manager with 20 years of experience at Google. When I give the interface definition, \\n# reply to me with a unit test. There are several requirements:\\n# 1. Only output one `@pytest.mark.parametrize` and the corresponding test_ function (inside pass, do not implement).\\n# -- The function parameter contains expected_msg for result verification.\\n# 2. The generated test cases use shorter text or numbers and are as compact as possible.\\n# 3. If comments are needed, use Chinese.\\n\\n# If you understand, please wait for me to give the interface definition and just answer \\\"Understood\\\" to save tokens.\\n\"}, {\"role\": \"user\", \"content\": \"Refer to the test types: such as SQL injection, cross-site scripting (XSS), unauthorized access and privilege escalation, \\nauthentication and authorization, parameter verification, exception handling, file upload and download.\\nPlease output 10 test cases within one `@pytest.mark.parametrize` scope.\\n```text\\nAPI Name: 获取managed folder详情(job专用)\\nAPI Path: /v1/projects/{project_key}/jobs/{job_id}/folders/{folder_key}\\nMethod: GET\\n\\nRequest Parameters:\\nPath Parameters:\\nproject_key \\njob_id \\nfolder_key \\n\\nBody Parameters:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\nproject_key\\tstring\\tYes\\t\\t\\njob_id\\tstring\\tYes\\t\\t\\nfolder_key\\tstring\\tYes\\t\\t\\n\\nResponse Data:\\nName\\tType\\tRequired\\tDefault Value\\tRemarks\\ncode\\tnumber\\tYes\\t\\t0成功,非0失败\\nmsg\\tstring\\tYes\\t\\t失败时这里有错误信息\\ndata\\tobject\\tYes\\t\\t\\n\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\tfolder\\tobject\\tNo\\t\\tfolder配置在这里\\n\\t\\tproject_key\\tstring\\tNo\\t\\tproject key\\n\\t\\tobject_key\\tstring\\tNo\\t\\tobject key\\n\\t\\tname\\tstring\\tNo\\t\\t用户可编辑的那个name\\n\\t\\ttype\\tstring\\tNo\\t\\tfolder类型,与connection有关\\n\\t\\tparams\\tobject\\tNo\\t\\t数据读写相关配置在这里\\n\\t\\t\\tconnection\\tstring\\tNo\\t\\tconnection id\\n\\t\\t\\tpath\\tstring\\tNo\\t\\t文件夹内容存放的相对路径\\n\\t\\t\\tnot_ready_if_empty\\tboolean\\tNo\\t\\treserved\\n\\t\\t\\tfiles_selection_rules\\tobject\\tNo\\t\\t文件过滤规则\\n\\t\\t\\t\\tmode\\tstring\\tNo\\t\\tALL\\n\\t\\t\\t\\texclude_rules\\tarray\\tNo\\t\\t排除规则\\n\\t\\t\\t\\tinclude_rules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\texplicit_files\\tarray\\tNo\\t\\t\\n\\t\\tflow_options\\tobject\\tNo\\t\\tflow参数\\n\\t\\t\\tvirtualizable\\tboolean\\tNo\\t\\t\\n\\t\\t\\trebuild_behavior\\tstring\\tNo\\t\\t构建方式\\n\\t\\t\\tcross_project_build_behavior\\tstring\\tNo\\t\\t\\n\\t\\tmetrics\\tobject\\tNo\\t\\t\\n\\t\\t\\tprobes\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\ttype\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\tcompute_on_build_mode\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tmeta\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tname\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\tlevel\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\tconfiguration\\tobject\\tNo\\t\\t\\n\\t\\t\\tengine_config\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpad_runs_with_metrics\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\thive\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\textra_conf\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tbasic\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tdss\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\tselection\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tuse_mem_table\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tfilter\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tdistinct\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tpartition_selection_method\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tlatest_partitions_n\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tordering\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\tenabled\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\t\\trules\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tsampling_method\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tmax_records\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\ttarget_ratio\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\twithin_first_n\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\t\\t\\tmax_read_uncompressed_bytes\\tnumber\\tNo\\t\\t\\n\\t\\t\\t\\tsql\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\timpala\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\tspark\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\t\\tactive\\tboolean\\tNo\\t\\t\\n\\t\\t\\t\\t\\textra_conf\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tpython\\tobject\\tNo\\t\\t\\n\\t\\t\\tdisplayed_state\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpartition\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tcolumns\\tarray\\tNo\\t\\t\\n\\t\\t\\t\\tmetrics\\tarray\\tNo\\t\\t\\n\\t\\tchecks\\tobject\\tNo\\t\\t\\n\\t\\t\\trun_on_build\\tboolean\\tNo\\t\\t\\n\\t\\t\\tchecks\\tarray\\tNo\\t\\t\\n\\t\\t\\tdisplayed_state\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tpartition\\tstring\\tNo\\t\\t\\n\\t\\t\\t\\tchecks\\tarray\\tNo\\t\\t\\n\\t\\tversion_tag\\tobject\\tNo\\t\\t配置版本信息\\n\\t\\t\\tversion_number\\tnumber\\tNo\\t\\t\\n\\t\\t\\tlast_modified_by\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tlogin\\tstring\\tNo\\t\\t\\n\\t\\t\\tlast_modified_on\\tnumber\\tNo\\t\\t修改时间unix time ms\\n\\t\\tcreation_tag\\tobject\\tNo\\t\\t配置创建时间\\n\\t\\t\\tversion_number\\tnumber\\tNo\\t\\t1\\n\\t\\t\\tlast_modified_by\\tobject\\tNo\\t\\t\\n\\t\\t\\t\\tlogin\\tstring\\tNo\\t\\t\\n\\t\\t\\tlast_modified_on\\tnumber\\tNo\\t\\t创建时间unix time ms\\n\\t\\ttags\\tarray\\tNo\\t\\t文件夹标签\\n\\t\\tcustom_fields\\tobject\\tNo\\t\\t\\n\\t\\tchecklists\\tobject\\tNo\\t\\t\\n\\t\\t\\tchecklists\\tarray\\tNo\\t\\t\\n\\n```\"}]": { - "code": "import string\nimport random\n\ndef random_string(length=10):\n return ''.join(random.choice(string.ascii_lowercase) for i in range(length))" - }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Code Steps:\\nStrictly follow steps below when you writing code if it's convenient.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", and their are two steps in 'Code Steps', the code be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n- The output code should contain all steps implemented correctly in 'Code Steps'.\\n\"}]": { - "code": "# Data preprocessing\n\n# Step 1: Fill missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create a copy of the dataframe to avoid modifying the original one\ndf_processed = df.copy()\n\n# Identify numeric columns to fill missing values\nnumeric_cols = df_processed.select_dtypes(include='number').columns.tolist()\n\n# Initialize the FillMissingValue tool for numeric columns\nfill_missing_value = FillMissingValue(features=numeric_cols, strategy='mean')\n\n# Fit and transform the dataframe\ndf_processed = fill_missing_value.fit_transform(df_processed)\n\n# Step 2: Scale numeric columns\nfrom metagpt.tools.libs.data_preprocess import MinMaxScale\n\n# Initialize the MinMaxScale tool for numeric columns\nminmax_scale = MinMaxScale(features=numeric_cols)\n\n# Fit and transform the dataframe\ndf_processed = minmax_scale.fit_transform(df_processed)\n\n# Display the first few rows of the processed dataframe\ndf_processed.head()" - }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code_steps='', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code_steps='', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code_steps\\\":\\\"\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Instantiate the FillMissingValue class for column 'b'\nfill_missing_b = FillMissingValue(features=['b'], strategy='mean')\n\n# Fit and transform the DataFrame for column 'b'\ndf = fill_missing_b.fit_transform(df)" + "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not handle the sorting of elements with the same number of ones in their binary representation correctly. According to the context, for similar number of ones, the elements should be sorted based on their decimal value in ascending order. However, the output [1, 2, 4, 3, 5] shows that 3 and 4 are not in the correct order, indicating that the sorting by decimal value is not functioning as intended. To fix this issue, we should ensure that the sorting is stable, so that when two elements have the same number of ones, they maintain their relative order if their decimal values are already in ascending order.", + "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(n).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n # We use a stable sorting algorithm to maintain the relative order of elements with the same number of ones\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x), reverse=True)\n sorted_arr = sorted(sorted_arr, key=count_ones)\n \n return sorted_arr\n" } } \ No newline at end of file diff --git a/tests/metagpt/actions/ci/test_write_analysis_code.py b/tests/metagpt/actions/ci/test_write_analysis_code.py index 72071fa35..dbd83fe40 100644 --- a/tests/metagpt/actions/ci/test_write_analysis_code.py +++ b/tests/metagpt/actions/ci/test_write_analysis_code.py @@ -8,8 +8,8 @@ from metagpt.actions.ci.write_analysis_code import ( WriteCodeWithTools, ) from metagpt.logs import logger -from metagpt.plan.planner import STRUCTURAL_CONTEXT from metagpt.schema import Message, Plan, Task +from metagpt.strategy.planner import STRUCTURAL_CONTEXT @pytest.mark.skip @@ -37,13 +37,12 @@ async def test_write_code_by_list_plan(): @pytest.mark.asyncio async def test_tool_recommendation(): task = "clean and preprocess the data" - code_steps = "" available_tools = { "FillMissingValue": "Filling missing values", "SplitBins": "Bin continuous data into intervals and return the bin identifier encoded as an integer value", } write_code = WriteCodeWithTools() - tools = await write_code._recommend_tool(task, code_steps, available_tools) + tools = await write_code._recommend_tool(task, available_tools) assert len(tools) == 1 assert "FillMissingValue" in tools diff --git a/tests/metagpt/roles/test_code_interpreter.py b/tests/metagpt/roles/ci/test_code_interpreter.py similarity index 90% rename from tests/metagpt/roles/test_code_interpreter.py rename to tests/metagpt/roles/ci/test_code_interpreter.py index 2d71fcbb0..f23292965 100644 --- a/tests/metagpt/roles/test_code_interpreter.py +++ b/tests/metagpt/roles/ci/test_code_interpreter.py @@ -1,7 +1,7 @@ import pytest from metagpt.logs import logger -from metagpt.roles.code_interpreter import CodeInterpreter +from metagpt.roles.ci.code_interpreter import CodeInterpreter @pytest.mark.asyncio diff --git a/tests/metagpt/roles/test_ml_engineer.py b/tests/metagpt/roles/ci/test_ml_engineer.py similarity index 96% rename from tests/metagpt/roles/test_ml_engineer.py rename to tests/metagpt/roles/ci/test_ml_engineer.py index 2728c6411..144201f85 100644 --- a/tests/metagpt/roles/test_ml_engineer.py +++ b/tests/metagpt/roles/ci/test_ml_engineer.py @@ -2,7 +2,7 @@ import pytest from metagpt.actions.ci.execute_nb_code import ExecuteNbCode from metagpt.logs import logger -from metagpt.roles.ml_engineer import MLEngineer +from metagpt.roles.ci.ml_engineer import MLEngineer from metagpt.schema import Message, Plan, Task from metagpt.tools.tool_types import ToolTypes from tests.metagpt.actions.ci.test_debug_code import CODE, DebugContext, ErrorStr @@ -22,7 +22,6 @@ MockPlan = Plan( dependent_task_ids=[], instruction="Perform exploratory data analysis on the train dataset to understand the features and target variable.", task_type="eda", - code_steps="", code="", result="", is_success=False, @@ -35,7 +34,6 @@ MockPlan = Plan( dependent_task_ids=[], instruction="Perform exploratory data analysis on the train dataset to understand the features and target variable.", task_type="eda", - code_steps="", code="", result="", is_success=False, From 8c65ed02b879e15b4bfff4c69fdac05651040678 Mon Sep 17 00:00:00 2001 From: yzlin Date: Sun, 4 Feb 2024 23:27:04 +0800 Subject: [PATCH 604/637] rm redundant docstring --- metagpt/tools/libs/data_preprocess.py | 15 --------------- 1 file changed, 15 deletions(-) diff --git a/metagpt/tools/libs/data_preprocess.py b/metagpt/tools/libs/data_preprocess.py index 66f579f66..2cfa0b389 100644 --- a/metagpt/tools/libs/data_preprocess.py +++ b/metagpt/tools/libs/data_preprocess.py @@ -71,26 +71,11 @@ class DataPreprocessTool(MLProcess): self.model = None # to be filled by specific subclass Tool def fit(self, df: pd.DataFrame): - """ - Fit a model to be used in subsequent transform. - - Args: - df (pd.DataFrame): The input DataFrame. - """ if len(self.features) == 0: return self.model.fit(df[self.features]) def transform(self, df: pd.DataFrame) -> pd.DataFrame: - """ - Transform the input DataFrame with the fitted model. - - Args: - df (pd.DataFrame): The input DataFrame. - - Returns: - pd.DataFrame: The transformed DataFrame. - """ if len(self.features) == 0: return df new_df = df.copy() From a609946029050b2a6fa278f218d082da42c4b2c1 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 5 Feb 2024 00:05:16 +0800 Subject: [PATCH 605/637] mv tool_type def --- metagpt/tools/tool_types.py | 95 ++++++++++++++----------------------- 1 file changed, 35 insertions(+), 60 deletions(-) diff --git a/metagpt/tools/tool_types.py b/metagpt/tools/tool_types.py index 40981f836..d96c0299c 100644 --- a/metagpt/tools/tool_types.py +++ b/metagpt/tools/tool_types.py @@ -9,68 +9,43 @@ from metagpt.prompts.tool_types import ( ) from metagpt.tools.tool_data_type import ToolType -Eda = ToolType(name="eda", desc="For performing exploratory data analysis") - -DataPreprocess = ToolType( - name="data_preprocess", - desc="Only for changing value inplace.", - usage_prompt=DATA_PREPROCESS_PROMPT, -) - - -FeatureEngineering = ToolType( - name="feature_engineering", - desc="Only for creating new columns for input data.", - usage_prompt=FEATURE_ENGINEERING_PROMPT, -) - - -ModelTrain = ToolType( - name="model_train", - desc="Only for training model.", - usage_prompt=MODEL_TRAIN_PROMPT, -) - - -ModelEvaluate = ToolType( - name="model_evaluate", - desc="Only for evaluating model.", - usage_prompt=MODEL_EVALUATE_PROMPT, -) - - -StableDiffusion = ToolType( - name="stable_diffusion", - desc="Related to text2image, image2image using stable diffusion model.", -) - - -Image2Webpage = ToolType( - name="image2webpage", - desc="For converting image into webpage code.", - usage_prompt=IMAGE2WEBPAGE_PROMPT, -) - - -WebScraping = ToolType( - name="web_scraping", - desc="For scraping data from web pages.", -) - - -Other = ToolType(name="other", desc="Any tools not in the defined categories") - class ToolTypes(Enum): - EDA = Eda - DATA_PREPROCESS = DataPreprocess - FEATURE_ENGINEERING = FeatureEngineering - MODEL_TRAIN = ModelTrain - MODEL_EVALUATE = ModelEvaluate - STABLE_DIFFUSION = StableDiffusion - IMAGE2WEBPAGE = Image2Webpage - WEBSCRAPING = WebScraping - OTHER = Other + EDA = ToolType(name="eda", desc="For performing exploratory data analysis") + DATA_PREPROCESS = ToolType( + name="data_preprocess", + desc="Only for changing value inplace.", + usage_prompt=DATA_PREPROCESS_PROMPT, + ) + FEATURE_ENGINEERING = ToolType( + name="feature_engineering", + desc="Only for creating new columns for input data.", + usage_prompt=FEATURE_ENGINEERING_PROMPT, + ) + MODEL_TRAIN = ToolType( + name="model_train", + desc="Only for training model.", + usage_prompt=MODEL_TRAIN_PROMPT, + ) + MODEL_EVALUATE = ToolType( + name="model_evaluate", + desc="Only for evaluating model.", + usage_prompt=MODEL_EVALUATE_PROMPT, + ) + STABLE_DIFFUSION = ToolType( + name="stable_diffusion", + desc="Related to text2image, image2image using stable diffusion model.", + ) + IMAGE2WEBPAGE = ToolType( + name="image2webpage", + desc="For converting image into webpage code.", + usage_prompt=IMAGE2WEBPAGE_PROMPT, + ) + WEBSCRAPING = ToolType( + name="web_scraping", + desc="For scraping data from web pages.", + ) + OTHER = ToolType(name="other", desc="Any tools not in the defined categories") def __missing__(self, key): return self.OTHER From a35f5366b87ed51f58a1c2fa771a75d778948101 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 5 Feb 2024 00:20:23 +0800 Subject: [PATCH 606/637] raise error directly if invalid json --- metagpt/provider/openai_api.py | 9 +-------- 1 file changed, 1 insertion(+), 8 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 94aef70da..63e68c9bd 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -209,14 +209,7 @@ class OpenAILLM(BaseLLM): and message.tool_calls[0].function.arguments is not None ): # reponse is code - try: - return json.loads(message.tool_calls[0].function.arguments, strict=False) - except json.decoder.JSONDecodeError as e: - error_msg = ( - f"Got JSONDecodeError for \n{'--'*40} \n{message.tool_calls[0].function.arguments}, {str(e)}" - ) - logger.error(error_msg) - raise json.decoder.JSONDecodeError(error_msg, e.doc, e.pos) + return json.loads(message.tool_calls[0].function.arguments, strict=False) elif message.tool_calls is None and message.content is not None: # reponse is code, fix openai tools_call respond bug, # The response content is `code``, but it appears in the content instead of the arguments. From 20393e9d7ace385442075411ba86cb40d1dcc3c5 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 5 Feb 2024 11:38:07 +0800 Subject: [PATCH 607/637] rename tool type --- metagpt/roles/ci/ml_engineer.py | 8 +++---- metagpt/tools/libs/data_preprocess.py | 4 ++-- metagpt/tools/libs/feature_engineering.py | 4 ++-- metagpt/tools/libs/gpt_v_generator.py | 4 ++-- metagpt/tools/libs/sd_engine.py | 4 ++-- metagpt/tools/libs/web_scraping.py | 4 ++-- metagpt/tools/tool_data_type.py | 4 ++-- metagpt/tools/tool_registry.py | 12 +++++----- metagpt/tools/{tool_types.py => tool_type.py} | 22 +++++++++---------- tests/metagpt/roles/ci/test_ml_engineer.py | 4 ++-- tests/metagpt/tools/test_tool_registry.py | 4 ++-- 11 files changed, 37 insertions(+), 37 deletions(-) rename metagpt/tools/{tool_types.py => tool_type.py} (71%) diff --git a/metagpt/roles/ci/ml_engineer.py b/metagpt/roles/ci/ml_engineer.py index 6fa6fe7b2..f8bcb2c89 100644 --- a/metagpt/roles/ci/ml_engineer.py +++ b/metagpt/roles/ci/ml_engineer.py @@ -3,7 +3,7 @@ from metagpt.actions.ci.execute_nb_code import ExecuteNbCode from metagpt.actions.ci.ml_action import UpdateDataColumns, WriteCodeWithToolsML from metagpt.logs import logger from metagpt.roles.ci.code_interpreter import CodeInterpreter -from metagpt.tools.tool_types import ToolTypes +from metagpt.tools.tool_type import ToolType from metagpt.utils.common import any_to_str @@ -51,9 +51,9 @@ class MLEngineer(CodeInterpreter): async def _update_data_columns(self): current_task = self.planner.plan.current_task if current_task.task_type not in [ - ToolTypes.DATA_PREPROCESS.type_name, - ToolTypes.FEATURE_ENGINEERING.type_name, - ToolTypes.MODEL_TRAIN.type_name, + ToolType.DATA_PREPROCESS.type_name, + ToolType.FEATURE_ENGINEERING.type_name, + ToolType.MODEL_TRAIN.type_name, ]: return "" logger.info("Check columns in updated data") diff --git a/metagpt/tools/libs/data_preprocess.py b/metagpt/tools/libs/data_preprocess.py index 2cfa0b389..c9ca657a5 100644 --- a/metagpt/tools/libs/data_preprocess.py +++ b/metagpt/tools/libs/data_preprocess.py @@ -14,9 +14,9 @@ from sklearn.preprocessing import ( ) from metagpt.tools.tool_registry import register_tool -from metagpt.tools.tool_types import ToolTypes +from metagpt.tools.tool_type import ToolType -TOOL_TYPE = ToolTypes.DATA_PREPROCESS.type_name +TOOL_TYPE = ToolType.DATA_PREPROCESS.type_name class MLProcess: diff --git a/metagpt/tools/libs/feature_engineering.py b/metagpt/tools/libs/feature_engineering.py index bbd16b681..325742105 100644 --- a/metagpt/tools/libs/feature_engineering.py +++ b/metagpt/tools/libs/feature_engineering.py @@ -17,9 +17,9 @@ from sklearn.preprocessing import KBinsDiscretizer, PolynomialFeatures from metagpt.tools.libs.data_preprocess import MLProcess from metagpt.tools.tool_registry import register_tool -from metagpt.tools.tool_types import ToolTypes +from metagpt.tools.tool_type import ToolType -TOOL_TYPE = ToolTypes.FEATURE_ENGINEERING.type_name +TOOL_TYPE = ToolType.FEATURE_ENGINEERING.type_name @register_tool(tool_type=TOOL_TYPE) diff --git a/metagpt/tools/libs/gpt_v_generator.py b/metagpt/tools/libs/gpt_v_generator.py index 63fda3e81..6953300d8 100644 --- a/metagpt/tools/libs/gpt_v_generator.py +++ b/metagpt/tools/libs/gpt_v_generator.py @@ -13,7 +13,7 @@ import requests from metagpt.const import DEFAULT_WORKSPACE_ROOT from metagpt.tools.tool_registry import register_tool -from metagpt.tools.tool_types import ToolTypes +from metagpt.tools.tool_type import ToolType ANALYZE_LAYOUT_PROMPT = """You are now a UI/UX, please generate layout information for this image: @@ -31,7 +31,7 @@ Now, please generate the corresponding webpage code including HTML, CSS and Java @register_tool( - tool_type=ToolTypes.IMAGE2WEBPAGE.type_name, include_functions=["__init__", "generate_webpages", "save_webpages"] + tool_type=ToolType.IMAGE2WEBPAGE.type_name, include_functions=["__init__", "generate_webpages", "save_webpages"] ) class GPTvGenerator: """Class for generating webpages at once. diff --git a/metagpt/tools/libs/sd_engine.py b/metagpt/tools/libs/sd_engine.py index 6229a60e3..58f34a152 100644 --- a/metagpt/tools/libs/sd_engine.py +++ b/metagpt/tools/libs/sd_engine.py @@ -17,7 +17,7 @@ from PIL import Image, PngImagePlugin from metagpt.const import SD_OUTPUT_FILE_REPO, SOURCE_ROOT from metagpt.logs import logger from metagpt.tools.tool_registry import register_tool -from metagpt.tools.tool_types import ToolTypes +from metagpt.tools.tool_type import ToolType payload = { "prompt": "", @@ -54,7 +54,7 @@ default_negative_prompt = "(easynegative:0.8),black, dark,Low resolution" @register_tool( - tool_type=ToolTypes.STABLE_DIFFUSION.type_name, + tool_type=ToolType.STABLE_DIFFUSION.type_name, include_functions=["__init__", "simple_run_t2i", "run_t2i", "construct_payload", "save"], ) class SDEngine: diff --git a/metagpt/tools/libs/web_scraping.py b/metagpt/tools/libs/web_scraping.py index f983c1215..6fd3b9435 100644 --- a/metagpt/tools/libs/web_scraping.py +++ b/metagpt/tools/libs/web_scraping.py @@ -1,9 +1,9 @@ from metagpt.tools.tool_registry import register_tool -from metagpt.tools.tool_types import ToolTypes +from metagpt.tools.tool_type import ToolType from metagpt.tools.web_browser_engine_playwright import PlaywrightWrapper -@register_tool(tool_type=ToolTypes.WEBSCRAPING.type_name) +@register_tool(tool_type=ToolType.WEBSCRAPING.type_name) async def scrape_web_playwright(url, *urls): """ Scrape and save the HTML structure and inner text content of a web page using Playwright. diff --git a/metagpt/tools/tool_data_type.py b/metagpt/tools/tool_data_type.py index fe42b5721..0ae46fa5c 100644 --- a/metagpt/tools/tool_data_type.py +++ b/metagpt/tools/tool_data_type.py @@ -1,14 +1,14 @@ from pydantic import BaseModel -class ToolType(BaseModel): +class ToolTypeDef(BaseModel): name: str desc: str = "" usage_prompt: str = "" class ToolSchema(BaseModel): - name: str + description: str class Tool(BaseModel): diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index 299d62ca3..87645d35a 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -16,8 +16,8 @@ from pydantic import BaseModel, field_validator from metagpt.const import TOOL_SCHEMA_PATH from metagpt.logs import logger from metagpt.tools.tool_convert import convert_code_to_tool_schema -from metagpt.tools.tool_data_type import Tool, ToolSchema, ToolType -from metagpt.tools.tool_types import ToolTypes +from metagpt.tools.tool_data_type import Tool, ToolSchema, ToolTypeDef +from metagpt.tools.tool_type import ToolType class ToolRegistry(BaseModel): @@ -27,7 +27,7 @@ class ToolRegistry(BaseModel): @field_validator("tool_types", mode="before") @classmethod - def init_tool_types(cls, tool_types: ToolTypes): + def init_tool_types(cls, tool_types: ToolType): return {tool_type.type_name: tool_type.value for tool_type in tool_types} def register_tool( @@ -47,9 +47,9 @@ class ToolRegistry(BaseModel): if tool_type not in self.tool_types: # register new tool type on the fly logger.warning( - f"{tool_type} not previously defined, will create a temporary ToolType with just a name. This ToolType is only effective during this runtime. You may consider add this ToolType with more configs permanently at metagpt.tools.tool_types" + f"{tool_type} not previously defined, will create a temporary tool type with just a name. This tool type is only effective during this runtime. You may consider add this tool type with more configs permanently at metagpt.tools.tool_type" ) - temp_tool_type_obj = ToolType(name=tool_type) + temp_tool_type_obj = ToolTypeDef(name=tool_type) self.tool_types[tool_type] = temp_tool_type_obj if verbose: logger.info(f"tool type {tool_type} registered") @@ -97,7 +97,7 @@ class ToolRegistry(BaseModel): # Registry instance -TOOL_REGISTRY = ToolRegistry(tool_types=ToolTypes) +TOOL_REGISTRY = ToolRegistry(tool_types=ToolType) def register_tool(tool_type: str = "other", schema_path: str = "", **kwargs): diff --git a/metagpt/tools/tool_types.py b/metagpt/tools/tool_type.py similarity index 71% rename from metagpt/tools/tool_types.py rename to metagpt/tools/tool_type.py index d96c0299c..6fa971c56 100644 --- a/metagpt/tools/tool_types.py +++ b/metagpt/tools/tool_type.py @@ -7,45 +7,45 @@ from metagpt.prompts.tool_types import ( MODEL_EVALUATE_PROMPT, MODEL_TRAIN_PROMPT, ) -from metagpt.tools.tool_data_type import ToolType +from metagpt.tools.tool_data_type import ToolTypeDef -class ToolTypes(Enum): - EDA = ToolType(name="eda", desc="For performing exploratory data analysis") - DATA_PREPROCESS = ToolType( +class ToolType(Enum): + EDA = ToolTypeDef(name="eda", desc="For performing exploratory data analysis") + DATA_PREPROCESS = ToolTypeDef( name="data_preprocess", desc="Only for changing value inplace.", usage_prompt=DATA_PREPROCESS_PROMPT, ) - FEATURE_ENGINEERING = ToolType( + FEATURE_ENGINEERING = ToolTypeDef( name="feature_engineering", desc="Only for creating new columns for input data.", usage_prompt=FEATURE_ENGINEERING_PROMPT, ) - MODEL_TRAIN = ToolType( + MODEL_TRAIN = ToolTypeDef( name="model_train", desc="Only for training model.", usage_prompt=MODEL_TRAIN_PROMPT, ) - MODEL_EVALUATE = ToolType( + MODEL_EVALUATE = ToolTypeDef( name="model_evaluate", desc="Only for evaluating model.", usage_prompt=MODEL_EVALUATE_PROMPT, ) - STABLE_DIFFUSION = ToolType( + STABLE_DIFFUSION = ToolTypeDef( name="stable_diffusion", desc="Related to text2image, image2image using stable diffusion model.", ) - IMAGE2WEBPAGE = ToolType( + IMAGE2WEBPAGE = ToolTypeDef( name="image2webpage", desc="For converting image into webpage code.", usage_prompt=IMAGE2WEBPAGE_PROMPT, ) - WEBSCRAPING = ToolType( + WEBSCRAPING = ToolTypeDef( name="web_scraping", desc="For scraping data from web pages.", ) - OTHER = ToolType(name="other", desc="Any tools not in the defined categories") + OTHER = ToolTypeDef(name="other", desc="Any tools not in the defined categories") def __missing__(self, key): return self.OTHER diff --git a/tests/metagpt/roles/ci/test_ml_engineer.py b/tests/metagpt/roles/ci/test_ml_engineer.py index 144201f85..3bf9f3b92 100644 --- a/tests/metagpt/roles/ci/test_ml_engineer.py +++ b/tests/metagpt/roles/ci/test_ml_engineer.py @@ -4,7 +4,7 @@ from metagpt.actions.ci.execute_nb_code import ExecuteNbCode from metagpt.logs import logger from metagpt.roles.ci.ml_engineer import MLEngineer from metagpt.schema import Message, Plan, Task -from metagpt.tools.tool_types import ToolTypes +from metagpt.tools.tool_type import ToolType from tests.metagpt.actions.ci.test_debug_code import CODE, DebugContext, ErrorStr @@ -61,7 +61,7 @@ async def test_mle_update_data_columns(mocker): mle.planner.plan = MockPlan # manually update task type to test update - mle.planner.plan.current_task.task_type = ToolTypes.DATA_PREPROCESS.value + mle.planner.plan.current_task.task_type = ToolType.DATA_PREPROCESS.value result = await mle._update_data_columns() assert result is not None diff --git a/tests/metagpt/tools/test_tool_registry.py b/tests/metagpt/tools/test_tool_registry.py index e41ddfa79..2fd487fb7 100644 --- a/tests/metagpt/tools/test_tool_registry.py +++ b/tests/metagpt/tools/test_tool_registry.py @@ -1,7 +1,7 @@ import pytest from metagpt.tools.tool_registry import ToolRegistry -from metagpt.tools.tool_types import ToolTypes +from metagpt.tools.tool_type import ToolType @pytest.fixture @@ -11,7 +11,7 @@ def tool_registry(): @pytest.fixture def tool_registry_full(): - return ToolRegistry(tool_types=ToolTypes) + return ToolRegistry(tool_types=ToolType) # Test Initialization From 748aabce70ae7fac999426194a7af909b32eb9c8 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 5 Feb 2024 12:00:18 +0800 Subject: [PATCH 608/637] add future; rename writecodebygenerate tools --- metagpt/actions/__init__.py | 5 +++-- metagpt/actions/ci/ask_review.py | 2 ++ metagpt/actions/ci/debug_code.py | 6 +++--- metagpt/actions/ci/execute_nb_code.py | 8 +++++--- metagpt/actions/ci/ml_action.py | 8 +++++--- metagpt/actions/ci/write_analysis_code.py | 4 +++- metagpt/actions/ci/write_plan.py | 14 ++++++++------ metagpt/roles/ci/code_interpreter.py | 6 ++++-- metagpt/strategy/planner.py | 2 ++ metagpt/tools/libs/data_preprocess.py | 2 ++ metagpt/tools/libs/feature_engineering.py | 2 ++ metagpt/tools/libs/sd_engine.py | 7 ++++--- metagpt/tools/tool_registry.py | 2 ++ .../metagpt/actions/ci/test_write_analysis_code.py | 12 ++++++------ 14 files changed, 51 insertions(+), 29 deletions(-) diff --git a/metagpt/actions/__init__.py b/metagpt/actions/__init__.py index 6c0a2addc..363b4fd33 100644 --- a/metagpt/actions/__init__.py +++ b/metagpt/actions/__init__.py @@ -23,7 +23,7 @@ from metagpt.actions.write_prd import WritePRD from metagpt.actions.write_prd_review import WritePRDReview from metagpt.actions.write_test import WriteTest from metagpt.actions.ci.execute_nb_code import ExecuteNbCode -from metagpt.actions.ci.write_analysis_code import WriteCodeByGenerate +from metagpt.actions.ci.write_analysis_code import WriteCodeWithoutTools, WriteCodeWithTools from metagpt.actions.ci.write_plan import WritePlan @@ -46,7 +46,8 @@ class ActionType(Enum): WEB_BROWSE_AND_SUMMARIZE = WebBrowseAndSummarize CONDUCT_RESEARCH = ConductResearch EXECUTE_NB_CODE = ExecuteNbCode - WRITE_CODE_BY_GENERATE = WriteCodeByGenerate + WRITE_CODE_WITHOUT_TOOLS = WriteCodeWithoutTools + WRITE_CODE_WITH_TOOLS = WriteCodeWithTools WRITE_PLAN = WritePlan diff --git a/metagpt/actions/ci/ask_review.py b/metagpt/actions/ci/ask_review.py index 25b4314fe..041011e80 100644 --- a/metagpt/actions/ci/ask_review.py +++ b/metagpt/actions/ci/ask_review.py @@ -1,3 +1,5 @@ +from __future__ import annotations + from typing import Tuple from metagpt.actions import Action diff --git a/metagpt/actions/ci/debug_code.py b/metagpt/actions/ci/debug_code.py index f6b86b8bf..4a6617dc6 100644 --- a/metagpt/actions/ci/debug_code.py +++ b/metagpt/actions/ci/debug_code.py @@ -1,4 +1,4 @@ -from typing import List +from __future__ import annotations from metagpt.actions.ci.write_analysis_code import BaseWriteAnalysisCode from metagpt.logs import logger @@ -75,7 +75,7 @@ CODE_REFLECTION = { class DebugCode(BaseWriteAnalysisCode): async def run( self, - context: List[Message] = None, + context: list[Message] = None, code: str = "", runtime_result: str = "", ) -> str: @@ -83,7 +83,7 @@ class DebugCode(BaseWriteAnalysisCode): Execute the debugging process based on the provided context, code, and runtime_result. Args: - context (List[Message]): A list of Message objects representing the context. + context (list[Message]): A list of Message objects representing the context. code (str): The code to be debugged. runtime_result (str): The result of the code execution. diff --git a/metagpt/actions/ci/execute_nb_code.py b/metagpt/actions/ci/execute_nb_code.py index ee2faa0cb..300ee3807 100644 --- a/metagpt/actions/ci/execute_nb_code.py +++ b/metagpt/actions/ci/execute_nb_code.py @@ -2,13 +2,15 @@ """ @Date : 2023/11/17 14:22:15 @Author : orange-crow -@File : code_executor.py +@File : execute_nb_code.py """ +from __future__ import annotations + import asyncio import base64 import re import traceback -from typing import List, Literal, Tuple +from typing import Literal, Tuple import nbformat from nbclient import NotebookClient @@ -90,7 +92,7 @@ class ExecuteNbCode(Action): else: cell["outputs"].append(new_output(output_type="stream", name="stdout", text=str(output))) - def parse_outputs(self, outputs: List[str]) -> str: + def parse_outputs(self, outputs: list[str]) -> str: """Parses the outputs received from notebook execution.""" assert isinstance(outputs, list) parsed_output = "" diff --git a/metagpt/actions/ci/ml_action.py b/metagpt/actions/ci/ml_action.py index 9640a7918..60fe18c1b 100644 --- a/metagpt/actions/ci/ml_action.py +++ b/metagpt/actions/ci/ml_action.py @@ -1,4 +1,6 @@ -from typing import List, Tuple +from __future__ import annotations + +from typing import Tuple from metagpt.actions import Action from metagpt.actions.ci.write_analysis_code import WriteCodeWithTools @@ -16,11 +18,11 @@ from metagpt.utils.common import create_func_call_config, remove_comments class WriteCodeWithToolsML(WriteCodeWithTools): async def run( self, - context: List[Message], + context: list[Message], plan: Plan = None, column_info: str = "", **kwargs, - ) -> Tuple[List[Message], str]: + ) -> Tuple[list[Message], str]: # prepare tool schemas and tool-type-specific instruction tool_schemas, tool_type_usage_prompt = await self._prepare_tools(plan=plan) diff --git a/metagpt/actions/ci/write_analysis_code.py b/metagpt/actions/ci/write_analysis_code.py index 38fe107fd..72fe4e7a6 100644 --- a/metagpt/actions/ci/write_analysis_code.py +++ b/metagpt/actions/ci/write_analysis_code.py @@ -4,6 +4,8 @@ @Author : orange-crow @File : write_analysis_code.py """ +from __future__ import annotations + from typing import Tuple from metagpt.actions import Action @@ -42,7 +44,7 @@ class BaseWriteAnalysisCode(Action): raise NotImplementedError -class WriteCodeByGenerate(BaseWriteAnalysisCode): +class WriteCodeWithoutTools(BaseWriteAnalysisCode): """Ask LLM to generate codes purely by itself without local user-defined tools""" async def run(self, context: list[Message], plan: Plan = None, system_msg: str = None, **kwargs) -> dict: diff --git a/metagpt/actions/ci/write_plan.py b/metagpt/actions/ci/write_plan.py index 885611c68..e88f64724 100644 --- a/metagpt/actions/ci/write_plan.py +++ b/metagpt/actions/ci/write_plan.py @@ -4,9 +4,11 @@ @Author : orange-crow @File : plan.py """ +from __future__ import annotations + import json from copy import deepcopy -from typing import Dict, List, Tuple +from typing import Tuple from metagpt.actions import Action from metagpt.logs import logger @@ -40,14 +42,14 @@ class WritePlan(Action): ``` """ - async def assign_task_type(self, tasks: List[Dict]) -> str: + async def assign_task_type(self, tasks: list[dict]) -> str: """Assign task type to each task in tasks Args: - tasks (List[Dict]): tasks to be assigned task type + tasks (list[dict]): tasks to be assigned task type Returns: - List[Dict]: tasks with task type assigned + list[dict]: tasks with task type assigned """ task_list = "\n".join([f"Task {task['task_id']}: {task['instruction']}" for task in tasks]) task_type_desc = "\n".join( @@ -64,7 +66,7 @@ class WritePlan(Action): task["task_type"] = task_type return json.dumps(tasks) - async def run(self, context: List[Message], max_tasks: int = 5, use_tools: bool = False) -> str: + async def run(self, context: list[Message], max_tasks: int = 5, use_tools: bool = False) -> str: prompt = ( self.PROMPT_TEMPLATE.replace("__context__", "\n".join([str(ct) for ct in context])) # .replace("__current_plan__", current_plan) @@ -77,7 +79,7 @@ class WritePlan(Action): return rsp -def rsp_to_tasks(rsp: str) -> List[Task]: +def rsp_to_tasks(rsp: str) -> list[Task]: rsp = json.loads(rsp) tasks = [Task(**task_config) for task_config in rsp] return tasks diff --git a/metagpt/roles/ci/code_interpreter.py b/metagpt/roles/ci/code_interpreter.py index f8d00bb91..2572d09c5 100644 --- a/metagpt/roles/ci/code_interpreter.py +++ b/metagpt/roles/ci/code_interpreter.py @@ -1,9 +1,11 @@ +from __future__ import annotations + from pydantic import Field from metagpt.actions.ci.ask_review import ReviewConst from metagpt.actions.ci.execute_nb_code import ExecuteNbCode from metagpt.actions.ci.write_analysis_code import ( - WriteCodeByGenerate, + WriteCodeWithoutTools, WriteCodeWithTools, ) from metagpt.logs import logger @@ -80,7 +82,7 @@ class CodeInterpreter(Role): return py_code, result, success async def _write_code(self): - todo = WriteCodeByGenerate() if not self.use_tools else WriteCodeWithTools(selected_tools=self.tools) + todo = WriteCodeWithoutTools() if not self.use_tools else WriteCodeWithTools(selected_tools=self.tools) logger.info(f"ready to {todo.name}") context = self.planner.get_useful_memories() diff --git a/metagpt/strategy/planner.py b/metagpt/strategy/planner.py index bcb0bda9b..fd635df39 100644 --- a/metagpt/strategy/planner.py +++ b/metagpt/strategy/planner.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import json from pydantic import BaseModel, Field diff --git a/metagpt/tools/libs/data_preprocess.py b/metagpt/tools/libs/data_preprocess.py index c9ca657a5..7a3d019bf 100644 --- a/metagpt/tools/libs/data_preprocess.py +++ b/metagpt/tools/libs/data_preprocess.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import json import numpy as np diff --git a/metagpt/tools/libs/feature_engineering.py b/metagpt/tools/libs/feature_engineering.py index 325742105..40bfb2fc7 100644 --- a/metagpt/tools/libs/feature_engineering.py +++ b/metagpt/tools/libs/feature_engineering.py @@ -4,6 +4,8 @@ # @Author : lidanyang # @File : feature_engineering.py # @Desc : Feature Engineering Tools +from __future__ import annotations + import itertools # import lightgbm as lgb diff --git a/metagpt/tools/libs/sd_engine.py b/metagpt/tools/libs/sd_engine.py index 58f34a152..347f4a430 100644 --- a/metagpt/tools/libs/sd_engine.py +++ b/metagpt/tools/libs/sd_engine.py @@ -2,12 +2,13 @@ # @Date : 2023/7/19 16:28 # @Author : stellahong (stellahong@deepwisdom.ai) # @Desc : +from __future__ import annotations + import base64 import hashlib import io import json from os.path import join -from typing import List import requests from aiohttp import ClientSession @@ -135,11 +136,11 @@ class SDEngine: self.save(results, save_name=f"output_{save_name}") return results - async def run_t2i(self, payloads: List): + async def run_t2i(self, payloads: list): """Run the stable diffusion API for multiple prompts asynchronously. Args: - payloads (list): List of payload, each payload is a dictionary of input parameters for the stable diffusion API. + payloads (list): list of payload, each payload is a dictionary of input parameters for the stable diffusion API. """ session = ClientSession() for payload_idx, payload in enumerate(payloads): diff --git a/metagpt/tools/tool_registry.py b/metagpt/tools/tool_registry.py index 87645d35a..5fbd39421 100644 --- a/metagpt/tools/tool_registry.py +++ b/metagpt/tools/tool_registry.py @@ -5,6 +5,8 @@ @Author : garylin2099 @File : tool_registry.py """ +from __future__ import annotations + import inspect import os import re diff --git a/tests/metagpt/actions/ci/test_write_analysis_code.py b/tests/metagpt/actions/ci/test_write_analysis_code.py index dbd83fe40..95c7dfca8 100644 --- a/tests/metagpt/actions/ci/test_write_analysis_code.py +++ b/tests/metagpt/actions/ci/test_write_analysis_code.py @@ -4,7 +4,7 @@ import pytest from metagpt.actions.ci.execute_nb_code import ExecuteNbCode from metagpt.actions.ci.write_analysis_code import ( - WriteCodeByGenerate, + WriteCodeWithoutTools, WriteCodeWithTools, ) from metagpt.logs import logger @@ -15,7 +15,7 @@ from metagpt.strategy.planner import STRUCTURAL_CONTEXT @pytest.mark.skip @pytest.mark.asyncio async def test_write_code_by_list_plan(): - write_code = WriteCodeByGenerate() + write_code = WriteCodeWithoutTools() execute_code = ExecuteNbCode() messages = [] plan = ["随机生成一个pandas DataFrame时间序列", "绘制这个时间序列的直方图", "回顾已完成的任务", "求均值", "总结"] @@ -144,7 +144,7 @@ async def test_write_code_to_correct_error(): Message(content=wrong_code, role="assistant"), Message(content=error, role="user"), ] - new_code = await WriteCodeByGenerate().run(context=context) + new_code = await WriteCodeWithoutTools().run(context=context) new_code = new_code["code"] print(new_code) assert "read_csv" in new_code # should correct read_excel to read_csv @@ -184,7 +184,7 @@ async def test_write_code_reuse_code_simple(): context = [ Message(content=structural_context, role="user"), ] - code = await WriteCodeByGenerate().run(context=context) + code = await WriteCodeWithoutTools().run(context=context) code = code["code"] print(code) assert "pandas" not in code and "read_csv" not in code # should reuse import and read statement from previous one @@ -239,7 +239,7 @@ async def test_write_code_reuse_code_long(): Message(content=structural_context, role="user"), ] trials_num = 5 - trials = [WriteCodeByGenerate().run(context=context, temperature=0.0) for _ in range(trials_num)] + trials = [WriteCodeWithoutTools().run(context=context, temperature=0.0) for _ in range(trials_num)] trial_results = await asyncio.gather(*trials) print(*trial_results, sep="\n\n***\n\n") success = [ @@ -313,7 +313,7 @@ async def test_write_code_reuse_code_long_for_wine(): Message(content=structural_context, role="user"), ] trials_num = 5 - trials = [WriteCodeByGenerate().run(context=context, temperature=0.0) for _ in range(trials_num)] + trials = [WriteCodeWithoutTools().run(context=context, temperature=0.0) for _ in range(trials_num)] trial_results = await asyncio.gather(*trials) print(*trial_results, sep="\n\n***\n\n") success = [ From 9343a6bd2cf998877ccde4b0b9942474e05526d6 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 5 Feb 2024 15:40:41 +0800 Subject: [PATCH 609/637] mv pip success logic, rm redundant prompt --- metagpt/actions/ci/execute_nb_code.py | 8 ++++- metagpt/actions/ci/ml_action.py | 4 +-- metagpt/prompts/ci/ml_action.py | 50 ++++----------------------- metagpt/roles/ci/code_interpreter.py | 5 +-- 4 files changed, 17 insertions(+), 50 deletions(-) diff --git a/metagpt/actions/ci/execute_nb_code.py b/metagpt/actions/ci/execute_nb_code.py index 300ee3807..6a8c32b7f 100644 --- a/metagpt/actions/ci/execute_nb_code.py +++ b/metagpt/actions/ci/execute_nb_code.py @@ -181,7 +181,13 @@ class ExecuteNbCode(Action): # code success outputs = self.parse_outputs(self.nb.cells[-1].outputs) - return truncate(remove_escape_and_color_codes(outputs), is_success=success) + outputs, success = truncate(remove_escape_and_color_codes(outputs), is_success=success) + + if "!pip" in outputs: + success = False + + return outputs, success + elif language == "markdown": # add markdown content to markdown cell in a notebook. self.add_markdown_cell(code) diff --git a/metagpt/actions/ci/ml_action.py b/metagpt/actions/ci/ml_action.py index 60fe18c1b..e18d0fd20 100644 --- a/metagpt/actions/ci/ml_action.py +++ b/metagpt/actions/ci/ml_action.py @@ -5,7 +5,7 @@ from typing import Tuple from metagpt.actions import Action from metagpt.actions.ci.write_analysis_code import WriteCodeWithTools from metagpt.prompts.ci.ml_action import ( - GENERATE_CODE_PROMPT, + ML_GENERATE_CODE_PROMPT, ML_TOOL_USAGE_PROMPT, PRINT_DATA_COLUMNS, UPDATE_DATA_COLUMNS, @@ -43,7 +43,7 @@ class WriteCodeWithToolsML(WriteCodeWithTools): ) else: - prompt = GENERATE_CODE_PROMPT.format( + prompt = ML_GENERATE_CODE_PROMPT.format( user_requirement=plan.goal, history_code=code_context, current_task=plan.current_task.instruction, diff --git a/metagpt/prompts/ci/ml_action.py b/metagpt/prompts/ci/ml_action.py index 46d419dfb..5d27c7ff0 100644 --- a/metagpt/prompts/ci/ml_action.py +++ b/metagpt/prompts/ci/ml_action.py @@ -27,28 +27,6 @@ print(column_info) - Import `get_column_info` only if it's not already imported. """ -GEN_DATA_DESC_PROMPT = """ -Here is the head 5 rows of the dataset: -{data_head} - -Please provide a brief one-sentence background of the dataset, and concise meaning for each column. Keep descriptions short. - -Output the information in a JSON format, as shown in this example: -```json -{ - "data_desc": "Brief dataset background.", - "column_desc": { - "column_name1": "Abstract meaning of the first column.", - "column_name2": "Abstract meaning of the second column.", - ... - } -} -``` - -# Constraints: -- Don't contain specific values or examples found in the data column. -""" - PRINT_DATA_COLUMNS = { "name": "print_column_info", "description": "Print the latest column information after 'Done Tasks' code if first read or data changed.", @@ -64,7 +42,7 @@ PRINT_DATA_COLUMNS = { }, } -GENERATE_CODE_PROMPT = """ +ML_COMMON_PROMPT = """ # Background As a data scientist, you need to help user to achieve their goal [{user_requirement}] step-by-step in an continuous Jupyter notebook. @@ -83,7 +61,9 @@ Latest data info after previous tasks: # Task Write complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc. Specifically, {tool_type_usage_prompt} +""" +USE_NO_TOOLS_EXAMPLE = """ # Output Example: when current task is "train a lightgbm model on training data", the code can be like: ```python @@ -105,26 +85,7 @@ model.fit(train, y_train) - Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed. """ -ML_TOOL_USAGE_PROMPT = """ -# Background -As a data scientist, you need to help user to achieve their goal [{user_requirement}] step-by-step in an continuous Jupyter notebook. - -## Done Tasks -```python -{history_code} -```end - -## Current Task -{current_task} - -# Latest Data Info -Latest data info after previous tasks: -{column_info} - -# Task -Write complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc. -Specifically, {tool_type_usage_prompt} - +USE_TOOLS_EXAMPLE = """ # Capabilities - You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class. - You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc.. @@ -162,3 +123,6 @@ for col in num_cols: - Always prioritize using pre-defined tools for the same functionality. - Always copy the DataFrame before processing it and use the copy to process. """ + +ML_GENERATE_CODE_PROMPT = ML_COMMON_PROMPT + USE_NO_TOOLS_EXAMPLE +ML_TOOL_USAGE_PROMPT = ML_COMMON_PROMPT + USE_TOOLS_EXAMPLE diff --git a/metagpt/roles/ci/code_interpreter.py b/metagpt/roles/ci/code_interpreter.py index 2572d09c5..796abba04 100644 --- a/metagpt/roles/ci/code_interpreter.py +++ b/metagpt/roles/ci/code_interpreter.py @@ -64,9 +64,6 @@ class CodeInterpreter(Role): self.working_memory.add(Message(content=result, role="user", cause_by=ExecuteNbCode)) ### process execution result ### - if "!pip" in code["code"]: - success = False - counter += 1 if not success and counter >= max_retry: @@ -76,7 +73,7 @@ class CodeInterpreter(Role): counter = 0 # redo the task again with help of human suggestions py_code = ( - code["code"] if code.get("language") != "markdown" else "" + code["code"] if code.get("language") == "python" else "" ) # use python code as final code; for markdown, return the rendered result instead of the code itself return py_code, result, success From 402704379ccfb5bc2e92c15630df5a6dfebbb7a2 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 5 Feb 2024 16:32:41 +0800 Subject: [PATCH 610/637] improve details --- metagpt/actions/ci/execute_nb_code.py | 16 ++++------------ metagpt/actions/ci/write_analysis_code.py | 11 ++++------- metagpt/actions/ci/write_plan.py | 8 ++++---- metagpt/prompts/ci/write_analysis_code.py | 2 +- tests/data/rsp_cache.json | 6 ++++++ tests/metagpt/actions/ci/test_execute_nb_code.py | 5 ----- 6 files changed, 19 insertions(+), 29 deletions(-) diff --git a/metagpt/actions/ci/execute_nb_code.py b/metagpt/actions/ci/execute_nb_code.py index 6a8c32b7f..0ff00de8f 100644 --- a/metagpt/actions/ci/execute_nb_code.py +++ b/metagpt/actions/ci/execute_nb_code.py @@ -39,10 +39,9 @@ class ExecuteNbCode(Action): def __init__( self, - nb=None, + nb=nbformat.v4.new_notebook(), timeout=600, ): - nb = nb or nbformat.v4.new_notebook() super().__init__( nb=nb, nb_client=NotebookClient(nb, timeout=timeout), @@ -199,17 +198,10 @@ class ExecuteNbCode(Action): def truncate(result: str, keep_len: int = 2000, is_success: bool = True): """对于超出keep_len个字符的result: 执行失败的代码, 展示result后keep_len个字符; 执行成功的代码, 展示result前keep_len个字符。""" - desc = f"Executed code {'successfully. ' if is_success else 'failed, please reflect the cause of bug and then debug. '}" - is_same_desc = False - if is_success: - desc += f"Truncated to show only first {keep_len} characters\n" + desc = f"Executed code successfully. Truncated to show only first {keep_len} characters\n" else: - desc += f"Truncated to show only last {keep_len} characters\n" - - if result.startswith(desc): - result = result[len(desc) :] - is_same_desc = True + desc = f"Executed code failed, please reflect the cause of bug and then debug. Truncated to show only last {keep_len} characters\n" if result.strip().startswith(" dict: @@ -71,18 +71,15 @@ class WriteCodeWithTools(BaseWriteAnalysisCode): """ candidate_tools = TOOL_REGISTRY.get_tools_by_type(tool_type) if self.selected_tools: - candidate_tools = { - tool_name: candidate_tools[tool_name] - for tool_name in self.selected_tools - if tool_name in candidate_tools - } + candidate_tool_names = set(self.selected_tools) & candidate_tools.keys() + candidate_tools = {tool_name: candidate_tools[tool_name] for tool_name in candidate_tool_names} return candidate_tools async def _recommend_tool( self, task: str, available_tools: dict, - ) -> list: + ) -> dict: """ Recommend tools for the specified task. diff --git a/metagpt/actions/ci/write_plan.py b/metagpt/actions/ci/write_plan.py index e88f64724..dd9363260 100644 --- a/metagpt/actions/ci/write_plan.py +++ b/metagpt/actions/ci/write_plan.py @@ -49,19 +49,19 @@ class WritePlan(Action): tasks (list[dict]): tasks to be assigned task type Returns: - list[dict]: tasks with task type assigned + str: tasks with task type assigned in a json string """ - task_list = "\n".join([f"Task {task['task_id']}: {task['instruction']}" for task in tasks]) + task_info = "\n".join([f"Task {task['task_id']}: {task['instruction']}" for task in tasks]) task_type_desc = "\n".join( [f"- **{tool_type.name}**: {tool_type.desc}" for tool_type in TOOL_REGISTRY.get_tool_types().values()] ) # task type are binded with tool type now, should be improved in the future prompt = ASSIGN_TASK_TYPE_PROMPT.format( - task_list=task_list, task_type_desc=task_type_desc + task_info=task_info, task_type_desc=task_type_desc ) # task types are set to be the same as tool types, for now tool_config = create_func_call_config(ASSIGN_TASK_TYPE_CONFIG) rsp = await self.llm.aask_code(prompt, **tool_config) task_type_list = rsp["task_type"] - print(f"assigned task types: {task_type_list}") + logger.info(f"assigned task types: {task_type_list}") for task, task_type in zip(tasks, task_type_list): task["task_type"] = task_type return json.dumps(tasks) diff --git a/metagpt/prompts/ci/write_analysis_code.py b/metagpt/prompts/ci/write_analysis_code.py index 15d8b1443..4eccefcd1 100644 --- a/metagpt/prompts/ci/write_analysis_code.py +++ b/metagpt/prompts/ci/write_analysis_code.py @@ -1,6 +1,6 @@ ASSIGN_TASK_TYPE_PROMPT = """ Please assign a task type to each task in the list below from the given categories: -{task_list} +{task_info} ## All Task Type: {task_type_desc} diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index c5f2e9643..d6cbe60e7 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -388,5 +388,11 @@ "[{\"role\": \"system\", \"content\": \"You are an AI Python assistant. You will be given your previous implementation code of a task, runtime error results, and a hint to change the implementation appropriately. Write your full implementation \"}, {\"role\": \"user\", \"content\": \"\\nHere is an example for you.\\n\\nExample 1:\\n[previous impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a - b\\n```\\n\\n[runtime Error]:\\nTested passed:\\n\\nTests failed:\\nassert add(1, 2) == 3 # output: -1\\nassert add(1, 2) == 4 # output: -1\\n\\n[reflection on previous impl]:\\nThe implementation failed the test cases where the input integers are 1 and 2. The issue arises because the code does not add the two integers together, but instead subtracts the second integer from the first. To fix this issue, we should change the operator from `-` to `+` in the return statement. This will ensure that the function returns the correct output for the given input.\\n\\n[improved impl]:\\n```python\\ndef add(a: int, b: int) -> int:\\n \\\"\\\"\\\"\\n Given integers a and b, return the total value of a and b.\\n \\\"\\\"\\\"\\n return a + b\\n```\\n\\n[context]\\nSolve the problem in Python:\\ndef sort_array(arr):\\n \\\"\\\"\\\"\\n In this Kata, you have to sort an array of non-negative integers according to\\n number of ones in their binary representation in ascending order.\\n For similar number of ones, sort based on decimal value.\\n\\n It must be implemented like this:\\n >>> sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5]\\n >>> sort_array([-2, -3, -4, -5, -6]) == [-6, -5, -4, -3, -2]\\n >>> sort_array([1, 0, 2, 3, 4]) [0, 1, 2, 3, 4]\\n \\\"\\\"\\\"\\n\\n\\n[previous impl]\\n\\ndef sort_array(arr):\\n # Helper function to count the number of ones in the binary representation\\n def count_ones(n):\\n return bin(n).count('1')\\n \\n # Sort the array using a custom key function\\n # The key function returns a tuple (number of ones, value) for each element\\n # This ensures that if two elements have the same number of ones, they are sorted by their value\\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x))\\n \\n return sorted_arr\\n```\\n\\n[runtime Error]\\n[user: Tested passed:\\n\\nTests failed:\\nassert sort_array([1, 5, 2, 3, 4]) == [1, 2, 3, 4, 5] # output: [1, 2, 4, 3, 5]\\n]\\n\\nAnalysis the error step by step, provide me improve method and code. Remember to follow [context] requirement. Don't forget write code for steps behind the error step.\\n[reflection on previous impl]:\\nxxx\\n\"}]": { "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not handle the sorting of elements with the same number of ones in their binary representation correctly. According to the context, for similar number of ones, the elements should be sorted based on their decimal value in ascending order. However, the output [1, 2, 4, 3, 5] shows that 3 and 4 are not in the correct order, indicating that the sorting by decimal value is not functioning as intended. To fix this issue, we should ensure that the sorting is stable, so that when two elements have the same number of ones, they maintain their relative order if their decimal values are already in ascending order.", "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(n).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n # We use a stable sorting algorithm to maintain the relative order of elements with the same number of ones\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x), reverse=True)\n sorted_arr = sorted(sorted_arr, key=count_ones)\n \n return sorted_arr\n" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from sklearn import datasets\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\ndf_iris = pd.DataFrame(data=iris.data, columns=iris.feature_names)\ndf_iris['target'] = iris.target\n\n# Display basic information about the dataset\nprint(df_iris.info())\n\n# Display statistical summary of the dataset\nprint(df_iris.describe())\n\n# Display the first few rows of the dataset\nprint(df_iris.head())\n\n# Display the distribution of the target variable\ntarget_counts = df_iris['target'].value_counts()\nprint(target_counts)" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from sklearn import datasets\nimport matplotlib.pyplot as plt\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\niris_df = pd.DataFrame(data=iris.data, columns=iris.feature_names)\niris_df['target'] = iris.target\niris_df['target_name'] = iris_df['target'].apply(lambda x: iris.target_names[x])\n\n# Plotting\nfig, ax = plt.subplots(figsize=(12, 8))\nfor target, target_name in zip(iris.target_names, iris.target_names):\n subset = iris_df[iris_df['target_name'] == target_name]\n ax.scatter(subset[iris.feature_names[0]], subset[iris.feature_names[1]], label=target_name)\n\nax.set_xlabel(iris.feature_names[0])\nax.set_ylabel(iris.feature_names[1])\nax.legend()\nplt.show()" } } \ No newline at end of file diff --git a/tests/metagpt/actions/ci/test_execute_nb_code.py b/tests/metagpt/actions/ci/test_execute_nb_code.py index 6402cb883..72a85dd08 100644 --- a/tests/metagpt/actions/ci/test_execute_nb_code.py +++ b/tests/metagpt/actions/ci/test_execute_nb_code.py @@ -67,11 +67,6 @@ def test_truncate(): output, is_success = truncate(" Date: Mon, 5 Feb 2024 21:48:59 +0800 Subject: [PATCH 611/637] remove get_result method and improve gpt_v_generator.py and test_gpt_v_generator.py. --- metagpt/tools/libs/gpt_v_generator.py | 93 ++++--------------- metagpt/tools/tool_convert.py | 3 +- .../tools/libs/test_gpt_v_generator.py | 77 ++++++++++++--- 3 files changed, 88 insertions(+), 85 deletions(-) diff --git a/metagpt/tools/libs/gpt_v_generator.py b/metagpt/tools/libs/gpt_v_generator.py index 6953300d8..b1e8317ed 100644 --- a/metagpt/tools/libs/gpt_v_generator.py +++ b/metagpt/tools/libs/gpt_v_generator.py @@ -5,15 +5,13 @@ @Author : mannaandpoem @File : gpt_v_generator.py """ -import base64 import os from pathlib import Path -import requests - from metagpt.const import DEFAULT_WORKSPACE_ROOT from metagpt.tools.tool_registry import register_tool from metagpt.tools.tool_type import ToolType +from metagpt.utils.common import encode_image ANALYZE_LAYOUT_PROMPT = """You are now a UI/UX, please generate layout information for this image: @@ -43,27 +41,26 @@ class GPTvGenerator: def __init__(self): """Initialize GPTvGenerator class with default values from the configuration.""" from metagpt.config2 import config + from metagpt.llm import LLM - self.api_key = config.llm.api_key - self.api_base = config.llm.base_url - self.model = config.openai_vision_model - self.max_tokens = config.vision_max_tokens + self.llm = LLM(llm_config=config.get_openai_llm()) + self.llm.model = "gpt-4-vision-preview" - def analyze_layout(self, image_path): - """Analyze the layout of the given image and return the result. + async def analyze_layout(self, image_path: Path) -> str: + """Asynchronously analyze the layout of the given image and return the result. This is a helper method to generate a layout description based on the image. Args: - image_path (str): Path of the image to analyze. + image_path (Path): Path of the image to analyze. Returns: str: The layout analysis result. """ - return self.get_result(image_path, ANALYZE_LAYOUT_PROMPT) + return await self.llm.aask(msg=ANALYZE_LAYOUT_PROMPT, images=[encode_image(image_path)]) - def generate_webpages(self, image_path): - """Generate webpages including all code (HTML, CSS, and JavaScript) in one go based on the image. + async def generate_webpages(self, image_path: str) -> str: + """Asynchronously generate webpages including all code (HTML, CSS, and JavaScript) in one go based on the image. Args: image_path (str): The path of the image file. @@ -71,58 +68,14 @@ class GPTvGenerator: Returns: str: Generated webpages content. """ - layout = self.analyze_layout(image_path) + if isinstance(image_path, str): + image_path = Path(image_path) + layout = await self.analyze_layout(image_path) prompt = GENERATE_PROMPT + "\n\n # Context\n The layout information of the sketch image is: \n" + layout - result = self.get_result(image_path, prompt) - return result - - def get_result(self, image_path, prompt): - """Get the result from the vision model based on the given image path and prompt. - - Args: - image_path (str): Path of the image to analyze. - prompt (str): Prompt to use for the analysis. - - Returns: - str: The model's response as a string. - """ - base64_image = self.encode_image(image_path) - headers = {"Content-Type": "application/json", "Authorization": f"Bearer {self.api_key}"} - payload = { - "model": self.model, - "messages": [ - { - "role": "user", - "content": [ - {"type": "text", "text": prompt}, - {"type": "image_url", "image_url": {"url": f"data:image/jpeg;base64,{base64_image}"}}, - ], - } - ], - "max_tokens": self.max_tokens, - } - response = requests.post(f"{self.api_base}/chat/completions", headers=headers, json=payload) - - if response.status_code != 200: - raise ValueError(f"Request failed with status {response.status_code}, {response.text}") - else: - return response.json()["choices"][0]["message"]["content"] + return await self.llm.aask(msg=prompt, images=[encode_image(image_path)]) @staticmethod - def encode_image(image_path): - """Encode the image at the given path to a base64 string. - - Args: - image_path (str): Path of the image to encode. - - Returns: - str: The base64 encoded string of the image. - """ - with open(image_path, "rb") as image_file: - return base64.b64encode(image_file.read()).decode("utf-8") - - @staticmethod - def save_webpages(image_path, webpages) -> Path: + def save_webpages(image_path: str, webpages: str) -> Path: """Save webpages including all code (HTML, CSS, and JavaScript) at once. Args: @@ -132,35 +85,29 @@ class GPTvGenerator: Returns: Path: The path of the saved webpages. """ - # 在workspace目录下,创建一个名为下webpages的文件夹,用于存储html、css和js文件 + # Create a folder called webpages in the workspace directory to store HTML, CSS, and JavaScript files webpages_path = DEFAULT_WORKSPACE_ROOT / "webpages" / Path(image_path).stem os.makedirs(webpages_path, exist_ok=True) index_path = webpages_path / "index.html" - try: index = webpages.split("```html")[1].split("```")[0] - except IndexError: - index = "No html code found in the result, please check your image and try again." + "\n" + webpages - - try: + style_path = None if "styles.css" in index: style_path = webpages_path / "styles.css" elif "style.css" in index: style_path = webpages_path / "style.css" - else: - style_path = None style = webpages.split("```css")[1].split("```")[0] if style_path else "" + js_path = None if "scripts.js" in index: js_path = webpages_path / "scripts.js" elif "script.js" in index: js_path = webpages_path / "script.js" - else: - js_path = None + js = webpages.split("```javascript")[1].split("```")[0] if js_path else "" except IndexError: - raise ValueError("No css or js code found in the result, please check your image and try again.") + raise ValueError(f"No html or css or js code found in the result. \nWebpages: {webpages}") try: with open(index_path, "w", encoding="utf-8") as f: diff --git a/metagpt/tools/tool_convert.py b/metagpt/tools/tool_convert.py index 417a938e1..fc7cb9a15 100644 --- a/metagpt/tools/tool_convert.py +++ b/metagpt/tools/tool_convert.py @@ -15,7 +15,8 @@ def convert_code_to_tool_schema(obj, include: list[str] = []): # method_doc = inspect.getdoc(method) method_doc = get_class_method_docstring(obj, name) if method_doc: - schema["methods"][name] = docstring_to_schema(method_doc) + function_type = "function" if not inspect.iscoroutinefunction(method) else "async_function" + schema["methods"][name] = {"type": function_type, **docstring_to_schema(method_doc)} elif inspect.isfunction(obj): schema = { diff --git a/tests/metagpt/tools/libs/test_gpt_v_generator.py b/tests/metagpt/tools/libs/test_gpt_v_generator.py index d686d38ba..1b8b756e1 100644 --- a/tests/metagpt/tools/libs/test_gpt_v_generator.py +++ b/tests/metagpt/tools/libs/test_gpt_v_generator.py @@ -5,36 +5,91 @@ @Author : mannaandpoem @File : test_gpt_v_generator.py """ +from pathlib import Path + import pytest from metagpt import logs +from metagpt.const import METAGPT_ROOT from metagpt.tools.libs.gpt_v_generator import GPTvGenerator @pytest.fixture -def mock_webpages(mocker): +def mock_webpage_filename_with_styles_and_scripts(mocker): mock_data = """```html\n\n -\n\n```\n -```css\n.class { ... }\n```\n -```javascript\nfunction() { ... }\n```\n""" - mocker.patch("metagpt.tools.libs.gpt_v_generator.GPTvGenerator.generate_webpages", return_value=mock_data) +\n\n```\n +```css\n/* styles.css */\n```\n +```javascript\n// scripts.js\n```\n""" + mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", return_value=mock_data) return mocker -def test_vision_generate_webpages(mock_webpages): - image_path = "image.png" +@pytest.fixture +def mock_webpage_filename_with_style_and_script(mocker): + mock_data = """```html\n\n +\n\n```\n +```css\n/* style.css */\n```\n +```javascript\n// script.js\n```\n""" + mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", return_value=mock_data) + return mocker + + +@pytest.fixture +def mock_image_layout(mocker): + image_layout = "The layout information of the sketch image is ..." + mocker.patch("metagpt.provider.base_llm.BaseLLM.aask", return_value=image_layout) + return mocker + + +@pytest.fixture +def image_path(): + return f"{METAGPT_ROOT}/docs/resources/workspace/content_rec_sys/resources/competitive_analysis.png" + + +@pytest.mark.asyncio +async def test_generate_webpages_with_suffix_s(mock_webpage_filename_with_styles_and_scripts, image_path): generator = GPTvGenerator() - rsp = generator.generate_webpages(image_path=image_path) + rsp = await generator.generate_webpages(image_path=image_path) logs.logger.info(rsp) assert "html" in rsp assert "css" in rsp assert "javascript" in rsp -def test_save_webpages(mock_webpages): - image_path = "image.png" +@pytest.mark.asyncio +async def test_generate_webpages_without_suffix_s(mock_webpage_filename_with_style_and_script, image_path): generator = GPTvGenerator() - webpages = generator.generate_webpages(image_path) + rsp = await generator.generate_webpages(image_path=image_path) + logs.logger.info(rsp) + assert "html" in rsp + assert "css" in rsp + assert "javascript" in rsp + + +@pytest.mark.asyncio +async def test_save_webpages_with_suffix_s(mock_webpage_filename_with_styles_and_scripts, image_path): + generator = GPTvGenerator() + webpages = await generator.generate_webpages(image_path) webpages_dir = generator.save_webpages(image_path=image_path, webpages=webpages) logs.logger.info(webpages_dir) assert webpages_dir.exists() + + +@pytest.mark.asyncio +async def test_save_webpages_without_suffix_s(mock_webpage_filename_with_style_and_script, image_path): + generator = GPTvGenerator() + webpages = await generator.generate_webpages(image_path) + webpages_dir = generator.save_webpages(image_path=image_path, webpages=webpages) + logs.logger.info(webpages_dir) + assert webpages_dir.exists() + + +@pytest.mark.asyncio +async def test_analyze_layout(mock_image_layout, image_path): + layout = await GPTvGenerator().analyze_layout(Path(image_path)) + logs.logger.info(layout) + assert layout + + +if __name__ == "__main__": + pytest.main([__file__, "-s"]) From 5abde78767cf3861e74be1fce3dc1f4cd1fd8c93 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Mon, 5 Feb 2024 21:54:09 +0800 Subject: [PATCH 612/637] remove get_result method and improve gpt_v_generator.py and test_gpt_v_generator.py. --- tests/metagpt/tools/libs/test_gpt_v_generator.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/tests/metagpt/tools/libs/test_gpt_v_generator.py b/tests/metagpt/tools/libs/test_gpt_v_generator.py index 1b8b756e1..76ada8622 100644 --- a/tests/metagpt/tools/libs/test_gpt_v_generator.py +++ b/tests/metagpt/tools/libs/test_gpt_v_generator.py @@ -47,7 +47,7 @@ def image_path(): @pytest.mark.asyncio -async def test_generate_webpages_with_suffix_s(mock_webpage_filename_with_styles_and_scripts, image_path): +async def test_generate_webpages_with_styles_and_scripts(mock_webpage_filename_with_styles_and_scripts, image_path): generator = GPTvGenerator() rsp = await generator.generate_webpages(image_path=image_path) logs.logger.info(rsp) @@ -57,7 +57,7 @@ async def test_generate_webpages_with_suffix_s(mock_webpage_filename_with_styles @pytest.mark.asyncio -async def test_generate_webpages_without_suffix_s(mock_webpage_filename_with_style_and_script, image_path): +async def test_generate_webpages_with_style_and_script(mock_webpage_filename_with_style_and_script, image_path): generator = GPTvGenerator() rsp = await generator.generate_webpages(image_path=image_path) logs.logger.info(rsp) @@ -67,7 +67,7 @@ async def test_generate_webpages_without_suffix_s(mock_webpage_filename_with_sty @pytest.mark.asyncio -async def test_save_webpages_with_suffix_s(mock_webpage_filename_with_styles_and_scripts, image_path): +async def test_save_webpages_with_styles_and_scripts(mock_webpage_filename_with_styles_and_scripts, image_path): generator = GPTvGenerator() webpages = await generator.generate_webpages(image_path) webpages_dir = generator.save_webpages(image_path=image_path, webpages=webpages) @@ -76,7 +76,7 @@ async def test_save_webpages_with_suffix_s(mock_webpage_filename_with_styles_and @pytest.mark.asyncio -async def test_save_webpages_without_suffix_s(mock_webpage_filename_with_style_and_script, image_path): +async def test_save_webpages_with_style_and_script(mock_webpage_filename_with_style_and_script, image_path): generator = GPTvGenerator() webpages = await generator.generate_webpages(image_path) webpages_dir = generator.save_webpages(image_path=image_path, webpages=webpages) From 9b72370cbebda21d0ad120ef0f42bc1199cb7922 Mon Sep 17 00:00:00 2001 From: yzlin Date: Mon, 5 Feb 2024 22:15:47 +0800 Subject: [PATCH 613/637] update webscraping tool --- examples/crawl_webpage.py | 2 +- metagpt/tools/libs/web_scraping.py | 9 ++++----- 2 files changed, 5 insertions(+), 6 deletions(-) diff --git a/examples/crawl_webpage.py b/examples/crawl_webpage.py index 7dcbf7993..2db9e407b 100644 --- a/examples/crawl_webpage.py +++ b/examples/crawl_webpage.py @@ -10,7 +10,7 @@ from metagpt.roles.ci.code_interpreter import CodeInterpreter async def main(): prompt = """Get data from `paperlist` table in https://papercopilot.com/statistics/iclr-statistics/iclr-2024-statistics/, - and save it to a csv file. paper title must include `multiagent` or `large language model`. *notice: print key data*""" + and save it to a csv file. paper title must include `multiagent` or `large language model`. *notice: print key variables*""" ci = CodeInterpreter(goal=prompt, use_tools=True) await ci.run(prompt) diff --git a/metagpt/tools/libs/web_scraping.py b/metagpt/tools/libs/web_scraping.py index 6fd3b9435..d01e69d09 100644 --- a/metagpt/tools/libs/web_scraping.py +++ b/metagpt/tools/libs/web_scraping.py @@ -4,19 +4,18 @@ from metagpt.tools.web_browser_engine_playwright import PlaywrightWrapper @register_tool(tool_type=ToolType.WEBSCRAPING.type_name) -async def scrape_web_playwright(url, *urls): +async def scrape_web_playwright(url): """ - Scrape and save the HTML structure and inner text content of a web page using Playwright. + Asynchronously Scrape and save the HTML structure and inner text content of a web page using Playwright. Args: url (str): The main URL to fetch inner text from. - *urls (str): Additional URLs to fetch inner text from. Returns: - (dict): The inner text content and html structure of the web page, key are : 'inner_text', 'html'. + dict: The inner text content and html structure of the web page, keys are 'inner_text', 'html'. """ # Create a PlaywrightWrapper instance for the Chromium browser - web = await PlaywrightWrapper().run(url, *urls) + web = await PlaywrightWrapper().run(url) # Return the inner text content of the web page return {"inner_text": web.inner_text.strip(), "html": web.html.strip()} From 675b96b0f5c39ce008c00f278b7eb5b5dc9ca501 Mon Sep 17 00:00:00 2001 From: mannaandpoem <1580466765@qq.com> Date: Tue, 6 Feb 2024 09:07:29 +0800 Subject: [PATCH 614/637] remove attribute openai_vision_model and vision_max_tokens and method test_generate_webpages_with_style_and_script --- metagpt/config2.py | 2 -- tests/metagpt/tools/libs/test_gpt_v_generator.py | 12 +----------- 2 files changed, 1 insertion(+), 13 deletions(-) diff --git a/metagpt/config2.py b/metagpt/config2.py index d983a43c3..bc6af18c6 100644 --- a/metagpt/config2.py +++ b/metagpt/config2.py @@ -75,8 +75,6 @@ class Config(CLIParams, YamlModel): iflytek_api_key: str = "" azure_tts_subscription_key: str = "" azure_tts_region: str = "" - openai_vision_model: str = "gpt-4-vision-preview" - vision_max_tokens: int = 4096 @classmethod def from_home(cls, path): diff --git a/tests/metagpt/tools/libs/test_gpt_v_generator.py b/tests/metagpt/tools/libs/test_gpt_v_generator.py index 76ada8622..907006765 100644 --- a/tests/metagpt/tools/libs/test_gpt_v_generator.py +++ b/tests/metagpt/tools/libs/test_gpt_v_generator.py @@ -47,17 +47,7 @@ def image_path(): @pytest.mark.asyncio -async def test_generate_webpages_with_styles_and_scripts(mock_webpage_filename_with_styles_and_scripts, image_path): - generator = GPTvGenerator() - rsp = await generator.generate_webpages(image_path=image_path) - logs.logger.info(rsp) - assert "html" in rsp - assert "css" in rsp - assert "javascript" in rsp - - -@pytest.mark.asyncio -async def test_generate_webpages_with_style_and_script(mock_webpage_filename_with_style_and_script, image_path): +async def test_generate_webpages(mock_webpage_filename_with_styles_and_scripts, image_path): generator = GPTvGenerator() rsp = await generator.generate_webpages(image_path=image_path) logs.logger.info(rsp) From 1f172b307c45280795770fe41cd2df94b39906dc Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 6 Feb 2024 12:52:53 +0800 Subject: [PATCH 615/637] fix RecursionError: maximum recursion depth exceeded while calling a Python object when run tests/metagpt/environment/test_base_env.py --- metagpt/environment/base_env.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/metagpt/environment/base_env.py b/metagpt/environment/base_env.py index 9c195b023..0e583ffb3 100644 --- a/metagpt/environment/base_env.py +++ b/metagpt/environment/base_env.py @@ -4,7 +4,7 @@ import asyncio from enum import Enum -from typing import TYPE_CHECKING, Any, Iterable, Optional, Set, Union +from typing import TYPE_CHECKING, Any, Dict, Iterable, Optional, Set, Union from pydantic import BaseModel, ConfigDict, Field, SerializeAsAny, model_validator @@ -104,7 +104,7 @@ class Environment(ExtEnv): desc: str = Field(default="") # 环境描述 roles: dict[str, SerializeAsAny["Role"]] = Field(default_factory=dict, validate_default=True) - member_addrs: dict["Role", Set] = Field(default_factory=dict, exclude=True) + member_addrs: Dict["Role", Set] = Field(default_factory=dict, exclude=True) history: str = "" # For debug context: Context = Field(default_factory=Context, exclude=True) From 1855b2ba8ab9125883cb701f1ae1473674d9372d Mon Sep 17 00:00:00 2001 From: better629 Date: Tue, 6 Feb 2024 12:53:40 +0800 Subject: [PATCH 616/637] add cache result for gpt-4v --- tests/data/rsp_cache.json | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index d6cbe60e7..c006dbafe 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -394,5 +394,6 @@ }, "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { "code": "from sklearn import datasets\nimport matplotlib.pyplot as plt\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\niris_df = pd.DataFrame(data=iris.data, columns=iris.feature_names)\niris_df['target'] = iris.target\niris_df['target_name'] = iris_df['target'].apply(lambda x: iris.target_names[x])\n\n# Plotting\nfig, ax = plt.subplots(figsize=(12, 8))\nfor target, target_name in zip(iris.target_names, iris.target_names):\n subset = iris_df[iris_df['target_name'] == target_name]\n ax.scatter(subset[iris.feature_names[0]], subset[iris.feature_names[1]], label=target_name)\n\nax.set_xlabel(iris.feature_names[0])\nax.set_ylabel(iris.feature_names[1])\nax.legend()\nplt.show()" - } + }, + "\n## context\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"invoice\": \"False\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- invoice: # if it's a invoice file, return True else False\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"invoice\": \"True\"\n}\n[/CONTENT]" } \ No newline at end of file From 0fe854fa7f7145179ec726c50a8576b1503073d1 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 6 Feb 2024 14:13:02 +0800 Subject: [PATCH 617/637] fix save code --- tests/metagpt/utils/test_save_code.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/metagpt/utils/test_save_code.py b/tests/metagpt/utils/test_save_code.py index 57a19049b..35ad84baf 100644 --- a/tests/metagpt/utils/test_save_code.py +++ b/tests/metagpt/utils/test_save_code.py @@ -41,4 +41,4 @@ async def test_save_code_file_notebook(): notebook = nbformat.read(file_path, as_version=4) assert len(notebook.cells) > 0, "Notebook should have at least one cell" first_cell_source = notebook.cells[0].source - assert "print('Hello, World!')" in first_cell_source, "Notebook cell content does not match" + assert "print" in first_cell_source, "Notebook cell content does not match" From 5abc5c3812b518a36f6e720adb3e9eaef53a663c Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 6 Feb 2024 15:49:14 +0800 Subject: [PATCH 618/637] add async fn type to tool schema --- metagpt/tools/tool_convert.py | 13 +++++++------ tests/data/rsp_cache.json | 22 ++++++++-------------- tests/metagpt/tools/test_tool_convert.py | 21 +++++++++++++++++++++ 3 files changed, 36 insertions(+), 20 deletions(-) diff --git a/metagpt/tools/tool_convert.py b/metagpt/tools/tool_convert.py index fc7cb9a15..fc29d0693 100644 --- a/metagpt/tools/tool_convert.py +++ b/metagpt/tools/tool_convert.py @@ -15,18 +15,19 @@ def convert_code_to_tool_schema(obj, include: list[str] = []): # method_doc = inspect.getdoc(method) method_doc = get_class_method_docstring(obj, name) if method_doc: - function_type = "function" if not inspect.iscoroutinefunction(method) else "async_function" - schema["methods"][name] = {"type": function_type, **docstring_to_schema(method_doc)} + schema["methods"][name] = function_docstring_to_schema(method, method_doc) elif inspect.isfunction(obj): - schema = { - "type": "function", - **docstring_to_schema(docstring), - } + schema = function_docstring_to_schema(obj, docstring) return schema +def function_docstring_to_schema(fn_obj, docstring): + function_type = "function" if not inspect.iscoroutinefunction(fn_obj) else "async_function" + return {"type": function_type, **docstring_to_schema(docstring)} + + def docstring_to_schema(docstring: str): if docstring is None: return {} diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index c006dbafe..40d7d3953 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -335,16 +335,16 @@ "RobustScale" ] }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", the code can be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n\"}]": { - "code": "# Step 1: Fill missing values for numerical columns\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create a copy of the DataFrame to avoid modifying the original one\ndf_processed = df.copy()\n\n# Identify numerical columns with missing values\nnum_cols_with_missing = df_processed.select_dtypes(include='number').columns[df_processed.isnull().any()].tolist()\n\n# Initialize the FillMissingValue tool for numerical columns\nfill_missing_num = FillMissingValue(features=num_cols_with_missing, strategy='mean')\nfill_missing_num.fit(df_processed)\ndf_processed = fill_missing_num.transform(df_processed)\n\n# Step 2: Encode categorical columns\n# For simplicity, we will replace categorical values with a numerical code\n# This is a simple form of encoding that can be useful for tree-based models\n# and doesn't increase dimensionality like one-hot encoding\ncat_cols = df_processed.select_dtypes(include='object').columns.tolist()\nfor col in cat_cols:\n df_processed[col] = df_processed[col].astype('category').cat.codes\n\n# Display the processed DataFrame\ndisplay(df_processed)" + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", the code can be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n\"}]": { + "code": "# Step 1: Fill missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create a copy of the DataFrame to avoid modifying the original one\ndf_processed = df.copy()\n\n# Define the columns with missing values\nmissing_value_columns = df_processed.columns[df_processed.isnull().any()].tolist()\n\n# Initialize the FillMissingValue tool with the strategy to fill missing values with the mean\nfill_missing_value = FillMissingValue(features=missing_value_columns, strategy='mean')\n\n# Fit and transform the DataFrame\ndf_processed = fill_missing_value.fit_transform(df_processed)\n\n# Step 2: Scale numeric columns\nfrom metagpt.tools.libs.data_preprocess import MinMaxScale\n\n# Define the numeric columns to be scaled\nnumeric_columns = df_processed.select_dtypes(include=['int64', 'float64']).columns.tolist()\n\n# Initialize the MinMaxScale tool\nminmax_scale = MinMaxScale(features=numeric_columns)\n\n# Fit and transform the DataFrame\ndf_processed = minmax_scale.fit_transform(df_processed)\n\n# Display the processed DataFrame\ndisplay(df_processed)" }, "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nclean and preprocess the data\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\n\\n## Available Tools:\\n{'FillMissingValue': 'Filling missing values', 'SplitBins': 'Bin continuous data into intervals and return the bin identifier encoded as an integer value'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { "recommend_tools": [ "FillMissingValue" ] }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Instantiate the FillMissingValue class for column 'b'\nfill_missing_b = FillMissingValue(features=['b'], strategy='mean')\n\n# Fit and transform the DataFrame\ndf = fill_missing_b.fit_transform(df)" + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Instantiate the FillMissingValue class for column 'b'\nfill_missing_b = FillMissingValue(features=['b'], strategy='mean')\n\n# Fit and transform the DataFrame\n# Since we only have one DataFrame, we will use it as both train and test for demonstration\n# In practice, train and test data should be handled separately\ntrain_df = fill_missing_b.fit_transform(df)\ntest_df = fill_missing_b.transform(df)\n\n# Check the results\ntrain_df\n" }, "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n ## User Requirement\\n read a dataset test.csv and print its head\\n ## Current Plan\\n [\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n }\\n ]\\n ## Current Task\\n {\\\"task_id\\\": \\\"1\\\", \\\"dependent_task_ids\\\": [], \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\", \\\"task_type\\\": \\\"\\\", \\\"code\\\": \\\"\\\", \\\"result\\\": \\\"\\\", \\\"is_finished\\\": false}\\n \"}, {\"role\": \"assistant\", \"content\": \"import pandas as pd\\ndata = pd.read_excel('test.csv')\\ndata\"}, {\"role\": \"user\", \"content\": \"\\n Traceback (most recent call last):\\n File \\\"\\\", line 2, in \\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 478, in read_excel\\n io = ExcelFile(io, storage_options=storage_options, engine=engine)\\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 1500, in __init__\\n raise ValueError(\\n ValueError: Excel file format cannot be determined, you must specify an engine manually.\\n \"}]": { "language": "python", @@ -376,11 +376,11 @@ "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"1\\\",\\\"dependent_task_ids\\\":[],\\\"instruction\\\":\\\"Load the sklearn Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { "code": "from sklearn.datasets import load_iris\niris_data = load_iris()" }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "import pandas as pd\nimport numpy as np\n\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\niris_df = pd.DataFrame(data=np.c_[iris_data['data'], iris_data['target']], columns=iris_data['feature_names'] + ['target'])\n\n# Show basic information about the dataset\ndataset_info = iris_df.info()\n\n# Show basic statistics of the dataset\ndataset_description = iris_df.describe()\n\n# Show the first few rows of the dataset\ndataset_head = iris_df.head()\n\n# Count the number of samples for each class\nclass_distribution = iris_df['target'].value_counts()\n\n# Output the results\ndataset_info, dataset_description, dataset_head, class_distribution" + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from sklearn import datasets\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\ndf_iris = pd.DataFrame(data=iris.data, columns=iris.feature_names)\ndf_iris['target'] = iris.target\n\n# Display basic information about the dataset\nprint(df_iris.info())\n\n# Display statistical summary of the dataset\nprint(df_iris.describe())\n\n# Display the first few rows of the dataset\nprint(df_iris.head())\n\n# Display the distribution of the target variable\ntarget_counts = df_iris['target'].value_counts()\nprint(target_counts)" }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\nimport numpy as np\\\\n\\\\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=np.c_[iris_data['data'], iris_data['target']], columns=iris_data['feature_names'] + ['target'])\\\\n\\\\n# Show basic information about the dataset\\\\ndataset_info = iris_df.info()\\\\n\\\\n# Show basic statistics of the dataset\\\\ndataset_description = iris_df.describe()\\\\n\\\\n# Show the first few rows of the dataset\\\\ndataset_head = iris_df.head()\\\\n\\\\n# Count the number of samples for each class\\\\nclass_distribution = iris_df['target'].value_counts()\\\\n\\\\n# Output the results\\\\ndataset_info, dataset_description, dataset_head, class_distribution\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "import matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Set the style of seaborn\nsns.set(style=\"whitegrid\")\n\n# Create a pairplot to visualize the relationships between features\ng = sns.pairplot(iris_df, hue='target', palette='husl', markers=['o', 's', 'D'])\n\n# Add titles and labels\ng.fig.suptitle('Iris Dataset Pairplot', y=1.02) # y=1.02 to make space for the title\nplt.show()" + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from sklearn import datasets\nimport matplotlib.pyplot as plt\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\niris_df = pd.DataFrame(data=iris.data, columns=iris.feature_names)\niris_df['target'] = iris.target\niris_df['target_name'] = iris_df['target'].apply(lambda x: iris.target_names[x])\n\n# Plotting\nfig, ax = plt.subplots(figsize=(12, 8))\nfor target, target_name in zip(iris.target_names, iris.target_names):\n subset = iris_df[iris_df['target_name'] == target_name]\n ax.scatter(subset[iris.feature_names[0]], subset[iris.feature_names[1]], label=target_name)\n\nax.set_xlabel(iris.feature_names[0])\nax.set_ylabel(iris.feature_names[1])\nax.legend()\nplt.show()" }, "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n\\n```end\\n\\n## Current Task\\nPerform exploratory data analysis on the train dataset to understand the features and target variable.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", the code can be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nobj_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n\"}]": { "code": "# Perform exploratory data analysis on the train dataset\ndf_train = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\n\n# Display the first few rows of the dataset\ndisplay(df_train.head())\n\n# Summary statistics for numerical features\ndisplay(df_train.describe())\n\n# Summary information about the dataset including the data types and number of non-null values\ndisplay(df_train.info())\n\n# Distribution of the target variable 'Survived'\nsurvival_counts = df_train['Survived'].value_counts()\nprint(\"Survival counts:\\n\", survival_counts)\n\n# Visualizations\nimport matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Distribution of the target variable\nsns.countplot(x='Survived', data=df_train)\nplt.title('Distribution of Survival')\nplt.show()\n\n# Correlation matrix heatmap to understand the relationship between features\nplt.figure(figsize=(10, 8))\nsns.heatmap(df_train.corr(), annot=True, fmt='.2f')\nplt.title('Correlation Matrix')\nplt.show()\n\n# Pairplot to visualize the pairwise relationships between features\nsns.pairplot(df_train, hue='Survived')\nplt.title('Pairplot of Features')\nplt.show()" @@ -389,11 +389,5 @@ "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not handle the sorting of elements with the same number of ones in their binary representation correctly. According to the context, for similar number of ones, the elements should be sorted based on their decimal value in ascending order. However, the output [1, 2, 4, 3, 5] shows that 3 and 4 are not in the correct order, indicating that the sorting by decimal value is not functioning as intended. To fix this issue, we should ensure that the sorting is stable, so that when two elements have the same number of ones, they maintain their relative order if their decimal values are already in ascending order.", "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(n).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n # We use a stable sorting algorithm to maintain the relative order of elements with the same number of ones\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x), reverse=True)\n sorted_arr = sorted(sorted_arr, key=count_ones)\n \n return sorted_arr\n" }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from sklearn import datasets\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\ndf_iris = pd.DataFrame(data=iris.data, columns=iris.feature_names)\ndf_iris['target'] = iris.target\n\n# Display basic information about the dataset\nprint(df_iris.info())\n\n# Display statistical summary of the dataset\nprint(df_iris.describe())\n\n# Display the first few rows of the dataset\nprint(df_iris.head())\n\n# Display the distribution of the target variable\ntarget_counts = df_iris['target'].value_counts()\nprint(target_counts)" - }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from sklearn import datasets\nimport matplotlib.pyplot as plt\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\niris_df = pd.DataFrame(data=iris.data, columns=iris.feature_names)\niris_df['target'] = iris.target\niris_df['target_name'] = iris_df['target'].apply(lambda x: iris.target_names[x])\n\n# Plotting\nfig, ax = plt.subplots(figsize=(12, 8))\nfor target, target_name in zip(iris.target_names, iris.target_names):\n subset = iris_df[iris_df['target_name'] == target_name]\n ax.scatter(subset[iris.feature_names[0]], subset[iris.feature_names[1]], label=target_name)\n\nax.set_xlabel(iris.feature_names[0])\nax.set_ylabel(iris.feature_names[1])\nax.legend()\nplt.show()" - }, "\n## context\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"invoice\": \"False\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- invoice: # if it's a invoice file, return True else False\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"invoice\": \"True\"\n}\n[/CONTENT]" } \ No newline at end of file diff --git a/tests/metagpt/tools/test_tool_convert.py b/tests/metagpt/tools/test_tool_convert.py index 2ae2ea000..8f26a211c 100644 --- a/tests/metagpt/tools/test_tool_convert.py +++ b/tests/metagpt/tools/test_tool_convert.py @@ -95,12 +95,26 @@ def dummy_fn(df: pd.DataFrame) -> dict: pass +async def dummy_async_fn(df: pd.DataFrame) -> dict: + """ + A dummy async function for test + + Args: + df (pd.DataFrame): test args. + + Returns: + dict: test returns. + """ + pass + + def test_convert_code_to_tool_schema_class(): expected = { "type": "class", "description": "Completing missing values with simple strategies.", "methods": { "__init__": { + "type": "function", "description": "Initialize self.", "parameters": { "properties": { @@ -121,6 +135,7 @@ def test_convert_code_to_tool_schema_class(): }, }, "fit": { + "type": "function", "description": "Fit the FillMissingValue model.", "parameters": { "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, @@ -128,6 +143,7 @@ def test_convert_code_to_tool_schema_class(): }, }, "transform": { + "type": "function", "description": "Transform the input DataFrame with the fitted model.", "parameters": { "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, @@ -152,3 +168,8 @@ def test_convert_code_to_tool_schema_function(): } schema = convert_code_to_tool_schema(dummy_fn) assert schema == expected + + +def test_convert_code_to_tool_schema_async_function(): + schema = convert_code_to_tool_schema(dummy_async_fn) + assert schema.get("type") == "async_function" From beda29bc273f3204ce3fb4b0099997046fdc07f5 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 6 Feb 2024 15:49:14 +0800 Subject: [PATCH 619/637] add async fn type to tool schema --- metagpt/tools/tool_convert.py | 13 +++++++------ tests/data/rsp_cache.json | 22 ++++++++-------------- tests/metagpt/tools/test_tool_convert.py | 21 +++++++++++++++++++++ 3 files changed, 36 insertions(+), 20 deletions(-) diff --git a/metagpt/tools/tool_convert.py b/metagpt/tools/tool_convert.py index fc7cb9a15..fc29d0693 100644 --- a/metagpt/tools/tool_convert.py +++ b/metagpt/tools/tool_convert.py @@ -15,18 +15,19 @@ def convert_code_to_tool_schema(obj, include: list[str] = []): # method_doc = inspect.getdoc(method) method_doc = get_class_method_docstring(obj, name) if method_doc: - function_type = "function" if not inspect.iscoroutinefunction(method) else "async_function" - schema["methods"][name] = {"type": function_type, **docstring_to_schema(method_doc)} + schema["methods"][name] = function_docstring_to_schema(method, method_doc) elif inspect.isfunction(obj): - schema = { - "type": "function", - **docstring_to_schema(docstring), - } + schema = function_docstring_to_schema(obj, docstring) return schema +def function_docstring_to_schema(fn_obj, docstring): + function_type = "function" if not inspect.iscoroutinefunction(fn_obj) else "async_function" + return {"type": function_type, **docstring_to_schema(docstring)} + + def docstring_to_schema(docstring: str): if docstring is None: return {} diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index c006dbafe..40d7d3953 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -335,16 +335,16 @@ "RobustScale" ] }, - "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", the code can be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n\"}]": { - "code": "# Step 1: Fill missing values for numerical columns\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create a copy of the DataFrame to avoid modifying the original one\ndf_processed = df.copy()\n\n# Identify numerical columns with missing values\nnum_cols_with_missing = df_processed.select_dtypes(include='number').columns[df_processed.isnull().any()].tolist()\n\n# Initialize the FillMissingValue tool for numerical columns\nfill_missing_num = FillMissingValue(features=num_cols_with_missing, strategy='mean')\nfill_missing_num.fit(df_processed)\ndf_processed = fill_missing_num.transform(df_processed)\n\n# Step 2: Encode categorical columns\n# For simplicity, we will replace categorical values with a numerical code\n# This is a simple form of encoding that can be useful for tree-based models\n# and doesn't increase dimensionality like one-hot encoding\ncat_cols = df_processed.select_dtypes(include='object').columns.tolist()\nfor col in cat_cols:\n df_processed[col] = df_processed[col].astype('category').cat.codes\n\n# Display the processed DataFrame\ndisplay(df_processed)" + "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [构造数据集并进行数据清洗] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n import pandas as pd\\n df = pd.DataFrame({\\n 'a': [1, 2, 3, 4, 5],\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\n 'd': [1, 2, 3, 4, 5]\\n })\\n```end\\n\\n## Current Task\\n对数据集进行数据清洗\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools:\\nEach Class tool is described in JSON format. When you call a tool, import the tool from its path first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Output Example:\\nwhen current task is \\\"do data preprocess, like fill missing value, handle outliers, etc.\\\", the code can be like:\\n```python\\n# Step 1: fill missing value\\n# Tools used: ['FillMissingValue']\\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\\n\\ntrain_processed = train.copy()\\ntest_processed = test.copy()\\nnum_cols = train_processed.select_dtypes(include='number').columns.tolist()\\nif 'label' in num_cols:\\n num_cols.remove('label')\\nfill_missing_value = FillMissingValue(features=num_cols, strategy='mean')\\nfill_missing_value.fit(train_processed)\\ntrain_processed = fill_missing_value.transform(train_processed)\\ntest_processed = fill_missing_value.transform(test_processed)\\n\\n# Step 2: handle outliers\\nfor col in num_cols:\\n low, high = train_processed[col].quantile([0.01, 0.99])\\n train_processed[col] = train_processed[col].clip(low, high)\\n test_processed[col] = test_processed[col].clip(low, high)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n- Always copy the DataFrame before processing it and use the copy to process.\\n\"}]": { + "code": "# Step 1: Fill missing values\nfrom metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Create a copy of the DataFrame to avoid modifying the original one\ndf_processed = df.copy()\n\n# Define the columns with missing values\nmissing_value_columns = df_processed.columns[df_processed.isnull().any()].tolist()\n\n# Initialize the FillMissingValue tool with the strategy to fill missing values with the mean\nfill_missing_value = FillMissingValue(features=missing_value_columns, strategy='mean')\n\n# Fit and transform the DataFrame\ndf_processed = fill_missing_value.fit_transform(df_processed)\n\n# Step 2: Scale numeric columns\nfrom metagpt.tools.libs.data_preprocess import MinMaxScale\n\n# Define the numeric columns to be scaled\nnumeric_columns = df_processed.select_dtypes(include=['int64', 'float64']).columns.tolist()\n\n# Initialize the MinMaxScale tool\nminmax_scale = MinMaxScale(features=numeric_columns)\n\n# Fit and transform the DataFrame\ndf_processed = minmax_scale.fit_transform(df_processed)\n\n# Display the processed DataFrame\ndisplay(df_processed)" }, "[{\"role\": \"user\", \"content\": \"\\n## User Requirement:\\nclean and preprocess the data\\n\\n## Task\\nRecommend up to five tools from 'Available Tools' that can help solve the 'User Requirement'. \\n\\n## Available Tools:\\n{'FillMissingValue': 'Filling missing values', 'SplitBins': 'Bin continuous data into intervals and return the bin identifier encoded as an integer value'}\\n\\n## Tool Selection and Instructions:\\n- Select tools most relevant to completing the 'User Requirement'.\\n- If you believe that no tools are suitable, indicate with an empty list.\\n- Only list the names of the tools, not the full schema of each tool.\\n- Ensure selected tools are listed in 'Available Tools'.\\n\"}]": { "recommend_tools": [ "FillMissingValue" ] }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Instantiate the FillMissingValue class for column 'b'\nfill_missing_b = FillMissingValue(features=['b'], strategy='mean')\n\n# Fit and transform the DataFrame\ndf = fill_missing_b.fit_transform(df)" + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\n构造数据集并进行数据清洗\\n## Context\\n\\n## Current Plan\\n[Task(task_id='1', dependent_task_ids=[], instruction='随机生成一个pandas DataFrame数据集', task_type='other', code=\\\"\\\\n import pandas as pd\\\\n df = pd.DataFrame({\\\\n 'a': [1, 2, 3, 4, 5],\\\\n 'b': [1.1, 2.2, 3.3, 4.4, np.nan],\\\\n 'c': ['aa', 'bb', 'cc', 'dd', 'ee'],\\\\n 'd': [1, 2, 3, 4, 5]\\\\n })\\\\n \\\", result='', is_success=False, is_finished=True), Task(task_id='2', dependent_task_ids=['1'], instruction='对数据集进行数据清洗', task_type='data_preprocess', code='', result='', is_success=False, is_finished=False)]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"对数据集进行数据清洗\\\",\\\"task_type\\\":\\\"data_preprocess\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\nThe current task is about data preprocessing, please note the following:\\n- Monitor data types per column, applying appropriate methods.\\n- Ensure operations are on existing dataset columns.\\n- Avoid writing processed data to files.\\n- Avoid any change to label column, such as standardization, etc.\\n- Prefer alternatives to one-hot encoding for categorical data.\\n- Only encode or scale necessary columns to allow for potential feature-specific engineering tasks (like time_extract, binning, extraction, etc.) later.\\n- Each step do data preprocessing to train, must do same for test separately at the same time.\\n\\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{'FillMissingValue': {'type': 'class', 'description': 'Completing missing values with simple strategies.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}, 'strategy': {'type': 'str', 'description': \\\"The imputation strategy, notice 'mean' and 'median' can only be used for numeric features. Enum: ['mean', 'median', 'most_frequent', 'constant']. Defaults to 'mean'.\\\", 'default': \\\"'mean'\\\", 'enum': [\\\"'mean'\\\", \\\"'median'\\\", \\\"'most_frequent'\\\", \\\"'constant'\\\"]}, 'fill_value': {'type': 'int', 'description': 'Fill_value is used to replace all occurrences of missing_values. Defaults to None.', 'default': 'None'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MinMaxScale': {'type': 'class', 'description': 'Transform features by scaling each feature to a range, which is (0, 1).', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'StandardScale': {'type': 'class', 'description': 'Standardize features by removing the mean and scaling to unit variance.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'MaxAbsScale': {'type': 'class', 'description': 'Scale each feature by its maximum absolute value.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}, 'RobustScale': {'type': 'class', 'description': 'Apply the RobustScaler to scale features using statistics that are robust to outliers.', 'methods': {'__init__': {'type': 'function', 'description': 'Initialize self.', 'parameters': {'properties': {'features': {'type': 'list', 'description': 'Columns to be processed.'}}, 'required': ['features']}}, 'fit': {'type': 'function', 'description': 'Fit a model to be used in subsequent transform.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}}, 'fit_transform': {'type': 'function', 'description': 'Fit and transform the input DataFrame.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}, 'transform': {'type': 'function', 'description': 'Transform the input DataFrame with the fitted model.', 'parameters': {'properties': {'df': {'type': 'pd.DataFrame', 'description': 'The input DataFrame.'}}, 'required': ['df']}, 'returns': [{'type': 'pd.DataFrame', 'description': 'The transformed DataFrame.'}]}}, 'tool_path': 'metagpt/tools/libs/data_preprocess.py'}}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from metagpt.tools.libs.data_preprocess import FillMissingValue\n\n# Instantiate the FillMissingValue class for column 'b'\nfill_missing_b = FillMissingValue(features=['b'], strategy='mean')\n\n# Fit and transform the DataFrame\n# Since we only have one DataFrame, we will use it as both train and test for demonstration\n# In practice, train and test data should be handled separately\ntrain_df = fill_missing_b.fit_transform(df)\ntest_df = fill_missing_b.transform(df)\n\n# Check the results\ntrain_df\n" }, "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n ## User Requirement\\n read a dataset test.csv and print its head\\n ## Current Plan\\n [\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Print the head of the dataset to display the first few rows.\\\",\\n \\\"task_type\\\": \\\"\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_finished\\\": false\\n }\\n ]\\n ## Current Task\\n {\\\"task_id\\\": \\\"1\\\", \\\"dependent_task_ids\\\": [], \\\"instruction\\\": \\\"import pandas and load the dataset from 'test.csv'.\\\", \\\"task_type\\\": \\\"\\\", \\\"code\\\": \\\"\\\", \\\"result\\\": \\\"\\\", \\\"is_finished\\\": false}\\n \"}, {\"role\": \"assistant\", \"content\": \"import pandas as pd\\ndata = pd.read_excel('test.csv')\\ndata\"}, {\"role\": \"user\", \"content\": \"\\n Traceback (most recent call last):\\n File \\\"\\\", line 2, in \\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 478, in read_excel\\n io = ExcelFile(io, storage_options=storage_options, engine=engine)\\n File \\\"/Users/gary/miniconda3/envs/py39_scratch/lib/python3.9/site-packages/pandas/io/excel/_base.py\\\", line 1500, in __init__\\n raise ValueError(\\n ValueError: Excel file format cannot be determined, you must specify an engine manually.\\n \"}]": { "language": "python", @@ -376,11 +376,11 @@ "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"1\\\",\\\"dependent_task_ids\\\":[],\\\"instruction\\\":\\\"Load the sklearn Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { "code": "from sklearn.datasets import load_iris\niris_data = load_iris()" }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "import pandas as pd\nimport numpy as np\n\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\niris_df = pd.DataFrame(data=np.c_[iris_data['data'], iris_data['target']], columns=iris_data['feature_names'] + ['target'])\n\n# Show basic information about the dataset\ndataset_info = iris_df.info()\n\n# Show basic statistics of the dataset\ndataset_description = iris_df.describe()\n\n# Show the first few rows of the dataset\ndataset_head = iris_df.head()\n\n# Count the number of samples for each class\nclass_distribution = iris_df['target'].value_counts()\n\n# Output the results\ndataset_info, dataset_description, dataset_head, class_distribution" + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from sklearn import datasets\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\ndf_iris = pd.DataFrame(data=iris.data, columns=iris.feature_names)\ndf_iris['target'] = iris.target\n\n# Display basic information about the dataset\nprint(df_iris.info())\n\n# Display statistical summary of the dataset\nprint(df_iris.describe())\n\n# Display the first few rows of the dataset\nprint(df_iris.head())\n\n# Display the distribution of the target variable\ntarget_counts = df_iris['target'].value_counts()\nprint(target_counts)" }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\nimport numpy as np\\\\n\\\\n# Convert the loaded Iris dataset to a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=np.c_[iris_data['data'], iris_data['target']], columns=iris_data['feature_names'] + ['target'])\\\\n\\\\n# Show basic information about the dataset\\\\ndataset_info = iris_df.info()\\\\n\\\\n# Show basic statistics of the dataset\\\\ndataset_description = iris_df.describe()\\\\n\\\\n# Show the first few rows of the dataset\\\\ndataset_head = iris_df.head()\\\\n\\\\n# Count the number of samples for each class\\\\nclass_distribution = iris_df['target'].value_counts()\\\\n\\\\n# Output the results\\\\ndataset_info, dataset_description, dataset_head, class_distribution\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "import matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Set the style of seaborn\nsns.set(style=\"whitegrid\")\n\n# Create a pairplot to visualize the relationships between features\ng = sns.pairplot(iris_df, hue='target', palette='husl', markers=['o', 's', 'D'])\n\n# Add titles and labels\ng.fig.suptitle('Iris Dataset Pairplot', y=1.02) # y=1.02 to make space for the title\nplt.show()" + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "from sklearn import datasets\nimport matplotlib.pyplot as plt\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\niris_df = pd.DataFrame(data=iris.data, columns=iris.feature_names)\niris_df['target'] = iris.target\niris_df['target_name'] = iris_df['target'].apply(lambda x: iris.target_names[x])\n\n# Plotting\nfig, ax = plt.subplots(figsize=(12, 8))\nfor target, target_name in zip(iris.target_names, iris.target_names):\n subset = iris_df[iris_df['target_name'] == target_name]\n ax.scatter(subset[iris.feature_names[0]], subset[iris.feature_names[1]], label=target_name)\n\nax.set_xlabel(iris.feature_names[0])\nax.set_ylabel(iris.feature_names[1])\nax.legend()\nplt.show()" }, "[{\"role\": \"user\", \"content\": \"\\n# Background\\nAs a data scientist, you need to help user to achieve their goal [This is a titanic passenger survival dataset, your goal is to predict passenger survival outcome. The target column is Survived. Perform data analysis, data preprocessing, feature engineering, and modeling to predict the target. Report accuracy on the eval data. Train data path: 'tests/data/ml_datasets/titanic/split_train.csv', eval data path: 'tests/data/ml_datasets/titanic/split_eval.csv'.] step-by-step in an continuous Jupyter notebook.\\n\\n## Done Tasks\\n```python\\n\\n```end\\n\\n## Current Task\\nPerform exploratory data analysis on the train dataset to understand the features and target variable.\\n\\n# Latest Data Info\\nLatest data info after previous tasks:\\n\\n\\n# Task\\nWrite complete code for 'Current Task'. And avoid duplicating code from 'Done Tasks', such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Output Example:\\nwhen current task is \\\"train a lightgbm model on training data\\\", the code can be like:\\n```python\\n# Step 1: check data type and convert to numeric\\nobj_cols = train.select_dtypes(include='object').columns.tolist()\\n\\nfor col in obj_cols:\\n encoder = LabelEncoder()\\n train[col] = encoder.fit_transform(train[col].unique().tolist() + ['unknown'])\\n test[col] = test[col].apply(lambda x: x if x in encoder.classes_ else 'unknown')\\n test[col] = encoder.transform(test[col])\\n\\n# Step 2: train lightgbm model\\nmodel = LGBMClassifier()\\nmodel.fit(train, y_train)\\n```end\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n\"}]": { "code": "# Perform exploratory data analysis on the train dataset\ndf_train = pd.read_csv('tests/data/ml_datasets/titanic/split_train.csv')\n\n# Display the first few rows of the dataset\ndisplay(df_train.head())\n\n# Summary statistics for numerical features\ndisplay(df_train.describe())\n\n# Summary information about the dataset including the data types and number of non-null values\ndisplay(df_train.info())\n\n# Distribution of the target variable 'Survived'\nsurvival_counts = df_train['Survived'].value_counts()\nprint(\"Survival counts:\\n\", survival_counts)\n\n# Visualizations\nimport matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Distribution of the target variable\nsns.countplot(x='Survived', data=df_train)\nplt.title('Distribution of Survival')\nplt.show()\n\n# Correlation matrix heatmap to understand the relationship between features\nplt.figure(figsize=(10, 8))\nsns.heatmap(df_train.corr(), annot=True, fmt='.2f')\nplt.title('Correlation Matrix')\nplt.show()\n\n# Pairplot to visualize the pairwise relationships between features\nsns.pairplot(df_train, hue='Survived')\nplt.title('Pairplot of Features')\nplt.show()" @@ -389,11 +389,5 @@ "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not handle the sorting of elements with the same number of ones in their binary representation correctly. According to the context, for similar number of ones, the elements should be sorted based on their decimal value in ascending order. However, the output [1, 2, 4, 3, 5] shows that 3 and 4 are not in the correct order, indicating that the sorting by decimal value is not functioning as intended. To fix this issue, we should ensure that the sorting is stable, so that when two elements have the same number of ones, they maintain their relative order if their decimal values are already in ascending order.", "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(n).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n # We use a stable sorting algorithm to maintain the relative order of elements with the same number of ones\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x), reverse=True)\n sorted_arr = sorted(sorted_arr, key=count_ones)\n \n return sorted_arr\n" }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from sklearn import datasets\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\ndf_iris = pd.DataFrame(data=iris.data, columns=iris.feature_names)\ndf_iris['target'] = iris.target\n\n# Display basic information about the dataset\nprint(df_iris.info())\n\n# Display statistical summary of the dataset\nprint(df_iris.describe())\n\n# Display the first few rows of the dataset\nprint(df_iris.head())\n\n# Display the distribution of the target variable\ntarget_counts = df_iris['target'].value_counts()\nprint(target_counts)" - }, - "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { - "code": "from sklearn import datasets\nimport matplotlib.pyplot as plt\nimport pandas as pd\n\n# Load Iris dataset\niris = datasets.load_iris()\niris_df = pd.DataFrame(data=iris.data, columns=iris.feature_names)\niris_df['target'] = iris.target\niris_df['target_name'] = iris_df['target'].apply(lambda x: iris.target_names[x])\n\n# Plotting\nfig, ax = plt.subplots(figsize=(12, 8))\nfor target, target_name in zip(iris.target_names, iris.target_names):\n subset = iris_df[iris_df['target_name'] == target_name]\n ax.scatter(subset[iris.feature_names[0]], subset[iris.feature_names[1]], label=target_name)\n\nax.set_xlabel(iris.feature_names[0])\nax.set_ylabel(iris.feature_names[1])\nax.legend()\nplt.show()" - }, "\n## context\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"invoice\": \"False\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- invoice: # if it's a invoice file, return True else False\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"invoice\": \"True\"\n}\n[/CONTENT]" } \ No newline at end of file diff --git a/tests/metagpt/tools/test_tool_convert.py b/tests/metagpt/tools/test_tool_convert.py index 2ae2ea000..8f26a211c 100644 --- a/tests/metagpt/tools/test_tool_convert.py +++ b/tests/metagpt/tools/test_tool_convert.py @@ -95,12 +95,26 @@ def dummy_fn(df: pd.DataFrame) -> dict: pass +async def dummy_async_fn(df: pd.DataFrame) -> dict: + """ + A dummy async function for test + + Args: + df (pd.DataFrame): test args. + + Returns: + dict: test returns. + """ + pass + + def test_convert_code_to_tool_schema_class(): expected = { "type": "class", "description": "Completing missing values with simple strategies.", "methods": { "__init__": { + "type": "function", "description": "Initialize self.", "parameters": { "properties": { @@ -121,6 +135,7 @@ def test_convert_code_to_tool_schema_class(): }, }, "fit": { + "type": "function", "description": "Fit the FillMissingValue model.", "parameters": { "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, @@ -128,6 +143,7 @@ def test_convert_code_to_tool_schema_class(): }, }, "transform": { + "type": "function", "description": "Transform the input DataFrame with the fitted model.", "parameters": { "properties": {"df": {"type": "pd.DataFrame", "description": "The input DataFrame."}}, @@ -152,3 +168,8 @@ def test_convert_code_to_tool_schema_function(): } schema = convert_code_to_tool_schema(dummy_fn) assert schema == expected + + +def test_convert_code_to_tool_schema_async_function(): + schema = convert_code_to_tool_schema(dummy_async_fn) + assert schema.get("type") == "async_function" From ee4aba206e80e3d0f144f8713f1ae17458d77cd5 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 6 Feb 2024 23:22:53 +0800 Subject: [PATCH 620/637] fix empty code when aask_code not returning language --- metagpt/roles/ci/code_interpreter.py | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/metagpt/roles/ci/code_interpreter.py b/metagpt/roles/ci/code_interpreter.py index 796abba04..404c93b81 100644 --- a/metagpt/roles/ci/code_interpreter.py +++ b/metagpt/roles/ci/code_interpreter.py @@ -72,11 +72,7 @@ class CodeInterpreter(Role): if ReviewConst.CHANGE_WORDS[0] in review: counter = 0 # redo the task again with help of human suggestions - py_code = ( - code["code"] if code.get("language") == "python" else "" - ) # use python code as final code; for markdown, return the rendered result instead of the code itself - - return py_code, result, success + return code["code"], result, success async def _write_code(self): todo = WriteCodeWithoutTools() if not self.use_tools else WriteCodeWithTools(selected_tools=self.tools) From 78989b0eb7dd012442cb480ceed217d8ecc28f03 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 6 Feb 2024 23:37:24 +0800 Subject: [PATCH 621/637] skip two individual tests --- tests/metagpt/actions/test_rebuild_class_view.py | 1 + tests/metagpt/actions/test_summarize_code.py | 1 + 2 files changed, 2 insertions(+) diff --git a/tests/metagpt/actions/test_rebuild_class_view.py b/tests/metagpt/actions/test_rebuild_class_view.py index 403109cc0..2188d6b85 100644 --- a/tests/metagpt/actions/test_rebuild_class_view.py +++ b/tests/metagpt/actions/test_rebuild_class_view.py @@ -14,6 +14,7 @@ from metagpt.actions.rebuild_class_view import RebuildClassView from metagpt.llm import LLM +@pytest.mark.skip @pytest.mark.asyncio async def test_rebuild(context): action = RebuildClassView( diff --git a/tests/metagpt/actions/test_summarize_code.py b/tests/metagpt/actions/test_summarize_code.py index a404047c1..3cfe7ca81 100644 --- a/tests/metagpt/actions/test_summarize_code.py +++ b/tests/metagpt/actions/test_summarize_code.py @@ -176,6 +176,7 @@ class Snake: """ +@pytest.mark.skip @pytest.mark.asyncio async def test_summarize_code(context): git_dir = Path(__file__).parent / f"unittest/{uuid.uuid4().hex}" From ea6c440294650122c3e6df6f8d3a722b746bbe89 Mon Sep 17 00:00:00 2001 From: yzlin Date: Tue, 6 Feb 2024 23:57:39 +0800 Subject: [PATCH 622/637] add ut to check code saving --- tests/data/rsp_cache.json | 6 ++++++ tests/metagpt/roles/ci/test_code_interpreter.py | 4 ++++ 2 files changed, 10 insertions(+) diff --git a/tests/data/rsp_cache.json b/tests/data/rsp_cache.json index 40d7d3953..75fc9ceb2 100644 --- a/tests/data/rsp_cache.json +++ b/tests/data/rsp_cache.json @@ -389,5 +389,11 @@ "reflection": "The implementation failed the test case where the input array is [1, 5, 2, 3, 4]. The issue arises because the code does not handle the sorting of elements with the same number of ones in their binary representation correctly. According to the context, for similar number of ones, the elements should be sorted based on their decimal value in ascending order. However, the output [1, 2, 4, 3, 5] shows that 3 and 4 are not in the correct order, indicating that the sorting by decimal value is not functioning as intended. To fix this issue, we should ensure that the sorting is stable, so that when two elements have the same number of ones, they maintain their relative order if their decimal values are already in ascending order.", "improved_impl": "def sort_array(arr):\n # Helper function to count the number of ones in the binary representation\n def count_ones(n):\n return bin(n).count('1')\n \n # Sort the array using a custom key function\n # The key function returns a tuple (number of ones, value) for each element\n # This ensures that if two elements have the same number of ones, they are sorted by their value\n # We use a stable sorting algorithm to maintain the relative order of elements with the same number of ones\n sorted_arr = sorted(arr, key=lambda x: (count_ones(x), x), reverse=True)\n sorted_arr = sorted(sorted_arr, key=count_ones)\n \n return sorted_arr\n" }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"2\\\",\\\"dependent_task_ids\\\":[\\\"1\\\"],\\\"instruction\\\":\\\"Perform exploratory data analysis on the Iris dataset.\\\",\\\"task_type\\\":\\\"eda\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "import pandas as pd\n\n# Convert the Iris dataset to a DataFrame for easier manipulation\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\niris_df['target'] = iris_data.target\niris_df['target_names'] = iris_df['target'].apply(lambda x: iris_data.target_names[x])\n\n# Display basic information about the dataset\niris_df.info()\n\n# Display statistical summary of the dataset\niris_df.describe()\n\n# Display the first few rows of the dataset\ndisplay(iris_df.head())" + }, + "[{\"role\": \"system\", \"content\": \"You are Code Interpreter, a world-class programmer that can complete any goal by executing code. Strictly follow the plan and generate code step by step. Each step of the code will be executed on the user's machine, and the user will provide the code execution results to you.**Notice: The code for the next step depends on the code for the previous step. Must reuse variables in the lastest other code directly, dont creat it again, it is very import for you. Use !pip install in a standalone block to install missing packages.Usually the libraries you need are already installed.Dont check if packages already imported.**\"}, {\"role\": \"user\", \"content\": \"\\n## User Requirement\\nRun data analysis on sklearn Iris dataset, include a plot\\n## Context\\n\\n## Current Plan\\n[\\n {\\n \\\"task_id\\\": \\\"1\\\",\\n \\\"dependent_task_ids\\\": [],\\n \\\"instruction\\\": \\\"Load the sklearn Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"from sklearn.datasets import load_iris\\\\niris_data = load_iris()\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"2\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"1\\\"\\n ],\\n \\\"instruction\\\": \\\"Perform exploratory data analysis on the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"eda\\\",\\n \\\"code\\\": \\\"import pandas as pd\\\\n\\\\n# Convert the Iris dataset to a DataFrame for easier manipulation\\\\niris_df = pd.DataFrame(data=iris_data.data, columns=iris_data.feature_names)\\\\niris_df['target'] = iris_data.target\\\\niris_df['target_names'] = iris_df['target'].apply(lambda x: iris_data.target_names[x])\\\\n\\\\n# Display basic information about the dataset\\\\niris_df.info()\\\\n\\\\n# Display statistical summary of the dataset\\\\niris_df.describe()\\\\n\\\\n# Display the first few rows of the dataset\\\\ndisplay(iris_df.head())\\\",\\n \\\"result\\\": \\\"a successful run\\\",\\n \\\"is_success\\\": true,\\n \\\"is_finished\\\": true\\n },\\n {\\n \\\"task_id\\\": \\\"3\\\",\\n \\\"dependent_task_ids\\\": [\\n \\\"2\\\"\\n ],\\n \\\"instruction\\\": \\\"Create a plot visualizing the Iris dataset.\\\",\\n \\\"task_type\\\": \\\"other\\\",\\n \\\"code\\\": \\\"\\\",\\n \\\"result\\\": \\\"\\\",\\n \\\"is_success\\\": false,\\n \\\"is_finished\\\": false\\n }\\n]\\n## Current Task\\n{\\\"task_id\\\":\\\"3\\\",\\\"dependent_task_ids\\\":[\\\"2\\\"],\\\"instruction\\\":\\\"Create a plot visualizing the Iris dataset.\\\",\\\"task_type\\\":\\\"other\\\",\\\"code\\\":\\\"\\\",\\\"result\\\":\\\"\\\",\\\"is_success\\\":false,\\\"is_finished\\\":false}\\n\"}, {\"role\": \"user\", \"content\": \"\\n# Instruction\\nWrite complete code for 'Current Task'. And avoid duplicating code from finished tasks, such as repeated import of packages, reading data, etc.\\nSpecifically, \\n\\n# Capabilities\\n- You can utilize pre-defined tools in any code lines from 'Available Tools' in the form of Python Class.\\n- You can freely combine the use of any other public packages, like sklearn, numpy, pandas, etc..\\n\\n# Available Tools (can be empty):\\nEach Class tool is described in JSON format. When you call a tool, import the tool first.\\n{}\\n\\n# Constraints:\\n- Ensure the output new code is executable in the same Jupyter notebook with previous tasks code have been executed.\\n- Always prioritize using pre-defined tools for the same functionality.\\n\"}]": { + "code": "import matplotlib.pyplot as plt\nimport seaborn as sns\n\n# Set the style of seaborn\nsns.set_style('whitegrid')\n\n# Pairplot to visualize the relationships between features\nsns.pairplot(iris_df, hue='target_names', markers='+')\nplt.show()" + }, "\n## context\n\n\n-----\n\n## format example\n[CONTENT]\n{\n \"invoice\": \"False\"\n}\n[/CONTENT]\n\n## nodes: \": # \"\n- invoice: # if it's a invoice file, return True else False\n\n\n## constraint\nLanguage: Please use the same language as Human INPUT.\nFormat: output wrapped inside [CONTENT][/CONTENT] like format example, nothing else.\n\n## action\nFollow instructions of nodes, generate output and make sure it follows the format example.\n": "[CONTENT]\n{\n \"invoice\": \"True\"\n}\n[/CONTENT]" } \ No newline at end of file diff --git a/tests/metagpt/roles/ci/test_code_interpreter.py b/tests/metagpt/roles/ci/test_code_interpreter.py index f23292965..9d2f2429b 100644 --- a/tests/metagpt/roles/ci/test_code_interpreter.py +++ b/tests/metagpt/roles/ci/test_code_interpreter.py @@ -17,3 +17,7 @@ async def test_code_interpreter(mocker, auto_run): rsp = await ci.run(requirement) logger.info(rsp) assert len(rsp.content) > 0 + + finished_tasks = ci.planner.plan.get_finished_tasks() + assert len(finished_tasks) > 0 + assert len(finished_tasks[0].code) > 0 # check one task to see if code is recorded From 6f31289e7e0efd96a22400b31df8179eab286875 Mon Sep 17 00:00:00 2001 From: better629 Date: Wed, 7 Feb 2024 10:02:15 +0800 Subject: [PATCH 623/637] re-commit zhipu-api due to merge mistake --- examples/llm_hello_world.py | 8 ------- examples/llm_vision.py | 23 ++++++++++++++++++ metagpt/provider/general_api_requestor.py | 3 ++- metagpt/provider/zhipuai_api.py | 28 ++++++++++------------ metagpt/utils/token_counter.py | 7 +++--- tests/metagpt/provider/test_zhipuai_api.py | 4 ++-- 6 files changed, 43 insertions(+), 30 deletions(-) create mode 100644 examples/llm_vision.py diff --git a/examples/llm_hello_world.py b/examples/llm_hello_world.py index 1d132eb8a..219a303c8 100644 --- a/examples/llm_hello_world.py +++ b/examples/llm_hello_world.py @@ -6,11 +6,9 @@ @File : llm_hello_world.py """ import asyncio -from pathlib import Path from metagpt.llm import LLM from metagpt.logs import logger -from metagpt.utils.common import encode_image async def main(): @@ -29,12 +27,6 @@ async def main(): if hasattr(llm, "completion"): logger.info(llm.completion(hello_msg)) - # check if the configured llm supports llm-vision capacity. If not, it will throw a error - invoice_path = Path(__file__).parent.joinpath("..", "tests", "data", "invoices", "invoice-2.png") - img_base64 = encode_image(invoice_path) - res = await llm.aask(msg="if this is a invoice, just return True else return False", images=[img_base64]) - assert "true" in res.lower() - if __name__ == "__main__": asyncio.run(main()) diff --git a/examples/llm_vision.py b/examples/llm_vision.py new file mode 100644 index 000000000..276decd59 --- /dev/null +++ b/examples/llm_vision.py @@ -0,0 +1,23 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : example to run the ability of LLM vision + +import asyncio +from pathlib import Path + +from metagpt.llm import LLM +from metagpt.utils.common import encode_image + + +async def main(): + llm = LLM() + + # check if the configured llm supports llm-vision capacity. If not, it will throw a error + invoice_path = Path(__file__).parent.joinpath("..", "tests", "data", "invoices", "invoice-2.png") + img_base64 = encode_image(invoice_path) + res = await llm.aask(msg="if this is a invoice, just return True else return False", images=[img_base64]) + assert "true" in res.lower() + + +if __name__ == "__main__": + asyncio.run(main()) diff --git a/metagpt/provider/general_api_requestor.py b/metagpt/provider/general_api_requestor.py index 500cd1426..18f4dd909 100644 --- a/metagpt/provider/general_api_requestor.py +++ b/metagpt/provider/general_api_requestor.py @@ -60,7 +60,8 @@ class GeneralAPIRequestor(APIRequestor): self, result: requests.Response, stream: bool ) -> Tuple[Union[bytes, Iterator[Generator]], bytes]: """Returns the response(s) and a bool indicating whether it is a stream.""" - if stream and "text/event-stream" in result.headers.get("Content-Type", ""): + content_type = result.headers.get("Content-Type", "") + if stream and ("text/event-stream" in content_type or "application/x-ndjson" in content_type): return ( self._interpret_response_line(line, result.status_code, result.headers, stream=True) for line in parse_stream(result.iter_lines()) diff --git a/metagpt/provider/zhipuai_api.py b/metagpt/provider/zhipuai_api.py index 9108a1fba..9e8e5fb53 100644 --- a/metagpt/provider/zhipuai_api.py +++ b/metagpt/provider/zhipuai_api.py @@ -3,9 +3,8 @@ # @Desc : zhipuai LLM from https://open.bigmodel.cn/dev/api#sdk from enum import Enum +from typing import Optional -import openai -import zhipuai from requests import ConnectionError from tenacity import ( after_log, @@ -14,6 +13,7 @@ from tenacity import ( stop_after_attempt, wait_random_exponential, ) +from zhipuai.types.chat.chat_completion import Completion from metagpt.configs.llm_config import LLMConfig, LLMType from metagpt.logs import log_llm_stream, logger @@ -21,6 +21,7 @@ from metagpt.provider.base_llm import BaseLLM from metagpt.provider.llm_provider_registry import register_provider from metagpt.provider.openai_api import log_and_reraise from metagpt.provider.zhipuai.zhipu_model_api import ZhiPuModelAPI +from metagpt.utils.cost_manager import CostManager class ZhiPuEvent(Enum): @@ -38,20 +39,15 @@ class ZhiPuAILLM(BaseLLM): """ def __init__(self, config: LLMConfig): - self.__init_zhipuai(config) - self.llm = ZhiPuModelAPI - self.model = "chatglm_turbo" # so far only one model, just use it - self.use_system_prompt: bool = False # zhipuai has no system prompt when use api self.config = config + self.__init_zhipuai() + self.cost_manager: Optional[CostManager] = None - def __init_zhipuai(self, config: LLMConfig): - assert config.api_key - zhipuai.api_key = config.api_key - # due to use openai sdk, set the api_key but it will't be used. - # openai.api_key = zhipuai.api_key # due to use openai sdk, set the api_key but it will't be used. - if config.proxy: - # FIXME: openai v1.x sdk has no proxy support - openai.proxy = config.proxy + def __init_zhipuai(self): + assert self.config.api_key + self.api_key = self.config.api_key + self.model = self.config.model # so far, it support glm-3-turbo、glm-4 + self.llm = ZhiPuModelAPI(api_key=self.api_key) def _const_kwargs(self, messages: list[dict], stream: bool = False) -> dict: kwargs = {"model": self.model, "messages": messages, "stream": stream, "temperature": 0.3} @@ -63,12 +59,12 @@ class ZhiPuAILLM(BaseLLM): try: prompt_tokens = int(usage.get("prompt_tokens", 0)) completion_tokens = int(usage.get("completion_tokens", 0)) - self.config.cost_manager.update_cost(prompt_tokens, completion_tokens, self.model) + self.cost_manager.update_cost(prompt_tokens, completion_tokens, self.model) except Exception as e: logger.error(f"zhipuai updats costs failed! exp: {e}") def completion(self, messages: list[dict], timeout=3) -> dict: - resp = self.llm.chat.completions.create(**self._const_kwargs(messages)) + resp: Completion = self.llm.chat.completions.create(**self._const_kwargs(messages)) usage = resp.usage.model_dump() self._update_costs(usage) return resp.model_dump() diff --git a/metagpt/utils/token_counter.py b/metagpt/utils/token_counter.py index a0fb3b70d..65f5fe76f 100644 --- a/metagpt/utils/token_counter.py +++ b/metagpt/utils/token_counter.py @@ -32,8 +32,8 @@ TOKEN_COSTS = { "gpt-4-vision-preview": {"prompt": 0.01, "completion": 0.03}, # TODO add extra image price calculator "gpt-4-1106-vision-preview": {"prompt": 0.01, "completion": 0.03}, "text-embedding-ada-002": {"prompt": 0.0004, "completion": 0.0}, - "glm-3-turbo": {"prompt": 0.0, "completion": 0.0007}, # 128k version, prompt + completion tokens=0.005¥/k-tokens - "glm-4": {"prompt": 0.0, "completion": 0.014}, # 128k version, prompt + completion tokens=0.1¥/k-tokens + "glm-3-turbo": {"prompt": 0.0007, "completion": 0.0007}, # 128k version, prompt + completion tokens=0.005¥/k-tokens + "glm-4": {"prompt": 0.014, "completion": 0.014}, # 128k version, prompt + completion tokens=0.1¥/k-tokens "gemini-pro": {"prompt": 0.00025, "completion": 0.0005}, } @@ -58,7 +58,8 @@ TOKEN_MAX = { "gpt-4-vision-preview": 128000, "gpt-4-1106-vision-preview": 128000, "text-embedding-ada-002": 8192, - "chatglm_turbo": 32768, + "glm-3-turbo": 128000, + "glm-4": 128000, "gemini-pro": 32768, } diff --git a/tests/metagpt/provider/test_zhipuai_api.py b/tests/metagpt/provider/test_zhipuai_api.py index 798209710..ad2ececa2 100644 --- a/tests/metagpt/provider/test_zhipuai_api.py +++ b/tests/metagpt/provider/test_zhipuai_api.py @@ -17,7 +17,7 @@ default_resp = { } -async def mock_zhipuai_acreate_stream(**kwargs): +async def mock_zhipuai_acreate_stream(self, **kwargs): class MockResponse(object): async def _aread(self): class Iterator(object): @@ -37,7 +37,7 @@ async def mock_zhipuai_acreate_stream(**kwargs): return MockResponse() -async def mock_zhipuai_acreate(**kwargs) -> dict: +async def mock_zhipuai_acreate(self, **kwargs) -> dict: return default_resp From 3b4379d12569cae719ff58f6c39208eed05483aa Mon Sep 17 00:00:00 2001 From: voidking Date: Wed, 7 Feb 2024 10:34:04 +0800 Subject: [PATCH 624/637] chore: move the required playwright to requirements.txt --- requirements.txt | 2 +- setup.py | 1 - 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/requirements.txt b/requirements.txt index 6cb25d52b..804ff4359 100644 --- a/requirements.txt +++ b/requirements.txt @@ -63,7 +63,7 @@ gitignore-parser==0.1.9 websockets~=12.0 networkx~=3.2.1 google-generativeai==0.3.2 -# playwright==1.40.0 # playwright extras require +playwright>=1.26 # used at metagpt/tools/libs/web_scraping.py anytree ipywidgets==8.1.1 Pillow diff --git a/setup.py b/setup.py index b16d978cf..be3956ea4 100644 --- a/setup.py +++ b/setup.py @@ -24,7 +24,6 @@ requirements = (here / "requirements.txt").read_text(encoding="utf-8").splitline extras_require = { - "playwright": ["playwright>=1.26", "beautifulsoup4"], "selenium": ["selenium>4", "webdriver_manager", "beautifulsoup4"], "search-google": ["google-api-python-client==2.94.0"], "search-ddg": ["duckduckgo-search~=4.1.1"], From 63ab24a77bbb850baed77b515941342d48329aca Mon Sep 17 00:00:00 2001 From: voidking Date: Wed, 7 Feb 2024 11:54:31 +0800 Subject: [PATCH 625/637] chore: add one more space --- requirements.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/requirements.txt b/requirements.txt index 804ff4359..1426500ce 100644 --- a/requirements.txt +++ b/requirements.txt @@ -63,7 +63,7 @@ gitignore-parser==0.1.9 websockets~=12.0 networkx~=3.2.1 google-generativeai==0.3.2 -playwright>=1.26 # used at metagpt/tools/libs/web_scraping.py +playwright>=1.26 # used at metagpt/tools/libs/web_scraping.py anytree ipywidgets==8.1.1 Pillow From d180d3912e33aca2c5968f4a80c6a94b2189d020 Mon Sep 17 00:00:00 2001 From: better629 Date: Wed, 7 Feb 2024 15:56:01 +0800 Subject: [PATCH 626/637] add qianfan api support --- examples/llm_hello_world.py | 21 +++-- metagpt/configs/llm_config.py | 8 +- metagpt/provider/__init__.py | 2 + metagpt/provider/base_llm.py | 16 ++++ metagpt/provider/qianfan_api.py | 151 ++++++++++++++++++++++++++++++++ metagpt/utils/cost_manager.py | 4 +- metagpt/utils/token_counter.py | 53 +++++++++++ requirements.txt | 1 + 8 files changed, 245 insertions(+), 11 deletions(-) create mode 100644 metagpt/provider/qianfan_api.py diff --git a/examples/llm_hello_world.py b/examples/llm_hello_world.py index 1d132eb8a..e22edbdf2 100644 --- a/examples/llm_hello_world.py +++ b/examples/llm_hello_world.py @@ -6,16 +6,25 @@ @File : llm_hello_world.py """ import asyncio -from pathlib import Path from metagpt.llm import LLM from metagpt.logs import logger -from metagpt.utils.common import encode_image async def main(): llm = LLM() - logger.info(await llm.aask("hello world")) + # llm type check + id_ques = "what's your name" + logger.info(f"{id_ques}: ") + logger.info(await llm.aask(id_ques)) + logger.info("\n\n") + + logger.info( + await llm.aask( + "who are you", system_msgs=["act as a robot, answer 'I'am robot' if the question is 'who are you'"] + ) + ) + logger.info(await llm.aask_batch(["hi", "write python hello world."])) hello_msg = [{"role": "user", "content": "count from 1 to 10. split by newline."}] @@ -29,12 +38,6 @@ async def main(): if hasattr(llm, "completion"): logger.info(llm.completion(hello_msg)) - # check if the configured llm supports llm-vision capacity. If not, it will throw a error - invoice_path = Path(__file__).parent.joinpath("..", "tests", "data", "invoices", "invoice-2.png") - img_base64 = encode_image(invoice_path) - res = await llm.aask(msg="if this is a invoice, just return True else return False", images=[img_base64]) - assert "true" in res.lower() - if __name__ == "__main__": asyncio.run(main()) diff --git a/metagpt/configs/llm_config.py b/metagpt/configs/llm_config.py index fb923d3e4..1b05b5270 100644 --- a/metagpt/configs/llm_config.py +++ b/metagpt/configs/llm_config.py @@ -24,6 +24,7 @@ class LLMType(Enum): METAGPT = "metagpt" AZURE = "azure" OLLAMA = "ollama" + QIANFAN = "qianfan" # Baidu BCE def __missing__(self, key): return self.OPENAI @@ -36,13 +37,18 @@ class LLMConfig(YamlModel): Optional Fields in pydantic: https://docs.pydantic.dev/latest/migration/#required-optional-and-nullable-fields """ - api_key: str + api_key: str = "sk-" api_type: LLMType = LLMType.OPENAI base_url: str = "https://api.openai.com/v1" api_version: Optional[str] = None model: Optional[str] = None # also stands for DEPLOYMENT_NAME + # For Cloud Service Provider like Baidu/ Alibaba + access_key: Optional[str] = None + secret_key: Optional[str] = None + endpoint: Optional[str] = None # for self-deployed model on the cloud + # For Spark(Xunfei), maybe remove later app_id: Optional[str] = None api_secret: Optional[str] = None diff --git a/metagpt/provider/__init__.py b/metagpt/provider/__init__.py index 675734811..8c0aab836 100644 --- a/metagpt/provider/__init__.py +++ b/metagpt/provider/__init__.py @@ -16,6 +16,7 @@ from metagpt.provider.azure_openai_api import AzureOpenAILLM from metagpt.provider.metagpt_api import MetaGPTLLM from metagpt.provider.human_provider import HumanProvider from metagpt.provider.spark_api import SparkLLM +from metagpt.provider.qianfan_api import QianFanLLM __all__ = [ "FireworksLLM", @@ -28,4 +29,5 @@ __all__ = [ "OllamaLLM", "HumanProvider", "SparkLLM", + "QianFanLLM", ] diff --git a/metagpt/provider/base_llm.py b/metagpt/provider/base_llm.py index b144471b5..d3d9c829b 100644 --- a/metagpt/provider/base_llm.py +++ b/metagpt/provider/base_llm.py @@ -67,6 +67,22 @@ class BaseLLM(ABC): def _default_system_msg(self): return self._system_msg(self.system_prompt) + def _update_costs(self, usage: dict, model: str = None, local_calc_usage: bool = True): + """update each request's token cost + Args: + model (str): model name or in some scenarios called endpoint + local_calc_usage (bool): some models don't calculate usage, it will overwrite calc_usage + """ + calc_usage = self.config.calc_usage and local_calc_usage + model = model if model else self.model + if calc_usage and self.cost_manager: + try: + prompt_tokens = int(usage.get("prompt_tokens", 0)) + completion_tokens = int(usage.get("completion_tokens", 0)) + self.cost_manager.update_cost(prompt_tokens, completion_tokens, model) + except Exception as e: + logger.error(f"{self.__class__.__name__} updats costs failed! exp: {e}") + async def aask( self, msg: str, diff --git a/metagpt/provider/qianfan_api.py b/metagpt/provider/qianfan_api.py new file mode 100644 index 000000000..180935e61 --- /dev/null +++ b/metagpt/provider/qianfan_api.py @@ -0,0 +1,151 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : llm api of qianfan from Baidu, supports ERNIE(wen xin yi yan) and opensource models +import copy +import os + +import qianfan +from qianfan.resources.typing import JsonBody +from tenacity import ( + after_log, + retry, + retry_if_exception_type, + stop_after_attempt, + wait_random_exponential, +) + +from metagpt.configs.llm_config import LLMConfig, LLMType +from metagpt.logs import log_llm_stream, logger +from metagpt.provider.base_llm import BaseLLM +from metagpt.provider.llm_provider_registry import register_provider +from metagpt.provider.openai_api import log_and_reraise +from metagpt.utils.cost_manager import CostManager +from metagpt.utils.token_counter import ( + QianFan_EndPoint_TOKEN_COSTS, + QianFan_MODEL_TOKEN_COSTS, +) + + +@register_provider(LLMType.QIANFAN) +class QianFanLLM(BaseLLM): + """ + Refs + Auth: https://cloud.baidu.com/doc/WENXINWORKSHOP/s/3lmokh7n6#%E3%80%90%E6%8E%A8%E8%8D%90%E3%80%91%E4%BD%BF%E7%94%A8%E5%AE%89%E5%85%A8%E8%AE%A4%E8%AF%81aksk%E9%89%B4%E6%9D%83%E8%B0%83%E7%94%A8%E6%B5%81%E7%A8%8B + Token Price: https://cloud.baidu.com/doc/WENXINWORKSHOP/s/hlrk4akp7#tokens%E5%90%8E%E4%BB%98%E8%B4%B9 + Models: https://cloud.baidu.com/doc/WENXINWORKSHOP/s/wlmhm7vuo#%E5%AF%B9%E8%AF%9Dchat + https://cloud.baidu.com/doc/WENXINWORKSHOP/s/xlmokikxe#%E6%94%AF%E6%8C%81%E6%A8%A1%E5%9E%8B%E5%88%97%E8%A1%A8 + """ + + def __init__(self, config: LLMConfig): + self.config = config + self.use_system_prompt = False # only some ERNIE-x related models support system_prompt + self.__init_qianfan() + self.cost_manager = CostManager(token_costs=self.token_costs) + + def __init_qianfan(self): + if self.config.access_key and self.config.secret_key: + # for system level auth, use access_key and secret_key, recommended by official + # set environment variable due to official recommendation + os.environ.setdefault("QIANFAN_ACCESS_KEY", self.config.access_key) + os.environ.setdefault("QIANFAN_SECRET_KEY", self.config.secret_key) + elif self.config.api_key and self.config.secret_key: + # for application level auth, use api_key and secret_key + # set environment variable due to official recommendation + os.environ.setdefault("QIANFAN_AK", self.config.api_key) + os.environ.setdefault("QIANFAN_SK", self.config.secret_key) + else: + raise ValueError("Set the `access_key`&`secret_key` or `api_key`&`secret_key` first") + + support_system_pairs = [ + ("ERNIE-Bot-4", "completions_pro"), # (model, corresponding-endpoint) + ("ERNIE-Bot-8k", "ernie_bot_8k"), + ("ERNIE-Bot", "completions"), + ("ERNIE-Bot-turbo", "eb-instant"), + ("ERNIE-Speed", "ernie_speed"), + ("EB-turbo-AppBuilder", "ai_apaas"), + ] + if self.config.model in [pair[0] for pair in support_system_pairs]: + # only some ERNIE models support + self.use_system_prompt = True + if self.config.endpoint in [pair[1] for pair in support_system_pairs]: + self.use_system_prompt = True + + assert not (self.config.model and self.config.endpoint), "Only set `model` or `endpoint` in the config" + assert self.config.model or self.config.endpoint, "Should set one of `model` or `endpoint` in the config" + + self.token_costs = copy.deepcopy(QianFan_MODEL_TOKEN_COSTS) + self.token_costs.update(QianFan_EndPoint_TOKEN_COSTS) + + # self deployed model on the cloud not to calculate usage, it charges resource pool rental fee + self.calc_usage = self.config.calc_usage and self.config.endpoint is None + self.client = qianfan.ChatCompletion() + + def _const_kwargs(self, messages: list[dict], stream: bool = False) -> dict: + kwargs = { + "messages": messages, + "stream": stream, + } + if self.config.temperature > 0: + # different model has default temperature. only set when it's specified. + kwargs["temperature"] = self.config.temperature + if self.config.endpoint: + kwargs["endpoint"] = self.config.endpoint + elif self.config.model: + kwargs["model"] = self.config.model + + if self.use_system_prompt: + # if the model support system prompt, extract and pass it + if messages[0]["role"] == "system": + kwargs["messages"] = messages[1:] + kwargs["system"] = messages[0]["content"] # set system prompt here + return kwargs + + def _update_costs(self, usage: dict): + """update each request's token cost""" + model_or_endpoint = self.config.model if self.config.model else self.config.endpoint + local_calc_usage = True if model_or_endpoint in self.token_costs else False + super()._update_costs(usage, model_or_endpoint, local_calc_usage) + + def get_choice_text(self, resp: JsonBody) -> str: + return resp.get("result", "") + + def completion(self, messages: list[dict]) -> JsonBody: + resp = self.client.do(**self._const_kwargs(messages=messages, stream=False)) + self._update_costs(resp.body.get("usage", {})) + return resp.body + + async def _achat_completion(self, messages: list[dict]) -> JsonBody: + resp = await self.client.ado(**self._const_kwargs(messages=messages, stream=False)) + self._update_costs(resp.body.get("usage", {})) + return resp.body + + async def acompletion(self, messages: list[dict], timeout=3) -> JsonBody: + return await self._achat_completion(messages) + + async def _achat_completion_stream(self, messages: list[dict]) -> str: + resp = await self.client.ado(**self._const_kwargs(messages=messages, stream=True)) + collected_content = [] + usage = {} + async for chunk in resp: + content = chunk.body.get("result", "") + usage = chunk.body.get("usage", {}) + log_llm_stream(content) + collected_content.append(content) + log_llm_stream("\n") + + self._update_costs(usage) + full_content = "".join(collected_content) + return full_content + + @retry( + stop=stop_after_attempt(3), + wait=wait_random_exponential(min=1, max=60), + after=after_log(logger, logger.level("WARNING").name), + retry=retry_if_exception_type(ConnectionError), + retry_error_callback=log_and_reraise, + ) + async def acompletion_text(self, messages: list[dict], stream=False, timeout: int = 3) -> str: + if stream: + return await self._achat_completion_stream(messages) + resp = await self._achat_completion(messages) + return self.get_choice_text(resp) diff --git a/metagpt/utils/cost_manager.py b/metagpt/utils/cost_manager.py index 7bf5154b6..e1c0f415b 100644 --- a/metagpt/utils/cost_manager.py +++ b/metagpt/utils/cost_manager.py @@ -29,6 +29,7 @@ class CostManager(BaseModel): total_budget: float = 0 max_budget: float = 10.0 total_cost: float = 0 + token_costs: dict[str, dict[str, float]] = TOKEN_COSTS def update_cost(self, prompt_tokens, completion_tokens, model): """ @@ -42,7 +43,8 @@ class CostManager(BaseModel): self.total_prompt_tokens += prompt_tokens self.total_completion_tokens += completion_tokens cost = ( - prompt_tokens * TOKEN_COSTS[model]["prompt"] + completion_tokens * TOKEN_COSTS[model]["completion"] + prompt_tokens * self.token_costs[model]["prompt"] + + completion_tokens * self.token_costs[model]["completion"] ) / 1000 self.total_cost += cost logger.info( diff --git a/metagpt/utils/token_counter.py b/metagpt/utils/token_counter.py index a0fb3b70d..b69ec73d3 100644 --- a/metagpt/utils/token_counter.py +++ b/metagpt/utils/token_counter.py @@ -38,6 +38,59 @@ TOKEN_COSTS = { } +""" +QianFan Token Price https://cloud.baidu.com/doc/WENXINWORKSHOP/s/hlrk4akp7#tokens%E5%90%8E%E4%BB%98%E8%B4%B9 +Due to QianFan has multi price strategies, we unify `Tokens post-payment` as a statistical method. +""" +QianFan_MODEL_TOKEN_COSTS = { + "ERNIE-Bot-4": {"prompt": 0.017, "completion": 0.017}, + "ERNIE-Bot-8k": {"prompt": 0.0034, "completion": 0.0067}, + "ERNIE-Bot": {"prompt": 0.017, "completion": 0.017}, + "ERNIE-Bot-turbo": {"prompt": 0.0011, "completion": 0.0011}, + "EB-turbo-AppBuilder": {"prompt": 0.0011, "completion": 0.0011}, + "ERNIE-Speed": {"prompt": 0.00056, "completion": 0.0011}, + "BLOOMZ-7B": {"prompt": 0.00056, "completion": 0.00056}, + "Llama-2-7B-Chat": {"prompt": 0.00056, "completion": 0.00056}, + "Llama-2-13B-Chat": {"prompt": 0.00084, "completion": 0.00084}, + "Llama-2-70B-Chat": {"prompt": 0.0049, "completion": 0.0049}, + "ChatGLM2-6B-32K": {"prompt": 0.00056, "completion": 0.00056}, + "AquilaChat-7B": {"prompt": 0.00056, "completion": 0.00056}, + "Mixtral-8x7B-Instruct": {"prompt": 0.0049, "completion": 0.0049}, + "SQLCoder-7B": {"prompt": 0.00056, "completion": 0.00056}, + "CodeLlama-7B-Instruct": {"prompt": 0.00056, "completion": 0.00056}, + "XuanYuan-70B-Chat-4bit": {"prompt": 0.0049, "completion": 0.0049}, + "Qianfan-BLOOMZ-7B-compressed": {"prompt": 0.00056, "completion": 0.00056}, + "Qianfan-Chinese-Llama-2-7B": {"prompt": 0.00056, "completion": 0.00056}, + "Qianfan-Chinese-Llama-2-13B": {"prompt": 0.00084, "completion": 0.00084}, + "ChatLaw": {"prompt": 0.0011, "completion": 0.0011}, + "Yi-34B-Chat": {"prompt": 0.0, "completion": 0.0}, +} + +QianFan_EndPoint_TOKEN_COSTS = { + "completions_pro": QianFan_MODEL_TOKEN_COSTS["ERNIE-Bot-4"], + "ernie_bot_8k": QianFan_MODEL_TOKEN_COSTS["ERNIE-Bot-8k"], + "completions": QianFan_MODEL_TOKEN_COSTS["ERNIE-Bot"], + "eb-instant": QianFan_MODEL_TOKEN_COSTS["ERNIE-Bot-turbo"], + "ai_apaas": QianFan_MODEL_TOKEN_COSTS["EB-turbo-AppBuilder"], + "ernie_speed": QianFan_MODEL_TOKEN_COSTS["ERNIE-Speed"], + "bloomz_7b1": QianFan_MODEL_TOKEN_COSTS["BLOOMZ-7B"], + "llama_2_7b": QianFan_MODEL_TOKEN_COSTS["Llama-2-7B-Chat"], + "llama_2_13b": QianFan_MODEL_TOKEN_COSTS["Llama-2-13B-Chat"], + "llama_2_70b": QianFan_MODEL_TOKEN_COSTS["Llama-2-70B-Chat"], + "chatglm2_6b_32k": QianFan_MODEL_TOKEN_COSTS["ChatGLM2-6B-32K"], + "aquilachat_7b": QianFan_MODEL_TOKEN_COSTS["AquilaChat-7B"], + "mixtral_8x7b_instruct": QianFan_MODEL_TOKEN_COSTS["Mixtral-8x7B-Instruct"], + "sqlcoder_7b": QianFan_MODEL_TOKEN_COSTS["SQLCoder-7B"], + "codellama_7b_instruct": QianFan_MODEL_TOKEN_COSTS["CodeLlama-7B-Instruct"], + "xuanyuan_70b_chat": QianFan_MODEL_TOKEN_COSTS["XuanYuan-70B-Chat-4bit"], + "qianfan_bloomz_7b_compressed": QianFan_MODEL_TOKEN_COSTS["Qianfan-BLOOMZ-7B-compressed"], + "qianfan_chinese_llama_2_7b": QianFan_MODEL_TOKEN_COSTS["Qianfan-Chinese-Llama-2-7B"], + "qianfan_chinese_llama_2_13b": QianFan_MODEL_TOKEN_COSTS["Qianfan-Chinese-Llama-2-13B"], + "chatlaw": QianFan_MODEL_TOKEN_COSTS["ChatLaw"], + "yi_34b_chat": QianFan_MODEL_TOKEN_COSTS["Yi-34B-Chat"], +} + + TOKEN_MAX = { "gpt-3.5-turbo": 4096, "gpt-3.5-turbo-0301": 4096, diff --git a/requirements.txt b/requirements.txt index 6cb25d52b..c893bd713 100644 --- a/requirements.txt +++ b/requirements.txt @@ -67,3 +67,4 @@ google-generativeai==0.3.2 anytree ipywidgets==8.1.1 Pillow +qianfan==0.3.1 From 15a9c5e94135992e9854a57d14d581040879386f Mon Sep 17 00:00:00 2001 From: better629 Date: Wed, 7 Feb 2024 16:16:14 +0800 Subject: [PATCH 627/637] simplify _update_costs and related code --- metagpt/provider/base_llm.py | 13 ++++++++++--- metagpt/provider/fireworks_api.py | 15 ++------------- metagpt/provider/google_gemini_api.py | 10 ---------- metagpt/provider/ollama_api.py | 10 ---------- metagpt/provider/open_llm_api.py | 13 +------------ metagpt/provider/openai_api.py | 17 ++--------------- metagpt/provider/qianfan_api.py | 8 ++++---- metagpt/provider/zhipuai_api.py | 10 ---------- 8 files changed, 19 insertions(+), 77 deletions(-) diff --git a/metagpt/provider/base_llm.py b/metagpt/provider/base_llm.py index d3d9c829b..2f57b15aa 100644 --- a/metagpt/provider/base_llm.py +++ b/metagpt/provider/base_llm.py @@ -11,11 +11,12 @@ from abc import ABC, abstractmethod from typing import Optional, Union from openai import AsyncOpenAI +from pydantic import BaseModel from metagpt.configs.llm_config import LLMConfig from metagpt.logs import logger from metagpt.schema import Message -from metagpt.utils.cost_manager import CostManager +from metagpt.utils.cost_manager import CostManager, Costs class BaseLLM(ABC): @@ -67,14 +68,15 @@ class BaseLLM(ABC): def _default_system_msg(self): return self._system_msg(self.system_prompt) - def _update_costs(self, usage: dict, model: str = None, local_calc_usage: bool = True): + def _update_costs(self, usage: Union[dict, BaseModel], model: str = None, local_calc_usage: bool = True): """update each request's token cost Args: model (str): model name or in some scenarios called endpoint - local_calc_usage (bool): some models don't calculate usage, it will overwrite calc_usage + local_calc_usage (bool): some models don't calculate usage, it will overwrite LLMConfig.calc_usage """ calc_usage = self.config.calc_usage and local_calc_usage model = model if model else self.model + usage = usage.model_dump() if isinstance(usage, BaseModel) else usage if calc_usage and self.cost_manager: try: prompt_tokens = int(usage.get("prompt_tokens", 0)) @@ -83,6 +85,11 @@ class BaseLLM(ABC): except Exception as e: logger.error(f"{self.__class__.__name__} updats costs failed! exp: {e}") + def get_costs(self) -> Costs: + if not self.cost_manager: + return Costs(0, 0, 0, 0) + return self.cost_manager.get_costs() + async def aask( self, msg: str, diff --git a/metagpt/provider/fireworks_api.py b/metagpt/provider/fireworks_api.py index d56453a85..e62a7066e 100644 --- a/metagpt/provider/fireworks_api.py +++ b/metagpt/provider/fireworks_api.py @@ -19,7 +19,7 @@ from metagpt.configs.llm_config import LLMConfig, LLMType from metagpt.logs import logger from metagpt.provider.llm_provider_registry import register_provider from metagpt.provider.openai_api import OpenAILLM, log_and_reraise -from metagpt.utils.cost_manager import CostManager, Costs +from metagpt.utils.cost_manager import CostManager MODEL_GRADE_TOKEN_COSTS = { "-1": {"prompt": 0.0, "completion": 0.0}, # abnormal condition @@ -81,17 +81,6 @@ class FireworksLLM(OpenAILLM): kwargs = dict(api_key=self.config.api_key, base_url=self.config.base_url) return kwargs - def _update_costs(self, usage: CompletionUsage): - if self.config.calc_usage and usage: - try: - # use FireworksCostManager not context.cost_manager - self.cost_manager.update_cost(usage.prompt_tokens, usage.completion_tokens, self.model) - except Exception as e: - logger.error(f"updating costs failed!, exp: {e}") - - def get_costs(self) -> Costs: - return self.cost_manager.get_costs() - async def _achat_completion_stream(self, messages: list[dict], timeout=3) -> str: response: AsyncStream[ChatCompletionChunk] = await self.aclient.chat.completions.create( **self._cons_kwargs(messages), stream=True @@ -113,7 +102,7 @@ class FireworksLLM(OpenAILLM): usage = CompletionUsage(**chunk.usage) full_content = "".join(collected_content) - self._update_costs(usage) + self._update_costs(usage.model_dump()) return full_content @retry( diff --git a/metagpt/provider/google_gemini_api.py b/metagpt/provider/google_gemini_api.py index 2647ab16b..87ea81c80 100644 --- a/metagpt/provider/google_gemini_api.py +++ b/metagpt/provider/google_gemini_api.py @@ -72,16 +72,6 @@ class GeminiLLM(BaseLLM): kwargs = {"contents": messages, "generation_config": GenerationConfig(temperature=0.3), "stream": stream} return kwargs - def _update_costs(self, usage: dict): - """update each request's token cost""" - if self.config.calc_usage: - try: - prompt_tokens = int(usage.get("prompt_tokens", 0)) - completion_tokens = int(usage.get("completion_tokens", 0)) - self.cost_manager.update_cost(prompt_tokens, completion_tokens, self.model) - except Exception as e: - logger.error(f"google gemini updats costs failed! exp: {e}") - def get_choice_text(self, resp: GenerateContentResponse) -> str: return resp.text diff --git a/metagpt/provider/ollama_api.py b/metagpt/provider/ollama_api.py index c9103b018..52e8dbe36 100644 --- a/metagpt/provider/ollama_api.py +++ b/metagpt/provider/ollama_api.py @@ -46,16 +46,6 @@ class OllamaLLM(BaseLLM): kwargs = {"model": self.model, "messages": messages, "options": {"temperature": 0.3}, "stream": stream} return kwargs - def _update_costs(self, usage: dict): - """update each request's token cost""" - if self.config.calc_usage: - try: - prompt_tokens = int(usage.get("prompt_tokens", 0)) - completion_tokens = int(usage.get("completion_tokens", 0)) - self._cost_manager.update_cost(prompt_tokens, completion_tokens, self.model) - except Exception as e: - logger.error(f"ollama updats costs failed! exp: {e}") - def get_choice_text(self, resp: dict) -> str: """get the resp content from llm response""" assist_msg = resp.get("message", {}) diff --git a/metagpt/provider/open_llm_api.py b/metagpt/provider/open_llm_api.py index a29b263a4..69371e379 100644 --- a/metagpt/provider/open_llm_api.py +++ b/metagpt/provider/open_llm_api.py @@ -8,7 +8,7 @@ from metagpt.configs.llm_config import LLMConfig, LLMType from metagpt.logs import logger from metagpt.provider.llm_provider_registry import register_provider from metagpt.provider.openai_api import OpenAILLM -from metagpt.utils.cost_manager import Costs, TokenCostManager +from metagpt.utils.cost_manager import TokenCostManager from metagpt.utils.token_counter import count_message_tokens, count_string_tokens @@ -34,14 +34,3 @@ class OpenLLM(OpenAILLM): logger.error(f"usage calculation failed!: {e}") return usage - - def _update_costs(self, usage: CompletionUsage): - if self.config.calc_usage and usage: - try: - # use OpenLLMCostManager not CONFIG.cost_manager - self._cost_manager.update_cost(usage.prompt_tokens, usage.completion_tokens, self.model) - except Exception as e: - logger.error(f"updating costs failed!, exp: {e}") - - def get_costs(self) -> Costs: - return self._cost_manager.get_costs() diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 63e68c9bd..1e5770d74 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -29,7 +29,7 @@ from metagpt.provider.constant import GENERAL_FUNCTION_SCHEMA from metagpt.provider.llm_provider_registry import register_provider from metagpt.schema import Message from metagpt.utils.common import CodeParser, decode_image -from metagpt.utils.cost_manager import CostManager, Costs +from metagpt.utils.cost_manager import CostManager from metagpt.utils.exceptions import handle_exception from metagpt.utils.token_counter import ( count_message_tokens, @@ -55,16 +55,13 @@ class OpenAILLM(BaseLLM): def __init__(self, config: LLMConfig): self.config = config - self._init_model() self._init_client() self.auto_max_tokens = False self.cost_manager: Optional[CostManager] = None - def _init_model(self): - self.model = self.config.model # Used in _calc_usage & _cons_kwargs - def _init_client(self): """https://github.com/openai/openai-python#async-usage""" + self.model = self.config.model # Used in _calc_usage & _cons_kwargs kwargs = self._make_client_kwargs() self.aclient = AsyncOpenAI(**kwargs) @@ -240,16 +237,6 @@ class OpenAILLM(BaseLLM): return usage - @handle_exception - def _update_costs(self, usage: CompletionUsage): - if self.config.calc_usage and usage and self.cost_manager: - self.cost_manager.update_cost(usage.prompt_tokens, usage.completion_tokens, self.model) - - def get_costs(self) -> Costs: - if not self.cost_manager: - return Costs(0, 0, 0, 0) - return self.cost_manager.get_costs() - def _get_max_tokens(self, messages: list[dict]): if not self.auto_max_tokens: return self.config.max_token diff --git a/metagpt/provider/qianfan_api.py b/metagpt/provider/qianfan_api.py index 180935e61..fbbff7085 100644 --- a/metagpt/provider/qianfan_api.py +++ b/metagpt/provider/qianfan_api.py @@ -78,7 +78,7 @@ class QianFanLLM(BaseLLM): # self deployed model on the cloud not to calculate usage, it charges resource pool rental fee self.calc_usage = self.config.calc_usage and self.config.endpoint is None - self.client = qianfan.ChatCompletion() + self.aclient = qianfan.ChatCompletion() def _const_kwargs(self, messages: list[dict], stream: bool = False) -> dict: kwargs = { @@ -110,12 +110,12 @@ class QianFanLLM(BaseLLM): return resp.get("result", "") def completion(self, messages: list[dict]) -> JsonBody: - resp = self.client.do(**self._const_kwargs(messages=messages, stream=False)) + resp = self.aclient.do(**self._const_kwargs(messages=messages, stream=False)) self._update_costs(resp.body.get("usage", {})) return resp.body async def _achat_completion(self, messages: list[dict]) -> JsonBody: - resp = await self.client.ado(**self._const_kwargs(messages=messages, stream=False)) + resp = await self.aclient.ado(**self._const_kwargs(messages=messages, stream=False)) self._update_costs(resp.body.get("usage", {})) return resp.body @@ -123,7 +123,7 @@ class QianFanLLM(BaseLLM): return await self._achat_completion(messages) async def _achat_completion_stream(self, messages: list[dict]) -> str: - resp = await self.client.ado(**self._const_kwargs(messages=messages, stream=True)) + resp = await self.aclient.ado(**self._const_kwargs(messages=messages, stream=True)) collected_content = [] usage = {} async for chunk in resp: diff --git a/metagpt/provider/zhipuai_api.py b/metagpt/provider/zhipuai_api.py index 9108a1fba..b7c160a41 100644 --- a/metagpt/provider/zhipuai_api.py +++ b/metagpt/provider/zhipuai_api.py @@ -57,16 +57,6 @@ class ZhiPuAILLM(BaseLLM): kwargs = {"model": self.model, "messages": messages, "stream": stream, "temperature": 0.3} return kwargs - def _update_costs(self, usage: dict): - """update each request's token cost""" - if self.config.calc_usage: - try: - prompt_tokens = int(usage.get("prompt_tokens", 0)) - completion_tokens = int(usage.get("completion_tokens", 0)) - self.config.cost_manager.update_cost(prompt_tokens, completion_tokens, self.model) - except Exception as e: - logger.error(f"zhipuai updats costs failed! exp: {e}") - def completion(self, messages: list[dict], timeout=3) -> dict: resp = self.llm.chat.completions.create(**self._const_kwargs(messages)) usage = resp.usage.model_dump() From 4370060802b3da936880aefb7aa28a6ba22780cd Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 7 Feb 2024 16:23:54 +0800 Subject: [PATCH 628/637] fix bug --- config/config2.yaml.example | 2 +- metagpt/actions/research.py | 2 +- metagpt/utils/cost_manager.py | 4 ++++ 3 files changed, 6 insertions(+), 2 deletions(-) diff --git a/config/config2.yaml.example b/config/config2.yaml.example index 8f4a33fc1..2217f1b2c 100644 --- a/config/config2.yaml.example +++ b/config/config2.yaml.example @@ -1,5 +1,5 @@ llm: - api_type: "openai" + api_type: "openai" # or azure / ollama etc. base_url: "YOUR_BASE_URL" api_key: "YOUR_API_KEY" model: "gpt-4-turbo-preview" # or gpt-3.5-turbo-1106 / gpt-4-1106-preview diff --git a/metagpt/actions/research.py b/metagpt/actions/research.py index 2ebeadb66..316e9f299 100644 --- a/metagpt/actions/research.py +++ b/metagpt/actions/research.py @@ -133,7 +133,7 @@ class CollectLinks(Action): if len(remove) == 0: break - model_name = config.get_openai_llm().model + model_name = config.model prompt = reduce_message_length(gen_msg(), model_name, system_text, 4096) logger.debug(prompt) queries = await self._aask(prompt, [system_text]) diff --git a/metagpt/utils/cost_manager.py b/metagpt/utils/cost_manager.py index 7bf5154b6..c4c93f91f 100644 --- a/metagpt/utils/cost_manager.py +++ b/metagpt/utils/cost_manager.py @@ -41,6 +41,10 @@ class CostManager(BaseModel): """ self.total_prompt_tokens += prompt_tokens self.total_completion_tokens += completion_tokens + if model not in TOKEN_COSTS: + logger.warning(f"Model {model} not found in TOKEN_COSTS.") + return + cost = ( prompt_tokens * TOKEN_COSTS[model]["prompt"] + completion_tokens * TOKEN_COSTS[model]["completion"] ) / 1000 From c0867643d828084e7503f05ae44987dccf3687d1 Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 7 Feb 2024 16:24:33 +0800 Subject: [PATCH 629/637] fix bug --- metagpt/actions/research.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/actions/research.py b/metagpt/actions/research.py index 316e9f299..ce8d8a967 100644 --- a/metagpt/actions/research.py +++ b/metagpt/actions/research.py @@ -133,7 +133,7 @@ class CollectLinks(Action): if len(remove) == 0: break - model_name = config.model + model_name = config.llm.model prompt = reduce_message_length(gen_msg(), model_name, system_text, 4096) logger.debug(prompt) queries = await self._aask(prompt, [system_text]) From d112371dadf02ee9a828c6708d0bbaa3e600c113 Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 7 Feb 2024 16:33:24 +0800 Subject: [PATCH 630/637] fix bug --- metagpt/utils/text.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/utils/text.py b/metagpt/utils/text.py index dd9678438..921efe706 100644 --- a/metagpt/utils/text.py +++ b/metagpt/utils/text.py @@ -25,7 +25,7 @@ def reduce_message_length( """ max_token = TOKEN_MAX.get(model_name, 2048) - count_string_tokens(system_text, model_name) - reserved for msg in msgs: - if count_string_tokens(msg, model_name) < max_token: + if count_string_tokens(msg, model_name) < max_token or model_name not in TOKEN_MAX: return msg raise RuntimeError("fail to reduce message length") From 50a14718baeacfada5cf7008e2761a801adbd968 Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 7 Feb 2024 16:37:23 +0800 Subject: [PATCH 631/637] refine log --- metagpt/provider/openai_api.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 63e68c9bd..120748d15 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -236,7 +236,7 @@ class OpenAILLM(BaseLLM): usage.prompt_tokens = count_message_tokens(messages, self.model) usage.completion_tokens = count_string_tokens(rsp, self.model) except Exception as e: - logger.error(f"usage calculation failed: {e}") + logger.warning(f"usage calculation failed: {e}") return usage From ce63e455dfe1071a99ee421c1e17df07db20200d Mon Sep 17 00:00:00 2001 From: geekan Date: Wed, 7 Feb 2024 17:03:10 +0800 Subject: [PATCH 632/637] fix bug --- metagpt/provider/openai_api.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 120748d15..756f8c483 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -253,7 +253,7 @@ class OpenAILLM(BaseLLM): def _get_max_tokens(self, messages: list[dict]): if not self.auto_max_tokens: return self.config.max_token - return get_max_completion_tokens(messages, self.model, self.config.max_tokens) + return get_max_completion_tokens(messages, self.model, self.config.max_token) @handle_exception async def amoderation(self, content: Union[str, list[str]]): From dc240a2efd161614f2e4b5090238f72682158ae5 Mon Sep 17 00:00:00 2001 From: better629 Date: Wed, 7 Feb 2024 17:40:27 +0800 Subject: [PATCH 633/637] simplify provider ut code --- .github/workflows/fulltest.yaml | 1 - .github/workflows/unittest.yaml | 2 +- tests/metagpt/provider/mock_llm_config.py | 10 +++ tests/metagpt/provider/req_resp_const.py | 80 +++++++++++++++++++ tests/metagpt/provider/test_anthropic_api.py | 12 +-- tests/metagpt/provider/test_base_llm.py | 53 +++++------- tests/metagpt/provider/test_fireworks_llm.py | 65 ++++----------- .../provider/test_google_gemini_api.py | 37 +++++---- tests/metagpt/provider/test_ollama_api.py | 20 +++-- tests/metagpt/provider/test_open_llm_api.py | 65 +++++---------- tests/metagpt/provider/test_qianfan_api.py | 15 ++++ tests/metagpt/provider/test_spark_api.py | 36 +++++---- tests/metagpt/provider/test_zhipuai_api.py | 33 ++++---- tests/spark.yaml | 7 -- 14 files changed, 235 insertions(+), 201 deletions(-) create mode 100644 tests/metagpt/provider/req_resp_const.py create mode 100644 tests/metagpt/provider/test_qianfan_api.py delete mode 100644 tests/spark.yaml diff --git a/.github/workflows/fulltest.yaml b/.github/workflows/fulltest.yaml index f5c6049e1..70c800481 100644 --- a/.github/workflows/fulltest.yaml +++ b/.github/workflows/fulltest.yaml @@ -54,7 +54,6 @@ jobs: export ALLOW_OPENAI_API_CALL=0 echo "${{ secrets.METAGPT_KEY_YAML }}" | base64 -d > config/key.yaml mkdir -p ~/.metagpt && echo "${{ secrets.METAGPT_CONFIG2_YAML }}" | base64 -d > ~/.metagpt/config2.yaml - echo "${{ secrets.SPARK_YAML }}" | base64 -d > ~/.metagpt/spark.yaml pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt - name: Show coverage report run: | diff --git a/.github/workflows/unittest.yaml b/.github/workflows/unittest.yaml index 2e7e3ce2b..afa9faba7 100644 --- a/.github/workflows/unittest.yaml +++ b/.github/workflows/unittest.yaml @@ -31,7 +31,7 @@ jobs: - name: Test with pytest run: | export ALLOW_OPENAI_API_CALL=0 - mkdir -p ~/.metagpt && cp tests/config2.yaml ~/.metagpt/config2.yaml && cp tests/spark.yaml ~/.metagpt/spark.yaml + mkdir -p ~/.metagpt && cp tests/config2.yaml ~/.metagpt/config2.yaml pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt - name: Show coverage report run: | diff --git a/tests/metagpt/provider/mock_llm_config.py b/tests/metagpt/provider/mock_llm_config.py index e2f626a6a..21780f914 100644 --- a/tests/metagpt/provider/mock_llm_config.py +++ b/tests/metagpt/provider/mock_llm_config.py @@ -42,3 +42,13 @@ mock_llm_config_zhipu = LLMConfig( model="mock_zhipu_model", proxy="http://localhost:8080", ) + + +mock_llm_config_spark = LLMConfig( + api_type="spark", + app_id="xxx", + api_key="xxx", + api_secret="xxx", + domain="generalv2", + base_url="wss://spark-api.xf-yun.com/v3.1/chat", +) diff --git a/tests/metagpt/provider/req_resp_const.py b/tests/metagpt/provider/req_resp_const.py new file mode 100644 index 000000000..a3a7a363c --- /dev/null +++ b/tests/metagpt/provider/req_resp_const.py @@ -0,0 +1,80 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : default request & response data for provider unittest + +from openai.types.chat.chat_completion import ( + ChatCompletion, + ChatCompletionMessage, + Choice, +) +from openai.types.chat.chat_completion_chunk import ChatCompletionChunk +from openai.types.chat.chat_completion_chunk import Choice as AChoice +from openai.types.chat.chat_completion_chunk import ChoiceDelta +from openai.types.completion_usage import CompletionUsage + +prompt = "who are you?" +messages = [{"role": "user", "content": prompt}] + +resp_cont_tmpl = "I'm {name}" +default_resp_cont = resp_cont_tmpl.format(name="GPT") + + +# part of whole ChatCompletion of openai like structure +def get_part_chat_completion(llm_name: str) -> dict: + part_chat_completion = { + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": resp_cont_tmpl.format(name=llm_name), + }, + "finish_reason": "stop", + } + ], + "usage": {"completion_tokens": 22, "prompt_tokens": 19, "total_tokens": 41}, + } + return part_chat_completion + + +def get_openai_chat_completion(llm_name: str) -> ChatCompletion: + openai_chat_completion = ChatCompletion( + id="cmpl-a6652c1bb181caae8dd19ad8", + model="xx/xxx", + object="chat.completion", + created=1703300855, + choices=[ + Choice( + finish_reason="stop", + index=0, + message=ChatCompletionMessage(role="assistant", content=resp_cont_tmpl.format(name=llm_name)), + logprobs=None, + ) + ], + usage=CompletionUsage(completion_tokens=110, prompt_tokens=92, total_tokens=202), + ) + return openai_chat_completion + + +def get_openai_chat_completion_chunk(llm_name: str, usage_as_dict: bool = False) -> ChatCompletionChunk: + usage = CompletionUsage(completion_tokens=110, prompt_tokens=92, total_tokens=202) + usage = usage if not usage_as_dict else usage.model_dump() + openai_chat_completion_chunk = ChatCompletionChunk( + id="cmpl-a6652c1bb181caae8dd19ad8", + model="xx/xxx", + object="chat.completion.chunk", + created=1703300855, + choices=[ + AChoice( + delta=ChoiceDelta(role="assistant", content=resp_cont_tmpl.format(name=llm_name)), + finish_reason="stop", + index=0, + logprobs=None, + ) + ], + usage=usage, + ) + return openai_chat_completion_chunk + + +gemini_messages = [{"role": "user", "parts": prompt}] diff --git a/tests/metagpt/provider/test_anthropic_api.py b/tests/metagpt/provider/test_anthropic_api.py index 6962ab064..93cfd7dbc 100644 --- a/tests/metagpt/provider/test_anthropic_api.py +++ b/tests/metagpt/provider/test_anthropic_api.py @@ -8,25 +8,25 @@ from anthropic.resources.completions import Completion from metagpt.provider.anthropic_api import Claude2 from tests.metagpt.provider.mock_llm_config import mock_llm_config +from tests.metagpt.provider.req_resp_const import prompt, resp_cont_tmpl -prompt = "who are you" -resp = "I'am Claude2" +resp_cont = resp_cont_tmpl.format(name="Claude") def mock_anthropic_completions_create(self, model: str, prompt: str, max_tokens_to_sample: int) -> Completion: - return Completion(id="xx", completion=resp, model="claude-2", stop_reason="stop_sequence", type="completion") + return Completion(id="xx", completion=resp_cont, model="claude-2", stop_reason="stop_sequence", type="completion") async def mock_anthropic_acompletions_create(self, model: str, prompt: str, max_tokens_to_sample: int) -> Completion: - return Completion(id="xx", completion=resp, model="claude-2", stop_reason="stop_sequence", type="completion") + return Completion(id="xx", completion=resp_cont, model="claude-2", stop_reason="stop_sequence", type="completion") def test_claude2_ask(mocker): mocker.patch("anthropic.resources.completions.Completions.create", mock_anthropic_completions_create) - assert resp == Claude2(mock_llm_config).ask(prompt) + assert resp_cont == Claude2(mock_llm_config).ask(prompt) @pytest.mark.asyncio async def test_claude2_aask(mocker): mocker.patch("anthropic.resources.completions.AsyncCompletions.create", mock_anthropic_acompletions_create) - assert resp == await Claude2(mock_llm_config).aask(prompt) + assert resp_cont == await Claude2(mock_llm_config).aask(prompt) diff --git a/tests/metagpt/provider/test_base_llm.py b/tests/metagpt/provider/test_base_llm.py index cc781f78a..0babd6d5f 100644 --- a/tests/metagpt/provider/test_base_llm.py +++ b/tests/metagpt/provider/test_base_llm.py @@ -11,21 +11,13 @@ import pytest from metagpt.configs.llm_config import LLMConfig from metagpt.provider.base_llm import BaseLLM from metagpt.schema import Message +from tests.metagpt.provider.req_resp_const import ( + default_resp_cont, + get_part_chat_completion, + prompt, +) -default_chat_resp = { - "choices": [ - { - "index": 0, - "message": { - "role": "assistant", - "content": "I'am GPT", - }, - "finish_reason": "stop", - } - ] -} -prompt_msg = "who are you" -resp_content = default_chat_resp["choices"][0]["message"]["content"] +llm_name = "GPT" class MockBaseLLM(BaseLLM): @@ -33,16 +25,13 @@ class MockBaseLLM(BaseLLM): pass def completion(self, messages: list[dict], timeout=3): - return default_chat_resp + return get_part_chat_completion(llm_name) async def acompletion(self, messages: list[dict], timeout=3): - return default_chat_resp + return get_part_chat_completion(llm_name) async def acompletion_text(self, messages: list[dict], stream=False, timeout=3) -> str: - return resp_content - - async def close(self): - return default_chat_resp + return default_resp_cont def test_base_llm(): @@ -86,25 +75,25 @@ def test_base_llm(): choice_text = base_llm.get_choice_text(openai_funccall_resp) assert choice_text == openai_funccall_resp["choices"][0]["message"]["content"] - # resp = base_llm.ask(prompt_msg) - # assert resp == resp_content + # resp = base_llm.ask(prompt) + # assert resp == default_resp_cont - # resp = base_llm.ask_batch([prompt_msg]) - # assert resp == resp_content + # resp = base_llm.ask_batch([prompt]) + # assert resp == default_resp_cont - # resp = base_llm.ask_code([prompt_msg]) - # assert resp == resp_content + # resp = base_llm.ask_code([prompt]) + # assert resp == default_resp_cont @pytest.mark.asyncio async def test_async_base_llm(): base_llm = MockBaseLLM() - resp = await base_llm.aask(prompt_msg) - assert resp == resp_content + resp = await base_llm.aask(prompt) + assert resp == default_resp_cont - resp = await base_llm.aask_batch([prompt_msg]) - assert resp == resp_content + resp = await base_llm.aask_batch([prompt]) + assert resp == default_resp_cont - # resp = await base_llm.aask_code([prompt_msg]) - # assert resp == resp_content + # resp = await base_llm.aask_code([prompt]) + # assert resp == default_resp_cont diff --git a/tests/metagpt/provider/test_fireworks_llm.py b/tests/metagpt/provider/test_fireworks_llm.py index 66b55e5b2..834f6305f 100644 --- a/tests/metagpt/provider/test_fireworks_llm.py +++ b/tests/metagpt/provider/test_fireworks_llm.py @@ -3,14 +3,7 @@ # @Desc : the unittest of fireworks api import pytest -from openai.types.chat.chat_completion import ( - ChatCompletion, - ChatCompletionMessage, - Choice, -) from openai.types.chat.chat_completion_chunk import ChatCompletionChunk -from openai.types.chat.chat_completion_chunk import Choice as AChoice -from openai.types.chat.chat_completion_chunk import ChoiceDelta from openai.types.completion_usage import CompletionUsage from metagpt.provider.fireworks_api import ( @@ -20,42 +13,18 @@ from metagpt.provider.fireworks_api import ( ) from metagpt.utils.cost_manager import Costs from tests.metagpt.provider.mock_llm_config import mock_llm_config - -resp_content = "I'm fireworks" -default_resp = ChatCompletion( - id="cmpl-a6652c1bb181caae8dd19ad8", - model="accounts/fireworks/models/llama-v2-13b-chat", - object="chat.completion", - created=1703300855, - choices=[ - Choice( - finish_reason="stop", - index=0, - message=ChatCompletionMessage(role="assistant", content=resp_content), - logprobs=None, - ) - ], - usage=CompletionUsage(completion_tokens=110, prompt_tokens=92, total_tokens=202), +from tests.metagpt.provider.req_resp_const import ( + get_openai_chat_completion, + get_openai_chat_completion_chunk, + messages, + prompt, + resp_cont_tmpl, ) -default_resp_chunk = ChatCompletionChunk( - id=default_resp.id, - model=default_resp.model, - object="chat.completion.chunk", - created=default_resp.created, - choices=[ - AChoice( - delta=ChoiceDelta(content=resp_content, role="assistant"), - finish_reason="stop", - index=0, - logprobs=None, - ) - ], - usage=dict(default_resp.usage), -) - -prompt_msg = "who are you" -messages = [{"role": "user", "content": prompt_msg}] +llm_name = "fireworks" +resp_cont = resp_cont_tmpl.format(name=llm_name) +default_resp = get_openai_chat_completion(llm_name) +default_resp_chunk = get_openai_chat_completion_chunk(llm_name, usage_as_dict=True) def test_fireworks_costmanager(): @@ -99,16 +68,16 @@ async def test_fireworks_acompletion(mocker): ) resp = await fireworks_gpt.acompletion(messages) - assert resp.choices[0].message.content in resp_content + assert resp.choices[0].message.content in resp_cont - resp = await fireworks_gpt.aask(prompt_msg, stream=False) - assert resp == resp_content + resp = await fireworks_gpt.aask(prompt, stream=False) + assert resp == resp_cont resp = await fireworks_gpt.acompletion_text(messages, stream=False) - assert resp == resp_content + assert resp == resp_cont resp = await fireworks_gpt.acompletion_text(messages, stream=True) - assert resp == resp_content + assert resp == resp_cont - resp = await fireworks_gpt.aask(prompt_msg) - assert resp == resp_content + resp = await fireworks_gpt.aask(prompt) + assert resp == resp_cont diff --git a/tests/metagpt/provider/test_google_gemini_api.py b/tests/metagpt/provider/test_google_gemini_api.py index 404ae1e90..ad0c7bbfe 100644 --- a/tests/metagpt/provider/test_google_gemini_api.py +++ b/tests/metagpt/provider/test_google_gemini_api.py @@ -11,6 +11,11 @@ from google.generativeai.types import content_types from metagpt.provider.google_gemini_api import GeminiLLM from tests.metagpt.provider.mock_llm_config import mock_llm_config +from tests.metagpt.provider.req_resp_const import ( + gemini_messages, + prompt, + resp_cont_tmpl, +) @dataclass @@ -18,10 +23,8 @@ class MockGeminiResponse(ABC): text: str -prompt_msg = "who are you" -messages = [{"role": "user", "parts": prompt_msg}] -resp_content = "I'm gemini from google" -default_resp = MockGeminiResponse(text=resp_content) +resp_cont = resp_cont_tmpl.format(name="gemini") +default_resp = MockGeminiResponse(text=resp_cont) def mock_gemini_count_tokens(self, contents: content_types.ContentsType) -> glm.CountTokensResponse: @@ -62,26 +65,26 @@ async def test_gemini_acompletion(mocker): gemini_gpt = GeminiLLM(mock_llm_config) - assert gemini_gpt._user_msg(prompt_msg) == {"role": "user", "parts": [prompt_msg]} - assert gemini_gpt._assistant_msg(prompt_msg) == {"role": "model", "parts": [prompt_msg]} + assert gemini_gpt._user_msg(prompt) == {"role": "user", "parts": [prompt]} + assert gemini_gpt._assistant_msg(prompt) == {"role": "model", "parts": [prompt]} - usage = gemini_gpt.get_usage(messages, resp_content) + usage = gemini_gpt.get_usage(gemini_messages, resp_cont) assert usage == {"prompt_tokens": 20, "completion_tokens": 20} - resp = gemini_gpt.completion(messages) + resp = gemini_gpt.completion(gemini_messages) assert resp == default_resp - resp = await gemini_gpt.acompletion(messages) + resp = await gemini_gpt.acompletion(gemini_messages) assert resp.text == default_resp.text - resp = await gemini_gpt.aask(prompt_msg, stream=False) - assert resp == resp_content + resp = await gemini_gpt.aask(prompt, stream=False) + assert resp == resp_cont - resp = await gemini_gpt.acompletion_text(messages, stream=False) - assert resp == resp_content + resp = await gemini_gpt.acompletion_text(gemini_messages, stream=False) + assert resp == resp_cont - resp = await gemini_gpt.acompletion_text(messages, stream=True) - assert resp == resp_content + resp = await gemini_gpt.acompletion_text(gemini_messages, stream=True) + assert resp == resp_cont - resp = await gemini_gpt.aask(prompt_msg) - assert resp == resp_content + resp = await gemini_gpt.aask(prompt) + assert resp == resp_cont diff --git a/tests/metagpt/provider/test_ollama_api.py b/tests/metagpt/provider/test_ollama_api.py index 5d942598b..8e2625e35 100644 --- a/tests/metagpt/provider/test_ollama_api.py +++ b/tests/metagpt/provider/test_ollama_api.py @@ -9,12 +9,10 @@ import pytest from metagpt.provider.ollama_api import OllamaLLM from tests.metagpt.provider.mock_llm_config import mock_llm_config +from tests.metagpt.provider.req_resp_const import messages, prompt, resp_cont_tmpl -prompt_msg = "who are you" -messages = [{"role": "user", "content": prompt_msg}] - -resp_content = "I'm ollama" -default_resp = {"message": {"role": "assistant", "content": resp_content}} +resp_cont = resp_cont_tmpl.format(name="ollama") +default_resp = {"message": {"role": "assistant", "content": resp_cont}} async def mock_ollama_arequest(self, stream: bool = False, **kwargs) -> Tuple[Any, Any, bool]: @@ -46,14 +44,14 @@ async def test_gemini_acompletion(mocker): resp = await ollama_gpt.acompletion(messages) assert resp["message"]["content"] == default_resp["message"]["content"] - resp = await ollama_gpt.aask(prompt_msg, stream=False) - assert resp == resp_content + resp = await ollama_gpt.aask(prompt, stream=False) + assert resp == resp_cont resp = await ollama_gpt.acompletion_text(messages, stream=False) - assert resp == resp_content + assert resp == resp_cont resp = await ollama_gpt.acompletion_text(messages, stream=True) - assert resp == resp_content + assert resp == resp_cont - resp = await ollama_gpt.aask(prompt_msg) - assert resp == resp_content + resp = await ollama_gpt.aask(prompt) + assert resp == resp_cont diff --git a/tests/metagpt/provider/test_open_llm_api.py b/tests/metagpt/provider/test_open_llm_api.py index fc7b510cc..5b8a506e9 100644 --- a/tests/metagpt/provider/test_open_llm_api.py +++ b/tests/metagpt/provider/test_open_llm_api.py @@ -3,53 +3,25 @@ # @Desc : import pytest -from openai.types.chat.chat_completion import ( - ChatCompletion, - ChatCompletionMessage, - Choice, -) from openai.types.chat.chat_completion_chunk import ChatCompletionChunk -from openai.types.chat.chat_completion_chunk import Choice as AChoice -from openai.types.chat.chat_completion_chunk import ChoiceDelta from openai.types.completion_usage import CompletionUsage from metagpt.provider.open_llm_api import OpenLLM -from metagpt.utils.cost_manager import Costs +from metagpt.utils.cost_manager import CostManager, Costs from tests.metagpt.provider.mock_llm_config import mock_llm_config - -resp_content = "I'm llama2" -default_resp = ChatCompletion( - id="cmpl-a6652c1bb181caae8dd19ad8", - model="llama-v2-13b-chat", - object="chat.completion", - created=1703302755, - choices=[ - Choice( - finish_reason="stop", - index=0, - message=ChatCompletionMessage(role="assistant", content=resp_content), - logprobs=None, - ) - ], +from tests.metagpt.provider.req_resp_const import ( + get_openai_chat_completion, + get_openai_chat_completion_chunk, + messages, + prompt, + resp_cont_tmpl, ) -default_resp_chunk = ChatCompletionChunk( - id=default_resp.id, - model=default_resp.model, - object="chat.completion.chunk", - created=default_resp.created, - choices=[ - AChoice( - delta=ChoiceDelta(content=resp_content, role="assistant"), - finish_reason="stop", - index=0, - logprobs=None, - ) - ], -) +llm_name = "llama2-7b" +resp_cont = resp_cont_tmpl.format(name=llm_name) +default_resp = get_openai_chat_completion(llm_name) -prompt_msg = "who are you" -messages = [{"role": "user", "content": prompt_msg}] +default_resp_chunk = get_openai_chat_completion_chunk(llm_name) async def mock_openai_acompletions_create(self, stream: bool = False, **kwargs) -> ChatCompletionChunk: @@ -71,22 +43,23 @@ async def test_openllm_acompletion(mocker): openllm_gpt = OpenLLM(mock_llm_config) openllm_gpt.model = "llama-v2-13b-chat" + openllm_gpt.cost_manager = CostManager() openllm_gpt._update_costs(usage=CompletionUsage(prompt_tokens=100, completion_tokens=100, total_tokens=200)) assert openllm_gpt.get_costs() == Costs( total_prompt_tokens=100, total_completion_tokens=100, total_cost=0, total_budget=0 ) resp = await openllm_gpt.acompletion(messages) - assert resp.choices[0].message.content in resp_content + assert resp.choices[0].message.content in resp_cont - resp = await openllm_gpt.aask(prompt_msg, stream=False) - assert resp == resp_content + resp = await openllm_gpt.aask(prompt, stream=False) + assert resp == resp_cont resp = await openllm_gpt.acompletion_text(messages, stream=False) - assert resp == resp_content + assert resp == resp_cont resp = await openllm_gpt.acompletion_text(messages, stream=True) - assert resp == resp_content + assert resp == resp_cont - resp = await openllm_gpt.aask(prompt_msg) - assert resp == resp_content + resp = await openllm_gpt.aask(prompt) + assert resp == resp_cont diff --git a/tests/metagpt/provider/test_qianfan_api.py b/tests/metagpt/provider/test_qianfan_api.py new file mode 100644 index 000000000..76271b1e8 --- /dev/null +++ b/tests/metagpt/provider/test_qianfan_api.py @@ -0,0 +1,15 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# @Desc : the unittest of qianfan api + +import pytest + +from metagpt.provider.qianfan_api import QianFanLLM +from tests.metagpt.provider.req_resp_const import prompt, messages, resp_cont_tmpl + + +resp_cont = resp_cont_tmpl.format(name="ERNIE-Bot-turbo") + + +def test_qianfan_acompletion(mocker): + assert True, True diff --git a/tests/metagpt/provider/test_spark_api.py b/tests/metagpt/provider/test_spark_api.py index f5a6f66fd..32a839393 100644 --- a/tests/metagpt/provider/test_spark_api.py +++ b/tests/metagpt/provider/test_spark_api.py @@ -4,12 +4,14 @@ import pytest -from metagpt.config2 import Config from metagpt.provider.spark_api import GetMessageFromWeb, SparkLLM -from tests.metagpt.provider.mock_llm_config import mock_llm_config +from tests.metagpt.provider.mock_llm_config import ( + mock_llm_config, + mock_llm_config_spark, +) +from tests.metagpt.provider.req_resp_const import prompt, resp_cont_tmpl -prompt_msg = "who are you" -resp_content = "I'm Spark" +resp_cont = resp_cont_tmpl.format(name="Spark") class MockWebSocketApp(object): @@ -23,7 +25,7 @@ class MockWebSocketApp(object): def test_get_msg_from_web(mocker): mocker.patch("websocket.WebSocketApp", MockWebSocketApp) - get_msg_from_web = GetMessageFromWeb(prompt_msg, mock_llm_config) + get_msg_from_web = GetMessageFromWeb(prompt, mock_llm_config) assert get_msg_from_web.gen_params()["parameter"]["chat"]["domain"] == "mock_domain" ret = get_msg_from_web.run() @@ -31,15 +33,17 @@ def test_get_msg_from_web(mocker): def mock_spark_get_msg_from_web_run(self) -> str: - return resp_content + return resp_cont @pytest.mark.asyncio -async def test_spark_aask(): - llm = SparkLLM(Config.from_home("spark.yaml").llm) +async def test_spark_aask(mocker): + mocker.patch("metagpt.provider.spark_api.GetMessageFromWeb.run", mock_spark_get_msg_from_web_run) + + llm = SparkLLM(mock_llm_config_spark) resp = await llm.aask("Hello!") - print(resp) + assert resp == resp_cont @pytest.mark.asyncio @@ -49,16 +53,16 @@ async def test_spark_acompletion(mocker): spark_gpt = SparkLLM(mock_llm_config) resp = await spark_gpt.acompletion([]) - assert resp == resp_content + assert resp == resp_cont - resp = await spark_gpt.aask(prompt_msg, stream=False) - assert resp == resp_content + resp = await spark_gpt.aask(prompt, stream=False) + assert resp == resp_cont resp = await spark_gpt.acompletion_text([], stream=False) - assert resp == resp_content + assert resp == resp_cont resp = await spark_gpt.acompletion_text([], stream=True) - assert resp == resp_content + assert resp == resp_cont - resp = await spark_gpt.aask(prompt_msg) - assert resp == resp_content + resp = await spark_gpt.aask(prompt) + assert resp == resp_cont diff --git a/tests/metagpt/provider/test_zhipuai_api.py b/tests/metagpt/provider/test_zhipuai_api.py index 798209710..064562bff 100644 --- a/tests/metagpt/provider/test_zhipuai_api.py +++ b/tests/metagpt/provider/test_zhipuai_api.py @@ -6,22 +6,23 @@ import pytest from metagpt.provider.zhipuai_api import ZhiPuAILLM from tests.metagpt.provider.mock_llm_config import mock_llm_config_zhipu +from tests.metagpt.provider.req_resp_const import ( + get_part_chat_completion, + messages, + prompt, + resp_cont_tmpl, +) -prompt_msg = "who are you" -messages = [{"role": "user", "content": prompt_msg}] - -resp_content = "I'm chatglm-turbo" -default_resp = { - "choices": [{"finish_reason": "stop", "index": 0, "message": {"content": resp_content, "role": "assistant"}}], - "usage": {"completion_tokens": 22, "prompt_tokens": 19, "total_tokens": 41}, -} +llm_name = "ChatGLM-4" +resp_cont = resp_cont_tmpl.format(name=llm_name) +default_resp = get_part_chat_completion(llm_name) async def mock_zhipuai_acreate_stream(**kwargs): class MockResponse(object): async def _aread(self): class Iterator(object): - events = [{"choices": [{"index": 0, "delta": {"content": resp_content, "role": "assistant"}}]}] + events = [{"choices": [{"index": 0, "delta": {"content": resp_cont, "role": "assistant"}}]}] async def __aiter__(self): for event in self.events: @@ -49,19 +50,19 @@ async def test_zhipuai_acompletion(mocker): zhipu_gpt = ZhiPuAILLM(mock_llm_config_zhipu) resp = await zhipu_gpt.acompletion(messages) - assert resp["choices"][0]["message"]["content"] == resp_content + assert resp["choices"][0]["message"]["content"] == resp_cont - resp = await zhipu_gpt.aask(prompt_msg, stream=False) - assert resp == resp_content + resp = await zhipu_gpt.aask(prompt, stream=False) + assert resp == resp_cont resp = await zhipu_gpt.acompletion_text(messages, stream=False) - assert resp == resp_content + assert resp == resp_cont resp = await zhipu_gpt.acompletion_text(messages, stream=True) - assert resp == resp_content + assert resp == resp_cont - resp = await zhipu_gpt.aask(prompt_msg) - assert resp == resp_content + resp = await zhipu_gpt.aask(prompt) + assert resp == resp_cont def test_zhipuai_proxy(): diff --git a/tests/spark.yaml b/tests/spark.yaml deleted file mode 100644 index a5bbd98bd..000000000 --- a/tests/spark.yaml +++ /dev/null @@ -1,7 +0,0 @@ -llm: - api_type: "spark" - app_id: "xxx" - api_key: "xxx" - api_secret: "xxx" - domain: "generalv2" - base_url: "wss://spark-api.xf-yun.com/v3.1/chat" \ No newline at end of file From d94f4fbfbc3bd4310669f06e9bac9a7c89001712 Mon Sep 17 00:00:00 2001 From: shenchucheng Date: Wed, 7 Feb 2024 17:44:36 +0800 Subject: [PATCH 634/637] fix research bugs --- metagpt/provider/openai_api.py | 4 +++- metagpt/utils/text.py | 2 +- tests/metagpt/utils/test_text.py | 1 + 3 files changed, 5 insertions(+), 2 deletions(-) diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 63e68c9bd..7b2cd6220 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -253,7 +253,9 @@ class OpenAILLM(BaseLLM): def _get_max_tokens(self, messages: list[dict]): if not self.auto_max_tokens: return self.config.max_token - return get_max_completion_tokens(messages, self.model, self.config.max_tokens) + # FIXME + # https://community.openai.com/t/why-is-gpt-3-5-turbo-1106-max-tokens-limited-to-4096/494973/3 + return min(get_max_completion_tokens(messages, self.model, self.config.max_tokens), 4096) @handle_exception async def amoderation(self, content: Union[str, list[str]]): diff --git a/metagpt/utils/text.py b/metagpt/utils/text.py index 921efe706..fb8b94232 100644 --- a/metagpt/utils/text.py +++ b/metagpt/utils/text.py @@ -93,7 +93,7 @@ def split_paragraph(paragraph: str, sep: str = ".,", count: int = 2) -> list[str continue ret = ["".join(j) for j in _split_by_count(sentences, count)] return ret - return _split_by_count(paragraph, count) + return list(_split_by_count(paragraph, count)) def decode_unicode_escape(text: str) -> str: diff --git a/tests/metagpt/utils/test_text.py b/tests/metagpt/utils/test_text.py index 7003c7767..c9a9753be 100644 --- a/tests/metagpt/utils/test_text.py +++ b/tests/metagpt/utils/test_text.py @@ -42,6 +42,7 @@ def test_reduce_message_length(msgs, model_name, system_text, reserved, expected (" ".join("Hello World." for _ in range(1000)), "Prompt: {}", "gpt-3.5-turbo-16k", "System", 3000, 1), (" ".join("Hello World." for _ in range(4000)), "Prompt: {}", "gpt-4", "System", 2000, 2), (" ".join("Hello World." for _ in range(8000)), "Prompt: {}", "gpt-4-32k", "System", 4000, 1), + (" ".join("Hello World" for _ in range(8000)), "Prompt: {}", "gpt-3.5-turbo", "System", 1000, 8), ], ) def test_generate_prompt_chunk(text, prompt_template, model_name, system_text, reserved, expected): From d3f6e38e8a9805d6bc80e5489dde99007bd20b6d Mon Sep 17 00:00:00 2001 From: better629 Date: Wed, 7 Feb 2024 18:32:32 +0800 Subject: [PATCH 635/637] add qianfan ut code and update xx_llm from xx_gpt --- metagpt/provider/qianfan_api.py | 3 +- tests/metagpt/provider/mock_llm_config.py | 7 +++ tests/metagpt/provider/req_resp_const.py | 57 ++++++++++++++++--- tests/metagpt/provider/test_base_llm.py | 6 +- tests/metagpt/provider/test_fireworks_llm.py | 26 ++++----- .../provider/test_google_gemini_api.py | 20 +++---- tests/metagpt/provider/test_ollama_api.py | 12 ++-- tests/metagpt/provider/test_open_llm_api.py | 28 ++++----- tests/metagpt/provider/test_qianfan_api.py | 39 +++++++++++-- tests/metagpt/provider/test_spark_api.py | 12 ++-- tests/metagpt/provider/test_zhipuai_api.py | 23 +++----- 11 files changed, 153 insertions(+), 80 deletions(-) diff --git a/metagpt/provider/qianfan_api.py b/metagpt/provider/qianfan_api.py index fbbff7085..6f94b9cea 100644 --- a/metagpt/provider/qianfan_api.py +++ b/metagpt/provider/qianfan_api.py @@ -5,6 +5,7 @@ import copy import os import qianfan +from qianfan import ChatCompletion from qianfan.resources.typing import JsonBody from tenacity import ( after_log, @@ -78,7 +79,7 @@ class QianFanLLM(BaseLLM): # self deployed model on the cloud not to calculate usage, it charges resource pool rental fee self.calc_usage = self.config.calc_usage and self.config.endpoint is None - self.aclient = qianfan.ChatCompletion() + self.aclient: ChatCompletion = qianfan.ChatCompletion() def _const_kwargs(self, messages: list[dict], stream: bool = False) -> dict: kwargs = { diff --git a/tests/metagpt/provider/mock_llm_config.py b/tests/metagpt/provider/mock_llm_config.py index 21780f914..e61e32e8b 100644 --- a/tests/metagpt/provider/mock_llm_config.py +++ b/tests/metagpt/provider/mock_llm_config.py @@ -52,3 +52,10 @@ mock_llm_config_spark = LLMConfig( domain="generalv2", base_url="wss://spark-api.xf-yun.com/v3.1/chat", ) + +mock_llm_config_qianfan = LLMConfig( + api_type="qianfan", + access_key="xxx", + secret_key="xxx", + model="ERNIE-Bot-turbo" +) diff --git a/tests/metagpt/provider/req_resp_const.py b/tests/metagpt/provider/req_resp_const.py index a3a7a363c..20d8e0914 100644 --- a/tests/metagpt/provider/req_resp_const.py +++ b/tests/metagpt/provider/req_resp_const.py @@ -2,6 +2,7 @@ # -*- coding: utf-8 -*- # @Desc : default request & response data for provider unittest +from typing import Dict from openai.types.chat.chat_completion import ( ChatCompletion, ChatCompletionMessage, @@ -11,6 +12,9 @@ from openai.types.chat.chat_completion_chunk import ChatCompletionChunk from openai.types.chat.chat_completion_chunk import Choice as AChoice from openai.types.chat.chat_completion_chunk import ChoiceDelta from openai.types.completion_usage import CompletionUsage +from qianfan.resources.typing import QfResponse, default_field + +from metagpt.provider.base_llm import BaseLLM prompt = "who are you?" messages = [{"role": "user", "content": prompt}] @@ -20,14 +24,14 @@ default_resp_cont = resp_cont_tmpl.format(name="GPT") # part of whole ChatCompletion of openai like structure -def get_part_chat_completion(llm_name: str) -> dict: +def get_part_chat_completion(name: str) -> dict: part_chat_completion = { "choices": [ { "index": 0, "message": { "role": "assistant", - "content": resp_cont_tmpl.format(name=llm_name), + "content": resp_cont_tmpl.format(name=name), }, "finish_reason": "stop", } @@ -37,7 +41,7 @@ def get_part_chat_completion(llm_name: str) -> dict: return part_chat_completion -def get_openai_chat_completion(llm_name: str) -> ChatCompletion: +def get_openai_chat_completion(name: str) -> ChatCompletion: openai_chat_completion = ChatCompletion( id="cmpl-a6652c1bb181caae8dd19ad8", model="xx/xxx", @@ -47,7 +51,7 @@ def get_openai_chat_completion(llm_name: str) -> ChatCompletion: Choice( finish_reason="stop", index=0, - message=ChatCompletionMessage(role="assistant", content=resp_cont_tmpl.format(name=llm_name)), + message=ChatCompletionMessage(role="assistant", content=resp_cont_tmpl.format(name=name)), logprobs=None, ) ], @@ -56,7 +60,7 @@ def get_openai_chat_completion(llm_name: str) -> ChatCompletion: return openai_chat_completion -def get_openai_chat_completion_chunk(llm_name: str, usage_as_dict: bool = False) -> ChatCompletionChunk: +def get_openai_chat_completion_chunk(name: str, usage_as_dict: bool = False) -> ChatCompletionChunk: usage = CompletionUsage(completion_tokens=110, prompt_tokens=92, total_tokens=202) usage = usage if not usage_as_dict else usage.model_dump() openai_chat_completion_chunk = ChatCompletionChunk( @@ -66,7 +70,7 @@ def get_openai_chat_completion_chunk(llm_name: str, usage_as_dict: bool = False) created=1703300855, choices=[ AChoice( - delta=ChoiceDelta(role="assistant", content=resp_cont_tmpl.format(name=llm_name)), + delta=ChoiceDelta(role="assistant", content=resp_cont_tmpl.format(name=name)), finish_reason="stop", index=0, logprobs=None, @@ -76,5 +80,44 @@ def get_openai_chat_completion_chunk(llm_name: str, usage_as_dict: bool = False) ) return openai_chat_completion_chunk - +# For gemini gemini_messages = [{"role": "user", "parts": prompt}] + + +# For QianFan +qf_jsonbody_dict = { + "id": "as-4v1h587fyv", + "object": "chat.completion", + "created": 1695021339, + "result": "", + "is_truncated": False, + "need_clear_history": False, + "usage": { + "prompt_tokens": 7, + "completion_tokens": 15, + "total_tokens": 22 + } +} + + +def get_qianfan_response(name: str) -> QfResponse: + qf_jsonbody_dict["result"] = resp_cont_tmpl.format(name=name) + return QfResponse( + code=200, + body=qf_jsonbody_dict + ) + + +# For llm general chat functions call +async def llm_general_chat_funcs_test(llm: BaseLLM, prompt: str, messages: list[dict], resp_cont: str): + resp = await llm.aask(prompt, stream=False) + assert resp == resp_cont + + resp = await llm.aask(prompt) + assert resp == resp_cont + + resp = await llm.acompletion_text(messages, stream=False) + assert resp == resp_cont + + resp = await llm.acompletion_text(messages, stream=True) + assert resp == resp_cont diff --git a/tests/metagpt/provider/test_base_llm.py b/tests/metagpt/provider/test_base_llm.py index 0babd6d5f..cf44343bc 100644 --- a/tests/metagpt/provider/test_base_llm.py +++ b/tests/metagpt/provider/test_base_llm.py @@ -17,7 +17,7 @@ from tests.metagpt.provider.req_resp_const import ( prompt, ) -llm_name = "GPT" +name = "GPT" class MockBaseLLM(BaseLLM): @@ -25,10 +25,10 @@ class MockBaseLLM(BaseLLM): pass def completion(self, messages: list[dict], timeout=3): - return get_part_chat_completion(llm_name) + return get_part_chat_completion(name) async def acompletion(self, messages: list[dict], timeout=3): - return get_part_chat_completion(llm_name) + return get_part_chat_completion(name) async def acompletion_text(self, messages: list[dict], stream=False, timeout=3) -> str: return default_resp_cont diff --git a/tests/metagpt/provider/test_fireworks_llm.py b/tests/metagpt/provider/test_fireworks_llm.py index 834f6305f..e28f7500b 100644 --- a/tests/metagpt/provider/test_fireworks_llm.py +++ b/tests/metagpt/provider/test_fireworks_llm.py @@ -21,10 +21,10 @@ from tests.metagpt.provider.req_resp_const import ( resp_cont_tmpl, ) -llm_name = "fireworks" -resp_cont = resp_cont_tmpl.format(name=llm_name) -default_resp = get_openai_chat_completion(llm_name) -default_resp_chunk = get_openai_chat_completion_chunk(llm_name, usage_as_dict=True) +name = "fireworks" +resp_cont = resp_cont_tmpl.format(name=name) +default_resp = get_openai_chat_completion(name) +default_resp_chunk = get_openai_chat_completion_chunk(name, usage_as_dict=True) def test_fireworks_costmanager(): @@ -57,27 +57,27 @@ async def mock_openai_acompletions_create(self, stream: bool = False, **kwargs) async def test_fireworks_acompletion(mocker): mocker.patch("openai.resources.chat.completions.AsyncCompletions.create", mock_openai_acompletions_create) - fireworks_gpt = FireworksLLM(mock_llm_config) - fireworks_gpt.model = "llama-v2-13b-chat" + fireworks_llm = FireworksLLM(mock_llm_config) + fireworks_llm.model = "llama-v2-13b-chat" - fireworks_gpt._update_costs( + fireworks_llm._update_costs( usage=CompletionUsage(prompt_tokens=500000, completion_tokens=500000, total_tokens=1000000) ) - assert fireworks_gpt.get_costs() == Costs( + assert fireworks_llm.get_costs() == Costs( total_prompt_tokens=500000, total_completion_tokens=500000, total_cost=0.5, total_budget=0 ) - resp = await fireworks_gpt.acompletion(messages) + resp = await fireworks_llm.acompletion(messages) assert resp.choices[0].message.content in resp_cont - resp = await fireworks_gpt.aask(prompt, stream=False) + resp = await fireworks_llm.aask(prompt, stream=False) assert resp == resp_cont - resp = await fireworks_gpt.acompletion_text(messages, stream=False) + resp = await fireworks_llm.acompletion_text(messages, stream=False) assert resp == resp_cont - resp = await fireworks_gpt.acompletion_text(messages, stream=True) + resp = await fireworks_llm.acompletion_text(messages, stream=True) assert resp == resp_cont - resp = await fireworks_gpt.aask(prompt) + resp = await fireworks_llm.aask(prompt) assert resp == resp_cont diff --git a/tests/metagpt/provider/test_google_gemini_api.py b/tests/metagpt/provider/test_google_gemini_api.py index ad0c7bbfe..dae9d123b 100644 --- a/tests/metagpt/provider/test_google_gemini_api.py +++ b/tests/metagpt/provider/test_google_gemini_api.py @@ -63,28 +63,28 @@ async def test_gemini_acompletion(mocker): mock_gemini_generate_content_async, ) - gemini_gpt = GeminiLLM(mock_llm_config) + gemini_llm = GeminiLLM(mock_llm_config) - assert gemini_gpt._user_msg(prompt) == {"role": "user", "parts": [prompt]} - assert gemini_gpt._assistant_msg(prompt) == {"role": "model", "parts": [prompt]} + assert gemini_llm._user_msg(prompt) == {"role": "user", "parts": [prompt]} + assert gemini_llm._assistant_msg(prompt) == {"role": "model", "parts": [prompt]} - usage = gemini_gpt.get_usage(gemini_messages, resp_cont) + usage = gemini_llm.get_usage(gemini_messages, resp_cont) assert usage == {"prompt_tokens": 20, "completion_tokens": 20} - resp = gemini_gpt.completion(gemini_messages) + resp = gemini_llm.completion(gemini_messages) assert resp == default_resp - resp = await gemini_gpt.acompletion(gemini_messages) + resp = await gemini_llm.acompletion(gemini_messages) assert resp.text == default_resp.text - resp = await gemini_gpt.aask(prompt, stream=False) + resp = await gemini_llm.aask(prompt, stream=False) assert resp == resp_cont - resp = await gemini_gpt.acompletion_text(gemini_messages, stream=False) + resp = await gemini_llm.acompletion_text(gemini_messages, stream=False) assert resp == resp_cont - resp = await gemini_gpt.acompletion_text(gemini_messages, stream=True) + resp = await gemini_llm.acompletion_text(gemini_messages, stream=True) assert resp == resp_cont - resp = await gemini_gpt.aask(prompt) + resp = await gemini_llm.aask(prompt) assert resp == resp_cont diff --git a/tests/metagpt/provider/test_ollama_api.py b/tests/metagpt/provider/test_ollama_api.py index 8e2625e35..01d53251c 100644 --- a/tests/metagpt/provider/test_ollama_api.py +++ b/tests/metagpt/provider/test_ollama_api.py @@ -39,19 +39,19 @@ async def mock_ollama_arequest(self, stream: bool = False, **kwargs) -> Tuple[An async def test_gemini_acompletion(mocker): mocker.patch("metagpt.provider.general_api_requestor.GeneralAPIRequestor.arequest", mock_ollama_arequest) - ollama_gpt = OllamaLLM(mock_llm_config) + ollama_llm = OllamaLLM(mock_llm_config) - resp = await ollama_gpt.acompletion(messages) + resp = await ollama_llm.acompletion(messages) assert resp["message"]["content"] == default_resp["message"]["content"] - resp = await ollama_gpt.aask(prompt, stream=False) + resp = await ollama_llm.aask(prompt, stream=False) assert resp == resp_cont - resp = await ollama_gpt.acompletion_text(messages, stream=False) + resp = await ollama_llm.acompletion_text(messages, stream=False) assert resp == resp_cont - resp = await ollama_gpt.acompletion_text(messages, stream=True) + resp = await ollama_llm.acompletion_text(messages, stream=True) assert resp == resp_cont - resp = await ollama_gpt.aask(prompt) + resp = await ollama_llm.aask(prompt) assert resp == resp_cont diff --git a/tests/metagpt/provider/test_open_llm_api.py b/tests/metagpt/provider/test_open_llm_api.py index 5b8a506e9..b2e759d06 100644 --- a/tests/metagpt/provider/test_open_llm_api.py +++ b/tests/metagpt/provider/test_open_llm_api.py @@ -17,11 +17,11 @@ from tests.metagpt.provider.req_resp_const import ( resp_cont_tmpl, ) -llm_name = "llama2-7b" -resp_cont = resp_cont_tmpl.format(name=llm_name) -default_resp = get_openai_chat_completion(llm_name) +name = "llama2-7b" +resp_cont = resp_cont_tmpl.format(name=name) +default_resp = get_openai_chat_completion(name) -default_resp_chunk = get_openai_chat_completion_chunk(llm_name) +default_resp_chunk = get_openai_chat_completion_chunk(name) async def mock_openai_acompletions_create(self, stream: bool = False, **kwargs) -> ChatCompletionChunk: @@ -40,26 +40,26 @@ async def mock_openai_acompletions_create(self, stream: bool = False, **kwargs) async def test_openllm_acompletion(mocker): mocker.patch("openai.resources.chat.completions.AsyncCompletions.create", mock_openai_acompletions_create) - openllm_gpt = OpenLLM(mock_llm_config) - openllm_gpt.model = "llama-v2-13b-chat" + openllm_llm = OpenLLM(mock_llm_config) + openllm_llm.model = "llama-v2-13b-chat" - openllm_gpt.cost_manager = CostManager() - openllm_gpt._update_costs(usage=CompletionUsage(prompt_tokens=100, completion_tokens=100, total_tokens=200)) - assert openllm_gpt.get_costs() == Costs( + openllm_llm.cost_manager = CostManager() + openllm_llm._update_costs(usage=CompletionUsage(prompt_tokens=100, completion_tokens=100, total_tokens=200)) + assert openllm_llm.get_costs() == Costs( total_prompt_tokens=100, total_completion_tokens=100, total_cost=0, total_budget=0 ) - resp = await openllm_gpt.acompletion(messages) + resp = await openllm_llm.acompletion(messages) assert resp.choices[0].message.content in resp_cont - resp = await openllm_gpt.aask(prompt, stream=False) + resp = await openllm_llm.aask(prompt, stream=False) assert resp == resp_cont - resp = await openllm_gpt.acompletion_text(messages, stream=False) + resp = await openllm_llm.acompletion_text(messages, stream=False) assert resp == resp_cont - resp = await openllm_gpt.acompletion_text(messages, stream=True) + resp = await openllm_llm.acompletion_text(messages, stream=True) assert resp == resp_cont - resp = await openllm_gpt.aask(prompt) + resp = await openllm_llm.aask(prompt) assert resp == resp_cont diff --git a/tests/metagpt/provider/test_qianfan_api.py b/tests/metagpt/provider/test_qianfan_api.py index 76271b1e8..30ac06911 100644 --- a/tests/metagpt/provider/test_qianfan_api.py +++ b/tests/metagpt/provider/test_qianfan_api.py @@ -2,14 +2,45 @@ # -*- coding: utf-8 -*- # @Desc : the unittest of qianfan api +from typing import Dict, Union, AsyncIterator import pytest +from qianfan.resources.typing import JsonBody, QfResponse + from metagpt.provider.qianfan_api import QianFanLLM -from tests.metagpt.provider.req_resp_const import prompt, messages, resp_cont_tmpl +from tests.metagpt.provider.mock_llm_config import mock_llm_config_qianfan +from tests.metagpt.provider.req_resp_const import resp_cont_tmpl, prompt, messages, llm_general_chat_funcs_test, get_qianfan_response + +name = "ERNIE-Bot-turbo" +resp_cont = resp_cont_tmpl.format(name=name) -resp_cont = resp_cont_tmpl.format(name="ERNIE-Bot-turbo") +def mock_qianfan_do(self, messages: list[dict], model: str, stream: bool = False, system: str = None) -> QfResponse: + return get_qianfan_response(name=name) -def test_qianfan_acompletion(mocker): - assert True, True +async def mock_qianfan_ado(self, messages: list[dict], model: str, stream: bool = True, system: str = None) -> Union[QfResponse, AsyncIterator[QfResponse]]: + resps = [get_qianfan_response(name=name)] + if stream: + async def aresp_iterator(resps: list[JsonBody]): + for resp in resps: + yield resp + return aresp_iterator(resps) + else: + return resps[0] + + +@pytest.mark.asyncio +async def test_qianfan_acompletion(mocker): + mocker.patch("qianfan.resources.llm.chat_completion.ChatCompletion.do", mock_qianfan_do) + mocker.patch("qianfan.resources.llm.chat_completion.ChatCompletion.ado", mock_qianfan_ado) + + qianfan_llm = QianFanLLM(mock_llm_config_qianfan) + + resp = qianfan_llm.completion(messages) + assert resp.get("result") == resp_cont + + resp = await qianfan_llm.acompletion(messages) + assert resp.get("result") == resp_cont + + await llm_general_chat_funcs_test(qianfan_llm, prompt, messages, resp_cont) diff --git a/tests/metagpt/provider/test_spark_api.py b/tests/metagpt/provider/test_spark_api.py index 32a839393..8aa8bc7a8 100644 --- a/tests/metagpt/provider/test_spark_api.py +++ b/tests/metagpt/provider/test_spark_api.py @@ -50,19 +50,19 @@ async def test_spark_aask(mocker): async def test_spark_acompletion(mocker): mocker.patch("metagpt.provider.spark_api.GetMessageFromWeb.run", mock_spark_get_msg_from_web_run) - spark_gpt = SparkLLM(mock_llm_config) + spark_llm = SparkLLM(mock_llm_config) - resp = await spark_gpt.acompletion([]) + resp = await spark_llm.acompletion([]) assert resp == resp_cont - resp = await spark_gpt.aask(prompt, stream=False) + resp = await spark_llm.aask(prompt, stream=False) assert resp == resp_cont - resp = await spark_gpt.acompletion_text([], stream=False) + resp = await spark_llm.acompletion_text([], stream=False) assert resp == resp_cont - resp = await spark_gpt.acompletion_text([], stream=True) + resp = await spark_llm.acompletion_text([], stream=True) assert resp == resp_cont - resp = await spark_gpt.aask(prompt) + resp = await spark_llm.aask(prompt) assert resp == resp_cont diff --git a/tests/metagpt/provider/test_zhipuai_api.py b/tests/metagpt/provider/test_zhipuai_api.py index 064562bff..3dada367c 100644 --- a/tests/metagpt/provider/test_zhipuai_api.py +++ b/tests/metagpt/provider/test_zhipuai_api.py @@ -11,11 +11,12 @@ from tests.metagpt.provider.req_resp_const import ( messages, prompt, resp_cont_tmpl, + llm_general_chat_funcs_test ) -llm_name = "ChatGLM-4" -resp_cont = resp_cont_tmpl.format(name=llm_name) -default_resp = get_part_chat_completion(llm_name) +name = "ChatGLM-4" +resp_cont = resp_cont_tmpl.format(name=name) +default_resp = get_part_chat_completion(name) async def mock_zhipuai_acreate_stream(**kwargs): @@ -47,22 +48,12 @@ async def test_zhipuai_acompletion(mocker): mocker.patch("metagpt.provider.zhipuai.zhipu_model_api.ZhiPuModelAPI.acreate", mock_zhipuai_acreate) mocker.patch("metagpt.provider.zhipuai.zhipu_model_api.ZhiPuModelAPI.acreate_stream", mock_zhipuai_acreate_stream) - zhipu_gpt = ZhiPuAILLM(mock_llm_config_zhipu) + zhipu_llm = ZhiPuAILLM(mock_llm_config_zhipu) - resp = await zhipu_gpt.acompletion(messages) + resp = await zhipu_llm.acompletion(messages) assert resp["choices"][0]["message"]["content"] == resp_cont - resp = await zhipu_gpt.aask(prompt, stream=False) - assert resp == resp_cont - - resp = await zhipu_gpt.acompletion_text(messages, stream=False) - assert resp == resp_cont - - resp = await zhipu_gpt.acompletion_text(messages, stream=True) - assert resp == resp_cont - - resp = await zhipu_gpt.aask(prompt) - assert resp == resp_cont + await llm_general_chat_funcs_test(zhipu_llm, prompt, messages, resp_cont) def test_zhipuai_proxy(): From 997e25e97d291d83e0fc587abde46bf383308a59 Mon Sep 17 00:00:00 2001 From: better629 Date: Wed, 7 Feb 2024 18:42:22 +0800 Subject: [PATCH 636/637] simplify provider ut code --- tests/metagpt/provider/mock_llm_config.py | 7 +------ tests/metagpt/provider/req_resp_const.py | 16 +++++----------- tests/metagpt/provider/test_fireworks_llm.py | 13 ++----------- .../metagpt/provider/test_google_gemini_api.py | 13 ++----------- tests/metagpt/provider/test_ollama_api.py | 16 +++++++--------- tests/metagpt/provider/test_open_llm_api.py | 13 ++----------- tests/metagpt/provider/test_qianfan_api.py | 18 ++++++++++++++---- tests/metagpt/provider/test_spark_api.py | 18 ++++++------------ tests/metagpt/provider/test_zhipuai_api.py | 2 +- 9 files changed, 40 insertions(+), 76 deletions(-) diff --git a/tests/metagpt/provider/mock_llm_config.py b/tests/metagpt/provider/mock_llm_config.py index e61e32e8b..e0afaa51e 100644 --- a/tests/metagpt/provider/mock_llm_config.py +++ b/tests/metagpt/provider/mock_llm_config.py @@ -53,9 +53,4 @@ mock_llm_config_spark = LLMConfig( base_url="wss://spark-api.xf-yun.com/v3.1/chat", ) -mock_llm_config_qianfan = LLMConfig( - api_type="qianfan", - access_key="xxx", - secret_key="xxx", - model="ERNIE-Bot-turbo" -) +mock_llm_config_qianfan = LLMConfig(api_type="qianfan", access_key="xxx", secret_key="xxx", model="ERNIE-Bot-turbo") diff --git a/tests/metagpt/provider/req_resp_const.py b/tests/metagpt/provider/req_resp_const.py index 20d8e0914..73939e1c6 100644 --- a/tests/metagpt/provider/req_resp_const.py +++ b/tests/metagpt/provider/req_resp_const.py @@ -2,7 +2,7 @@ # -*- coding: utf-8 -*- # @Desc : default request & response data for provider unittest -from typing import Dict + from openai.types.chat.chat_completion import ( ChatCompletion, ChatCompletionMessage, @@ -12,7 +12,7 @@ from openai.types.chat.chat_completion_chunk import ChatCompletionChunk from openai.types.chat.chat_completion_chunk import Choice as AChoice from openai.types.chat.chat_completion_chunk import ChoiceDelta from openai.types.completion_usage import CompletionUsage -from qianfan.resources.typing import QfResponse, default_field +from qianfan.resources.typing import QfResponse from metagpt.provider.base_llm import BaseLLM @@ -80,6 +80,7 @@ def get_openai_chat_completion_chunk(name: str, usage_as_dict: bool = False) -> ) return openai_chat_completion_chunk + # For gemini gemini_messages = [{"role": "user", "parts": prompt}] @@ -92,20 +93,13 @@ qf_jsonbody_dict = { "result": "", "is_truncated": False, "need_clear_history": False, - "usage": { - "prompt_tokens": 7, - "completion_tokens": 15, - "total_tokens": 22 - } + "usage": {"prompt_tokens": 7, "completion_tokens": 15, "total_tokens": 22}, } def get_qianfan_response(name: str) -> QfResponse: qf_jsonbody_dict["result"] = resp_cont_tmpl.format(name=name) - return QfResponse( - code=200, - body=qf_jsonbody_dict - ) + return QfResponse(code=200, body=qf_jsonbody_dict) # For llm general chat functions call diff --git a/tests/metagpt/provider/test_fireworks_llm.py b/tests/metagpt/provider/test_fireworks_llm.py index e28f7500b..1c1aa9caa 100644 --- a/tests/metagpt/provider/test_fireworks_llm.py +++ b/tests/metagpt/provider/test_fireworks_llm.py @@ -16,6 +16,7 @@ from tests.metagpt.provider.mock_llm_config import mock_llm_config from tests.metagpt.provider.req_resp_const import ( get_openai_chat_completion, get_openai_chat_completion_chunk, + llm_general_chat_funcs_test, messages, prompt, resp_cont_tmpl, @@ -70,14 +71,4 @@ async def test_fireworks_acompletion(mocker): resp = await fireworks_llm.acompletion(messages) assert resp.choices[0].message.content in resp_cont - resp = await fireworks_llm.aask(prompt, stream=False) - assert resp == resp_cont - - resp = await fireworks_llm.acompletion_text(messages, stream=False) - assert resp == resp_cont - - resp = await fireworks_llm.acompletion_text(messages, stream=True) - assert resp == resp_cont - - resp = await fireworks_llm.aask(prompt) - assert resp == resp_cont + await llm_general_chat_funcs_test(fireworks_llm, prompt, messages, resp_cont) diff --git a/tests/metagpt/provider/test_google_gemini_api.py b/tests/metagpt/provider/test_google_gemini_api.py index dae9d123b..50c15ee19 100644 --- a/tests/metagpt/provider/test_google_gemini_api.py +++ b/tests/metagpt/provider/test_google_gemini_api.py @@ -13,6 +13,7 @@ from metagpt.provider.google_gemini_api import GeminiLLM from tests.metagpt.provider.mock_llm_config import mock_llm_config from tests.metagpt.provider.req_resp_const import ( gemini_messages, + llm_general_chat_funcs_test, prompt, resp_cont_tmpl, ) @@ -77,14 +78,4 @@ async def test_gemini_acompletion(mocker): resp = await gemini_llm.acompletion(gemini_messages) assert resp.text == default_resp.text - resp = await gemini_llm.aask(prompt, stream=False) - assert resp == resp_cont - - resp = await gemini_llm.acompletion_text(gemini_messages, stream=False) - assert resp == resp_cont - - resp = await gemini_llm.acompletion_text(gemini_messages, stream=True) - assert resp == resp_cont - - resp = await gemini_llm.aask(prompt) - assert resp == resp_cont + await llm_general_chat_funcs_test(gemini_llm, prompt, gemini_messages, resp_cont) diff --git a/tests/metagpt/provider/test_ollama_api.py b/tests/metagpt/provider/test_ollama_api.py index 01d53251c..af2e929e9 100644 --- a/tests/metagpt/provider/test_ollama_api.py +++ b/tests/metagpt/provider/test_ollama_api.py @@ -9,7 +9,12 @@ import pytest from metagpt.provider.ollama_api import OllamaLLM from tests.metagpt.provider.mock_llm_config import mock_llm_config -from tests.metagpt.provider.req_resp_const import messages, prompt, resp_cont_tmpl +from tests.metagpt.provider.req_resp_const import ( + llm_general_chat_funcs_test, + messages, + prompt, + resp_cont_tmpl, +) resp_cont = resp_cont_tmpl.format(name="ollama") default_resp = {"message": {"role": "assistant", "content": resp_cont}} @@ -47,11 +52,4 @@ async def test_gemini_acompletion(mocker): resp = await ollama_llm.aask(prompt, stream=False) assert resp == resp_cont - resp = await ollama_llm.acompletion_text(messages, stream=False) - assert resp == resp_cont - - resp = await ollama_llm.acompletion_text(messages, stream=True) - assert resp == resp_cont - - resp = await ollama_llm.aask(prompt) - assert resp == resp_cont + await llm_general_chat_funcs_test(ollama_llm, prompt, messages, resp_cont) diff --git a/tests/metagpt/provider/test_open_llm_api.py b/tests/metagpt/provider/test_open_llm_api.py index b2e759d06..aa38b95a6 100644 --- a/tests/metagpt/provider/test_open_llm_api.py +++ b/tests/metagpt/provider/test_open_llm_api.py @@ -12,6 +12,7 @@ from tests.metagpt.provider.mock_llm_config import mock_llm_config from tests.metagpt.provider.req_resp_const import ( get_openai_chat_completion, get_openai_chat_completion_chunk, + llm_general_chat_funcs_test, messages, prompt, resp_cont_tmpl, @@ -52,14 +53,4 @@ async def test_openllm_acompletion(mocker): resp = await openllm_llm.acompletion(messages) assert resp.choices[0].message.content in resp_cont - resp = await openllm_llm.aask(prompt, stream=False) - assert resp == resp_cont - - resp = await openllm_llm.acompletion_text(messages, stream=False) - assert resp == resp_cont - - resp = await openllm_llm.acompletion_text(messages, stream=True) - assert resp == resp_cont - - resp = await openllm_llm.aask(prompt) - assert resp == resp_cont + await llm_general_chat_funcs_test(openllm_llm, prompt, messages, resp_cont) diff --git a/tests/metagpt/provider/test_qianfan_api.py b/tests/metagpt/provider/test_qianfan_api.py index 30ac06911..28341425c 100644 --- a/tests/metagpt/provider/test_qianfan_api.py +++ b/tests/metagpt/provider/test_qianfan_api.py @@ -2,14 +2,20 @@ # -*- coding: utf-8 -*- # @Desc : the unittest of qianfan api -from typing import Dict, Union, AsyncIterator -import pytest +from typing import AsyncIterator, Union +import pytest from qianfan.resources.typing import JsonBody, QfResponse from metagpt.provider.qianfan_api import QianFanLLM from tests.metagpt.provider.mock_llm_config import mock_llm_config_qianfan -from tests.metagpt.provider.req_resp_const import resp_cont_tmpl, prompt, messages, llm_general_chat_funcs_test, get_qianfan_response +from tests.metagpt.provider.req_resp_const import ( + get_qianfan_response, + llm_general_chat_funcs_test, + messages, + prompt, + resp_cont_tmpl, +) name = "ERNIE-Bot-turbo" resp_cont = resp_cont_tmpl.format(name=name) @@ -19,12 +25,16 @@ def mock_qianfan_do(self, messages: list[dict], model: str, stream: bool = False return get_qianfan_response(name=name) -async def mock_qianfan_ado(self, messages: list[dict], model: str, stream: bool = True, system: str = None) -> Union[QfResponse, AsyncIterator[QfResponse]]: +async def mock_qianfan_ado( + self, messages: list[dict], model: str, stream: bool = True, system: str = None +) -> Union[QfResponse, AsyncIterator[QfResponse]]: resps = [get_qianfan_response(name=name)] if stream: + async def aresp_iterator(resps: list[JsonBody]): for resp in resps: yield resp + return aresp_iterator(resps) else: return resps[0] diff --git a/tests/metagpt/provider/test_spark_api.py b/tests/metagpt/provider/test_spark_api.py index 8aa8bc7a8..9c278267d 100644 --- a/tests/metagpt/provider/test_spark_api.py +++ b/tests/metagpt/provider/test_spark_api.py @@ -9,7 +9,11 @@ from tests.metagpt.provider.mock_llm_config import ( mock_llm_config, mock_llm_config_spark, ) -from tests.metagpt.provider.req_resp_const import prompt, resp_cont_tmpl +from tests.metagpt.provider.req_resp_const import ( + llm_general_chat_funcs_test, + prompt, + resp_cont_tmpl, +) resp_cont = resp_cont_tmpl.format(name="Spark") @@ -55,14 +59,4 @@ async def test_spark_acompletion(mocker): resp = await spark_llm.acompletion([]) assert resp == resp_cont - resp = await spark_llm.aask(prompt, stream=False) - assert resp == resp_cont - - resp = await spark_llm.acompletion_text([], stream=False) - assert resp == resp_cont - - resp = await spark_llm.acompletion_text([], stream=True) - assert resp == resp_cont - - resp = await spark_llm.aask(prompt) - assert resp == resp_cont + await llm_general_chat_funcs_test(spark_llm, prompt, prompt, resp_cont) diff --git a/tests/metagpt/provider/test_zhipuai_api.py b/tests/metagpt/provider/test_zhipuai_api.py index 3dada367c..8ec9ab4f9 100644 --- a/tests/metagpt/provider/test_zhipuai_api.py +++ b/tests/metagpt/provider/test_zhipuai_api.py @@ -8,10 +8,10 @@ from metagpt.provider.zhipuai_api import ZhiPuAILLM from tests.metagpt.provider.mock_llm_config import mock_llm_config_zhipu from tests.metagpt.provider.req_resp_const import ( get_part_chat_completion, + llm_general_chat_funcs_test, messages, prompt, resp_cont_tmpl, - llm_general_chat_funcs_test ) name = "ChatGLM-4" From 5a2084cda87ba1753345e72889a6c18574092606 Mon Sep 17 00:00:00 2001 From: better629 Date: Thu, 8 Feb 2024 07:32:34 +0800 Subject: [PATCH 637/637] Revert "Feat add qianfan api support" --- .github/workflows/fulltest.yaml | 1 + .github/workflows/unittest.yaml | 2 +- examples/llm_hello_world.py | 13 +- metagpt/configs/llm_config.py | 8 +- metagpt/provider/__init__.py | 2 - metagpt/provider/base_llm.py | 25 +-- metagpt/provider/fireworks_api.py | 15 +- metagpt/provider/google_gemini_api.py | 10 ++ metagpt/provider/ollama_api.py | 10 ++ metagpt/provider/open_llm_api.py | 13 +- metagpt/provider/openai_api.py | 17 +- metagpt/provider/qianfan_api.py | 152 ------------------ metagpt/provider/zhipuai_api.py | 10 ++ metagpt/utils/cost_manager.py | 4 +- metagpt/utils/token_counter.py | 53 ------ requirements.txt | 1 - tests/metagpt/provider/mock_llm_config.py | 12 -- tests/metagpt/provider/req_resp_const.py | 117 -------------- tests/metagpt/provider/test_anthropic_api.py | 12 +- tests/metagpt/provider/test_base_llm.py | 53 +++--- tests/metagpt/provider/test_fireworks_llm.py | 76 ++++++--- .../provider/test_google_gemini_api.py | 36 +++-- tests/metagpt/provider/test_ollama_api.py | 30 ++-- tests/metagpt/provider/test_open_llm_api.py | 76 ++++++--- tests/metagpt/provider/test_qianfan_api.py | 56 ------- tests/metagpt/provider/test_spark_api.py | 44 ++--- tests/metagpt/provider/test_zhipuai_api.py | 38 +++-- tests/spark.yaml | 7 + 28 files changed, 319 insertions(+), 574 deletions(-) delete mode 100644 metagpt/provider/qianfan_api.py delete mode 100644 tests/metagpt/provider/req_resp_const.py delete mode 100644 tests/metagpt/provider/test_qianfan_api.py create mode 100644 tests/spark.yaml diff --git a/.github/workflows/fulltest.yaml b/.github/workflows/fulltest.yaml index 70c800481..f5c6049e1 100644 --- a/.github/workflows/fulltest.yaml +++ b/.github/workflows/fulltest.yaml @@ -54,6 +54,7 @@ jobs: export ALLOW_OPENAI_API_CALL=0 echo "${{ secrets.METAGPT_KEY_YAML }}" | base64 -d > config/key.yaml mkdir -p ~/.metagpt && echo "${{ secrets.METAGPT_CONFIG2_YAML }}" | base64 -d > ~/.metagpt/config2.yaml + echo "${{ secrets.SPARK_YAML }}" | base64 -d > ~/.metagpt/spark.yaml pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt - name: Show coverage report run: | diff --git a/.github/workflows/unittest.yaml b/.github/workflows/unittest.yaml index afa9faba7..2e7e3ce2b 100644 --- a/.github/workflows/unittest.yaml +++ b/.github/workflows/unittest.yaml @@ -31,7 +31,7 @@ jobs: - name: Test with pytest run: | export ALLOW_OPENAI_API_CALL=0 - mkdir -p ~/.metagpt && cp tests/config2.yaml ~/.metagpt/config2.yaml + mkdir -p ~/.metagpt && cp tests/config2.yaml ~/.metagpt/config2.yaml && cp tests/spark.yaml ~/.metagpt/spark.yaml pytest tests/ --doctest-modules --cov=./metagpt/ --cov-report=xml:cov.xml --cov-report=html:htmlcov --durations=20 | tee unittest.txt - name: Show coverage report run: | diff --git a/examples/llm_hello_world.py b/examples/llm_hello_world.py index e22edbdf2..219a303c8 100644 --- a/examples/llm_hello_world.py +++ b/examples/llm_hello_world.py @@ -13,18 +13,7 @@ from metagpt.logs import logger async def main(): llm = LLM() - # llm type check - id_ques = "what's your name" - logger.info(f"{id_ques}: ") - logger.info(await llm.aask(id_ques)) - logger.info("\n\n") - - logger.info( - await llm.aask( - "who are you", system_msgs=["act as a robot, answer 'I'am robot' if the question is 'who are you'"] - ) - ) - + logger.info(await llm.aask("hello world")) logger.info(await llm.aask_batch(["hi", "write python hello world."])) hello_msg = [{"role": "user", "content": "count from 1 to 10. split by newline."}] diff --git a/metagpt/configs/llm_config.py b/metagpt/configs/llm_config.py index 1b05b5270..fb923d3e4 100644 --- a/metagpt/configs/llm_config.py +++ b/metagpt/configs/llm_config.py @@ -24,7 +24,6 @@ class LLMType(Enum): METAGPT = "metagpt" AZURE = "azure" OLLAMA = "ollama" - QIANFAN = "qianfan" # Baidu BCE def __missing__(self, key): return self.OPENAI @@ -37,18 +36,13 @@ class LLMConfig(YamlModel): Optional Fields in pydantic: https://docs.pydantic.dev/latest/migration/#required-optional-and-nullable-fields """ - api_key: str = "sk-" + api_key: str api_type: LLMType = LLMType.OPENAI base_url: str = "https://api.openai.com/v1" api_version: Optional[str] = None model: Optional[str] = None # also stands for DEPLOYMENT_NAME - # For Cloud Service Provider like Baidu/ Alibaba - access_key: Optional[str] = None - secret_key: Optional[str] = None - endpoint: Optional[str] = None # for self-deployed model on the cloud - # For Spark(Xunfei), maybe remove later app_id: Optional[str] = None api_secret: Optional[str] = None diff --git a/metagpt/provider/__init__.py b/metagpt/provider/__init__.py index 8c0aab836..675734811 100644 --- a/metagpt/provider/__init__.py +++ b/metagpt/provider/__init__.py @@ -16,7 +16,6 @@ from metagpt.provider.azure_openai_api import AzureOpenAILLM from metagpt.provider.metagpt_api import MetaGPTLLM from metagpt.provider.human_provider import HumanProvider from metagpt.provider.spark_api import SparkLLM -from metagpt.provider.qianfan_api import QianFanLLM __all__ = [ "FireworksLLM", @@ -29,5 +28,4 @@ __all__ = [ "OllamaLLM", "HumanProvider", "SparkLLM", - "QianFanLLM", ] diff --git a/metagpt/provider/base_llm.py b/metagpt/provider/base_llm.py index 2f57b15aa..b144471b5 100644 --- a/metagpt/provider/base_llm.py +++ b/metagpt/provider/base_llm.py @@ -11,12 +11,11 @@ from abc import ABC, abstractmethod from typing import Optional, Union from openai import AsyncOpenAI -from pydantic import BaseModel from metagpt.configs.llm_config import LLMConfig from metagpt.logs import logger from metagpt.schema import Message -from metagpt.utils.cost_manager import CostManager, Costs +from metagpt.utils.cost_manager import CostManager class BaseLLM(ABC): @@ -68,28 +67,6 @@ class BaseLLM(ABC): def _default_system_msg(self): return self._system_msg(self.system_prompt) - def _update_costs(self, usage: Union[dict, BaseModel], model: str = None, local_calc_usage: bool = True): - """update each request's token cost - Args: - model (str): model name or in some scenarios called endpoint - local_calc_usage (bool): some models don't calculate usage, it will overwrite LLMConfig.calc_usage - """ - calc_usage = self.config.calc_usage and local_calc_usage - model = model if model else self.model - usage = usage.model_dump() if isinstance(usage, BaseModel) else usage - if calc_usage and self.cost_manager: - try: - prompt_tokens = int(usage.get("prompt_tokens", 0)) - completion_tokens = int(usage.get("completion_tokens", 0)) - self.cost_manager.update_cost(prompt_tokens, completion_tokens, model) - except Exception as e: - logger.error(f"{self.__class__.__name__} updats costs failed! exp: {e}") - - def get_costs(self) -> Costs: - if not self.cost_manager: - return Costs(0, 0, 0, 0) - return self.cost_manager.get_costs() - async def aask( self, msg: str, diff --git a/metagpt/provider/fireworks_api.py b/metagpt/provider/fireworks_api.py index e62a7066e..d56453a85 100644 --- a/metagpt/provider/fireworks_api.py +++ b/metagpt/provider/fireworks_api.py @@ -19,7 +19,7 @@ from metagpt.configs.llm_config import LLMConfig, LLMType from metagpt.logs import logger from metagpt.provider.llm_provider_registry import register_provider from metagpt.provider.openai_api import OpenAILLM, log_and_reraise -from metagpt.utils.cost_manager import CostManager +from metagpt.utils.cost_manager import CostManager, Costs MODEL_GRADE_TOKEN_COSTS = { "-1": {"prompt": 0.0, "completion": 0.0}, # abnormal condition @@ -81,6 +81,17 @@ class FireworksLLM(OpenAILLM): kwargs = dict(api_key=self.config.api_key, base_url=self.config.base_url) return kwargs + def _update_costs(self, usage: CompletionUsage): + if self.config.calc_usage and usage: + try: + # use FireworksCostManager not context.cost_manager + self.cost_manager.update_cost(usage.prompt_tokens, usage.completion_tokens, self.model) + except Exception as e: + logger.error(f"updating costs failed!, exp: {e}") + + def get_costs(self) -> Costs: + return self.cost_manager.get_costs() + async def _achat_completion_stream(self, messages: list[dict], timeout=3) -> str: response: AsyncStream[ChatCompletionChunk] = await self.aclient.chat.completions.create( **self._cons_kwargs(messages), stream=True @@ -102,7 +113,7 @@ class FireworksLLM(OpenAILLM): usage = CompletionUsage(**chunk.usage) full_content = "".join(collected_content) - self._update_costs(usage.model_dump()) + self._update_costs(usage) return full_content @retry( diff --git a/metagpt/provider/google_gemini_api.py b/metagpt/provider/google_gemini_api.py index 87ea81c80..2647ab16b 100644 --- a/metagpt/provider/google_gemini_api.py +++ b/metagpt/provider/google_gemini_api.py @@ -72,6 +72,16 @@ class GeminiLLM(BaseLLM): kwargs = {"contents": messages, "generation_config": GenerationConfig(temperature=0.3), "stream": stream} return kwargs + def _update_costs(self, usage: dict): + """update each request's token cost""" + if self.config.calc_usage: + try: + prompt_tokens = int(usage.get("prompt_tokens", 0)) + completion_tokens = int(usage.get("completion_tokens", 0)) + self.cost_manager.update_cost(prompt_tokens, completion_tokens, self.model) + except Exception as e: + logger.error(f"google gemini updats costs failed! exp: {e}") + def get_choice_text(self, resp: GenerateContentResponse) -> str: return resp.text diff --git a/metagpt/provider/ollama_api.py b/metagpt/provider/ollama_api.py index 52e8dbe36..c9103b018 100644 --- a/metagpt/provider/ollama_api.py +++ b/metagpt/provider/ollama_api.py @@ -46,6 +46,16 @@ class OllamaLLM(BaseLLM): kwargs = {"model": self.model, "messages": messages, "options": {"temperature": 0.3}, "stream": stream} return kwargs + def _update_costs(self, usage: dict): + """update each request's token cost""" + if self.config.calc_usage: + try: + prompt_tokens = int(usage.get("prompt_tokens", 0)) + completion_tokens = int(usage.get("completion_tokens", 0)) + self._cost_manager.update_cost(prompt_tokens, completion_tokens, self.model) + except Exception as e: + logger.error(f"ollama updats costs failed! exp: {e}") + def get_choice_text(self, resp: dict) -> str: """get the resp content from llm response""" assist_msg = resp.get("message", {}) diff --git a/metagpt/provider/open_llm_api.py b/metagpt/provider/open_llm_api.py index 69371e379..a29b263a4 100644 --- a/metagpt/provider/open_llm_api.py +++ b/metagpt/provider/open_llm_api.py @@ -8,7 +8,7 @@ from metagpt.configs.llm_config import LLMConfig, LLMType from metagpt.logs import logger from metagpt.provider.llm_provider_registry import register_provider from metagpt.provider.openai_api import OpenAILLM -from metagpt.utils.cost_manager import TokenCostManager +from metagpt.utils.cost_manager import Costs, TokenCostManager from metagpt.utils.token_counter import count_message_tokens, count_string_tokens @@ -34,3 +34,14 @@ class OpenLLM(OpenAILLM): logger.error(f"usage calculation failed!: {e}") return usage + + def _update_costs(self, usage: CompletionUsage): + if self.config.calc_usage and usage: + try: + # use OpenLLMCostManager not CONFIG.cost_manager + self._cost_manager.update_cost(usage.prompt_tokens, usage.completion_tokens, self.model) + except Exception as e: + logger.error(f"updating costs failed!, exp: {e}") + + def get_costs(self) -> Costs: + return self._cost_manager.get_costs() diff --git a/metagpt/provider/openai_api.py b/metagpt/provider/openai_api.py index 2ae14f437..fe41fb05f 100644 --- a/metagpt/provider/openai_api.py +++ b/metagpt/provider/openai_api.py @@ -29,7 +29,7 @@ from metagpt.provider.constant import GENERAL_FUNCTION_SCHEMA from metagpt.provider.llm_provider_registry import register_provider from metagpt.schema import Message from metagpt.utils.common import CodeParser, decode_image -from metagpt.utils.cost_manager import CostManager +from metagpt.utils.cost_manager import CostManager, Costs from metagpt.utils.exceptions import handle_exception from metagpt.utils.token_counter import ( count_message_tokens, @@ -55,13 +55,16 @@ class OpenAILLM(BaseLLM): def __init__(self, config: LLMConfig): self.config = config + self._init_model() self._init_client() self.auto_max_tokens = False self.cost_manager: Optional[CostManager] = None + def _init_model(self): + self.model = self.config.model # Used in _calc_usage & _cons_kwargs + def _init_client(self): """https://github.com/openai/openai-python#async-usage""" - self.model = self.config.model # Used in _calc_usage & _cons_kwargs kwargs = self._make_client_kwargs() self.aclient = AsyncOpenAI(**kwargs) @@ -237,6 +240,16 @@ class OpenAILLM(BaseLLM): return usage + @handle_exception + def _update_costs(self, usage: CompletionUsage): + if self.config.calc_usage and usage and self.cost_manager: + self.cost_manager.update_cost(usage.prompt_tokens, usage.completion_tokens, self.model) + + def get_costs(self) -> Costs: + if not self.cost_manager: + return Costs(0, 0, 0, 0) + return self.cost_manager.get_costs() + def _get_max_tokens(self, messages: list[dict]): if not self.auto_max_tokens: return self.config.max_token diff --git a/metagpt/provider/qianfan_api.py b/metagpt/provider/qianfan_api.py deleted file mode 100644 index 6f94b9cea..000000000 --- a/metagpt/provider/qianfan_api.py +++ /dev/null @@ -1,152 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Desc : llm api of qianfan from Baidu, supports ERNIE(wen xin yi yan) and opensource models -import copy -import os - -import qianfan -from qianfan import ChatCompletion -from qianfan.resources.typing import JsonBody -from tenacity import ( - after_log, - retry, - retry_if_exception_type, - stop_after_attempt, - wait_random_exponential, -) - -from metagpt.configs.llm_config import LLMConfig, LLMType -from metagpt.logs import log_llm_stream, logger -from metagpt.provider.base_llm import BaseLLM -from metagpt.provider.llm_provider_registry import register_provider -from metagpt.provider.openai_api import log_and_reraise -from metagpt.utils.cost_manager import CostManager -from metagpt.utils.token_counter import ( - QianFan_EndPoint_TOKEN_COSTS, - QianFan_MODEL_TOKEN_COSTS, -) - - -@register_provider(LLMType.QIANFAN) -class QianFanLLM(BaseLLM): - """ - Refs - Auth: https://cloud.baidu.com/doc/WENXINWORKSHOP/s/3lmokh7n6#%E3%80%90%E6%8E%A8%E8%8D%90%E3%80%91%E4%BD%BF%E7%94%A8%E5%AE%89%E5%85%A8%E8%AE%A4%E8%AF%81aksk%E9%89%B4%E6%9D%83%E8%B0%83%E7%94%A8%E6%B5%81%E7%A8%8B - Token Price: https://cloud.baidu.com/doc/WENXINWORKSHOP/s/hlrk4akp7#tokens%E5%90%8E%E4%BB%98%E8%B4%B9 - Models: https://cloud.baidu.com/doc/WENXINWORKSHOP/s/wlmhm7vuo#%E5%AF%B9%E8%AF%9Dchat - https://cloud.baidu.com/doc/WENXINWORKSHOP/s/xlmokikxe#%E6%94%AF%E6%8C%81%E6%A8%A1%E5%9E%8B%E5%88%97%E8%A1%A8 - """ - - def __init__(self, config: LLMConfig): - self.config = config - self.use_system_prompt = False # only some ERNIE-x related models support system_prompt - self.__init_qianfan() - self.cost_manager = CostManager(token_costs=self.token_costs) - - def __init_qianfan(self): - if self.config.access_key and self.config.secret_key: - # for system level auth, use access_key and secret_key, recommended by official - # set environment variable due to official recommendation - os.environ.setdefault("QIANFAN_ACCESS_KEY", self.config.access_key) - os.environ.setdefault("QIANFAN_SECRET_KEY", self.config.secret_key) - elif self.config.api_key and self.config.secret_key: - # for application level auth, use api_key and secret_key - # set environment variable due to official recommendation - os.environ.setdefault("QIANFAN_AK", self.config.api_key) - os.environ.setdefault("QIANFAN_SK", self.config.secret_key) - else: - raise ValueError("Set the `access_key`&`secret_key` or `api_key`&`secret_key` first") - - support_system_pairs = [ - ("ERNIE-Bot-4", "completions_pro"), # (model, corresponding-endpoint) - ("ERNIE-Bot-8k", "ernie_bot_8k"), - ("ERNIE-Bot", "completions"), - ("ERNIE-Bot-turbo", "eb-instant"), - ("ERNIE-Speed", "ernie_speed"), - ("EB-turbo-AppBuilder", "ai_apaas"), - ] - if self.config.model in [pair[0] for pair in support_system_pairs]: - # only some ERNIE models support - self.use_system_prompt = True - if self.config.endpoint in [pair[1] for pair in support_system_pairs]: - self.use_system_prompt = True - - assert not (self.config.model and self.config.endpoint), "Only set `model` or `endpoint` in the config" - assert self.config.model or self.config.endpoint, "Should set one of `model` or `endpoint` in the config" - - self.token_costs = copy.deepcopy(QianFan_MODEL_TOKEN_COSTS) - self.token_costs.update(QianFan_EndPoint_TOKEN_COSTS) - - # self deployed model on the cloud not to calculate usage, it charges resource pool rental fee - self.calc_usage = self.config.calc_usage and self.config.endpoint is None - self.aclient: ChatCompletion = qianfan.ChatCompletion() - - def _const_kwargs(self, messages: list[dict], stream: bool = False) -> dict: - kwargs = { - "messages": messages, - "stream": stream, - } - if self.config.temperature > 0: - # different model has default temperature. only set when it's specified. - kwargs["temperature"] = self.config.temperature - if self.config.endpoint: - kwargs["endpoint"] = self.config.endpoint - elif self.config.model: - kwargs["model"] = self.config.model - - if self.use_system_prompt: - # if the model support system prompt, extract and pass it - if messages[0]["role"] == "system": - kwargs["messages"] = messages[1:] - kwargs["system"] = messages[0]["content"] # set system prompt here - return kwargs - - def _update_costs(self, usage: dict): - """update each request's token cost""" - model_or_endpoint = self.config.model if self.config.model else self.config.endpoint - local_calc_usage = True if model_or_endpoint in self.token_costs else False - super()._update_costs(usage, model_or_endpoint, local_calc_usage) - - def get_choice_text(self, resp: JsonBody) -> str: - return resp.get("result", "") - - def completion(self, messages: list[dict]) -> JsonBody: - resp = self.aclient.do(**self._const_kwargs(messages=messages, stream=False)) - self._update_costs(resp.body.get("usage", {})) - return resp.body - - async def _achat_completion(self, messages: list[dict]) -> JsonBody: - resp = await self.aclient.ado(**self._const_kwargs(messages=messages, stream=False)) - self._update_costs(resp.body.get("usage", {})) - return resp.body - - async def acompletion(self, messages: list[dict], timeout=3) -> JsonBody: - return await self._achat_completion(messages) - - async def _achat_completion_stream(self, messages: list[dict]) -> str: - resp = await self.aclient.ado(**self._const_kwargs(messages=messages, stream=True)) - collected_content = [] - usage = {} - async for chunk in resp: - content = chunk.body.get("result", "") - usage = chunk.body.get("usage", {}) - log_llm_stream(content) - collected_content.append(content) - log_llm_stream("\n") - - self._update_costs(usage) - full_content = "".join(collected_content) - return full_content - - @retry( - stop=stop_after_attempt(3), - wait=wait_random_exponential(min=1, max=60), - after=after_log(logger, logger.level("WARNING").name), - retry=retry_if_exception_type(ConnectionError), - retry_error_callback=log_and_reraise, - ) - async def acompletion_text(self, messages: list[dict], stream=False, timeout: int = 3) -> str: - if stream: - return await self._achat_completion_stream(messages) - resp = await self._achat_completion(messages) - return self.get_choice_text(resp) diff --git a/metagpt/provider/zhipuai_api.py b/metagpt/provider/zhipuai_api.py index 4cbee4038..9e8e5fb53 100644 --- a/metagpt/provider/zhipuai_api.py +++ b/metagpt/provider/zhipuai_api.py @@ -53,6 +53,16 @@ class ZhiPuAILLM(BaseLLM): kwargs = {"model": self.model, "messages": messages, "stream": stream, "temperature": 0.3} return kwargs + def _update_costs(self, usage: dict): + """update each request's token cost""" + if self.config.calc_usage: + try: + prompt_tokens = int(usage.get("prompt_tokens", 0)) + completion_tokens = int(usage.get("completion_tokens", 0)) + self.cost_manager.update_cost(prompt_tokens, completion_tokens, self.model) + except Exception as e: + logger.error(f"zhipuai updats costs failed! exp: {e}") + def completion(self, messages: list[dict], timeout=3) -> dict: resp: Completion = self.llm.chat.completions.create(**self._const_kwargs(messages)) usage = resp.usage.model_dump() diff --git a/metagpt/utils/cost_manager.py b/metagpt/utils/cost_manager.py index 4e6b65b2c..c4c93f91f 100644 --- a/metagpt/utils/cost_manager.py +++ b/metagpt/utils/cost_manager.py @@ -29,7 +29,6 @@ class CostManager(BaseModel): total_budget: float = 0 max_budget: float = 10.0 total_cost: float = 0 - token_costs: dict[str, dict[str, float]] = TOKEN_COSTS def update_cost(self, prompt_tokens, completion_tokens, model): """ @@ -47,8 +46,7 @@ class CostManager(BaseModel): return cost = ( - prompt_tokens * self.token_costs[model]["prompt"] - + completion_tokens * self.token_costs[model]["completion"] + prompt_tokens * TOKEN_COSTS[model]["prompt"] + completion_tokens * TOKEN_COSTS[model]["completion"] ) / 1000 self.total_cost += cost logger.info( diff --git a/metagpt/utils/token_counter.py b/metagpt/utils/token_counter.py index 2ec0edc99..65f5fe76f 100644 --- a/metagpt/utils/token_counter.py +++ b/metagpt/utils/token_counter.py @@ -38,59 +38,6 @@ TOKEN_COSTS = { } -""" -QianFan Token Price https://cloud.baidu.com/doc/WENXINWORKSHOP/s/hlrk4akp7#tokens%E5%90%8E%E4%BB%98%E8%B4%B9 -Due to QianFan has multi price strategies, we unify `Tokens post-payment` as a statistical method. -""" -QianFan_MODEL_TOKEN_COSTS = { - "ERNIE-Bot-4": {"prompt": 0.017, "completion": 0.017}, - "ERNIE-Bot-8k": {"prompt": 0.0034, "completion": 0.0067}, - "ERNIE-Bot": {"prompt": 0.017, "completion": 0.017}, - "ERNIE-Bot-turbo": {"prompt": 0.0011, "completion": 0.0011}, - "EB-turbo-AppBuilder": {"prompt": 0.0011, "completion": 0.0011}, - "ERNIE-Speed": {"prompt": 0.00056, "completion": 0.0011}, - "BLOOMZ-7B": {"prompt": 0.00056, "completion": 0.00056}, - "Llama-2-7B-Chat": {"prompt": 0.00056, "completion": 0.00056}, - "Llama-2-13B-Chat": {"prompt": 0.00084, "completion": 0.00084}, - "Llama-2-70B-Chat": {"prompt": 0.0049, "completion": 0.0049}, - "ChatGLM2-6B-32K": {"prompt": 0.00056, "completion": 0.00056}, - "AquilaChat-7B": {"prompt": 0.00056, "completion": 0.00056}, - "Mixtral-8x7B-Instruct": {"prompt": 0.0049, "completion": 0.0049}, - "SQLCoder-7B": {"prompt": 0.00056, "completion": 0.00056}, - "CodeLlama-7B-Instruct": {"prompt": 0.00056, "completion": 0.00056}, - "XuanYuan-70B-Chat-4bit": {"prompt": 0.0049, "completion": 0.0049}, - "Qianfan-BLOOMZ-7B-compressed": {"prompt": 0.00056, "completion": 0.00056}, - "Qianfan-Chinese-Llama-2-7B": {"prompt": 0.00056, "completion": 0.00056}, - "Qianfan-Chinese-Llama-2-13B": {"prompt": 0.00084, "completion": 0.00084}, - "ChatLaw": {"prompt": 0.0011, "completion": 0.0011}, - "Yi-34B-Chat": {"prompt": 0.0, "completion": 0.0}, -} - -QianFan_EndPoint_TOKEN_COSTS = { - "completions_pro": QianFan_MODEL_TOKEN_COSTS["ERNIE-Bot-4"], - "ernie_bot_8k": QianFan_MODEL_TOKEN_COSTS["ERNIE-Bot-8k"], - "completions": QianFan_MODEL_TOKEN_COSTS["ERNIE-Bot"], - "eb-instant": QianFan_MODEL_TOKEN_COSTS["ERNIE-Bot-turbo"], - "ai_apaas": QianFan_MODEL_TOKEN_COSTS["EB-turbo-AppBuilder"], - "ernie_speed": QianFan_MODEL_TOKEN_COSTS["ERNIE-Speed"], - "bloomz_7b1": QianFan_MODEL_TOKEN_COSTS["BLOOMZ-7B"], - "llama_2_7b": QianFan_MODEL_TOKEN_COSTS["Llama-2-7B-Chat"], - "llama_2_13b": QianFan_MODEL_TOKEN_COSTS["Llama-2-13B-Chat"], - "llama_2_70b": QianFan_MODEL_TOKEN_COSTS["Llama-2-70B-Chat"], - "chatglm2_6b_32k": QianFan_MODEL_TOKEN_COSTS["ChatGLM2-6B-32K"], - "aquilachat_7b": QianFan_MODEL_TOKEN_COSTS["AquilaChat-7B"], - "mixtral_8x7b_instruct": QianFan_MODEL_TOKEN_COSTS["Mixtral-8x7B-Instruct"], - "sqlcoder_7b": QianFan_MODEL_TOKEN_COSTS["SQLCoder-7B"], - "codellama_7b_instruct": QianFan_MODEL_TOKEN_COSTS["CodeLlama-7B-Instruct"], - "xuanyuan_70b_chat": QianFan_MODEL_TOKEN_COSTS["XuanYuan-70B-Chat-4bit"], - "qianfan_bloomz_7b_compressed": QianFan_MODEL_TOKEN_COSTS["Qianfan-BLOOMZ-7B-compressed"], - "qianfan_chinese_llama_2_7b": QianFan_MODEL_TOKEN_COSTS["Qianfan-Chinese-Llama-2-7B"], - "qianfan_chinese_llama_2_13b": QianFan_MODEL_TOKEN_COSTS["Qianfan-Chinese-Llama-2-13B"], - "chatlaw": QianFan_MODEL_TOKEN_COSTS["ChatLaw"], - "yi_34b_chat": QianFan_MODEL_TOKEN_COSTS["Yi-34B-Chat"], -} - - TOKEN_MAX = { "gpt-3.5-turbo": 4096, "gpt-3.5-turbo-0301": 4096, diff --git a/requirements.txt b/requirements.txt index b5d8d7d51..1426500ce 100644 --- a/requirements.txt +++ b/requirements.txt @@ -67,4 +67,3 @@ playwright>=1.26 # used at metagpt/tools/libs/web_scraping.py anytree ipywidgets==8.1.1 Pillow -qianfan==0.3.1 diff --git a/tests/metagpt/provider/mock_llm_config.py b/tests/metagpt/provider/mock_llm_config.py index e0afaa51e..e2f626a6a 100644 --- a/tests/metagpt/provider/mock_llm_config.py +++ b/tests/metagpt/provider/mock_llm_config.py @@ -42,15 +42,3 @@ mock_llm_config_zhipu = LLMConfig( model="mock_zhipu_model", proxy="http://localhost:8080", ) - - -mock_llm_config_spark = LLMConfig( - api_type="spark", - app_id="xxx", - api_key="xxx", - api_secret="xxx", - domain="generalv2", - base_url="wss://spark-api.xf-yun.com/v3.1/chat", -) - -mock_llm_config_qianfan = LLMConfig(api_type="qianfan", access_key="xxx", secret_key="xxx", model="ERNIE-Bot-turbo") diff --git a/tests/metagpt/provider/req_resp_const.py b/tests/metagpt/provider/req_resp_const.py deleted file mode 100644 index 73939e1c6..000000000 --- a/tests/metagpt/provider/req_resp_const.py +++ /dev/null @@ -1,117 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Desc : default request & response data for provider unittest - - -from openai.types.chat.chat_completion import ( - ChatCompletion, - ChatCompletionMessage, - Choice, -) -from openai.types.chat.chat_completion_chunk import ChatCompletionChunk -from openai.types.chat.chat_completion_chunk import Choice as AChoice -from openai.types.chat.chat_completion_chunk import ChoiceDelta -from openai.types.completion_usage import CompletionUsage -from qianfan.resources.typing import QfResponse - -from metagpt.provider.base_llm import BaseLLM - -prompt = "who are you?" -messages = [{"role": "user", "content": prompt}] - -resp_cont_tmpl = "I'm {name}" -default_resp_cont = resp_cont_tmpl.format(name="GPT") - - -# part of whole ChatCompletion of openai like structure -def get_part_chat_completion(name: str) -> dict: - part_chat_completion = { - "choices": [ - { - "index": 0, - "message": { - "role": "assistant", - "content": resp_cont_tmpl.format(name=name), - }, - "finish_reason": "stop", - } - ], - "usage": {"completion_tokens": 22, "prompt_tokens": 19, "total_tokens": 41}, - } - return part_chat_completion - - -def get_openai_chat_completion(name: str) -> ChatCompletion: - openai_chat_completion = ChatCompletion( - id="cmpl-a6652c1bb181caae8dd19ad8", - model="xx/xxx", - object="chat.completion", - created=1703300855, - choices=[ - Choice( - finish_reason="stop", - index=0, - message=ChatCompletionMessage(role="assistant", content=resp_cont_tmpl.format(name=name)), - logprobs=None, - ) - ], - usage=CompletionUsage(completion_tokens=110, prompt_tokens=92, total_tokens=202), - ) - return openai_chat_completion - - -def get_openai_chat_completion_chunk(name: str, usage_as_dict: bool = False) -> ChatCompletionChunk: - usage = CompletionUsage(completion_tokens=110, prompt_tokens=92, total_tokens=202) - usage = usage if not usage_as_dict else usage.model_dump() - openai_chat_completion_chunk = ChatCompletionChunk( - id="cmpl-a6652c1bb181caae8dd19ad8", - model="xx/xxx", - object="chat.completion.chunk", - created=1703300855, - choices=[ - AChoice( - delta=ChoiceDelta(role="assistant", content=resp_cont_tmpl.format(name=name)), - finish_reason="stop", - index=0, - logprobs=None, - ) - ], - usage=usage, - ) - return openai_chat_completion_chunk - - -# For gemini -gemini_messages = [{"role": "user", "parts": prompt}] - - -# For QianFan -qf_jsonbody_dict = { - "id": "as-4v1h587fyv", - "object": "chat.completion", - "created": 1695021339, - "result": "", - "is_truncated": False, - "need_clear_history": False, - "usage": {"prompt_tokens": 7, "completion_tokens": 15, "total_tokens": 22}, -} - - -def get_qianfan_response(name: str) -> QfResponse: - qf_jsonbody_dict["result"] = resp_cont_tmpl.format(name=name) - return QfResponse(code=200, body=qf_jsonbody_dict) - - -# For llm general chat functions call -async def llm_general_chat_funcs_test(llm: BaseLLM, prompt: str, messages: list[dict], resp_cont: str): - resp = await llm.aask(prompt, stream=False) - assert resp == resp_cont - - resp = await llm.aask(prompt) - assert resp == resp_cont - - resp = await llm.acompletion_text(messages, stream=False) - assert resp == resp_cont - - resp = await llm.acompletion_text(messages, stream=True) - assert resp == resp_cont diff --git a/tests/metagpt/provider/test_anthropic_api.py b/tests/metagpt/provider/test_anthropic_api.py index 93cfd7dbc..6962ab064 100644 --- a/tests/metagpt/provider/test_anthropic_api.py +++ b/tests/metagpt/provider/test_anthropic_api.py @@ -8,25 +8,25 @@ from anthropic.resources.completions import Completion from metagpt.provider.anthropic_api import Claude2 from tests.metagpt.provider.mock_llm_config import mock_llm_config -from tests.metagpt.provider.req_resp_const import prompt, resp_cont_tmpl -resp_cont = resp_cont_tmpl.format(name="Claude") +prompt = "who are you" +resp = "I'am Claude2" def mock_anthropic_completions_create(self, model: str, prompt: str, max_tokens_to_sample: int) -> Completion: - return Completion(id="xx", completion=resp_cont, model="claude-2", stop_reason="stop_sequence", type="completion") + return Completion(id="xx", completion=resp, model="claude-2", stop_reason="stop_sequence", type="completion") async def mock_anthropic_acompletions_create(self, model: str, prompt: str, max_tokens_to_sample: int) -> Completion: - return Completion(id="xx", completion=resp_cont, model="claude-2", stop_reason="stop_sequence", type="completion") + return Completion(id="xx", completion=resp, model="claude-2", stop_reason="stop_sequence", type="completion") def test_claude2_ask(mocker): mocker.patch("anthropic.resources.completions.Completions.create", mock_anthropic_completions_create) - assert resp_cont == Claude2(mock_llm_config).ask(prompt) + assert resp == Claude2(mock_llm_config).ask(prompt) @pytest.mark.asyncio async def test_claude2_aask(mocker): mocker.patch("anthropic.resources.completions.AsyncCompletions.create", mock_anthropic_acompletions_create) - assert resp_cont == await Claude2(mock_llm_config).aask(prompt) + assert resp == await Claude2(mock_llm_config).aask(prompt) diff --git a/tests/metagpt/provider/test_base_llm.py b/tests/metagpt/provider/test_base_llm.py index cf44343bc..cc781f78a 100644 --- a/tests/metagpt/provider/test_base_llm.py +++ b/tests/metagpt/provider/test_base_llm.py @@ -11,13 +11,21 @@ import pytest from metagpt.configs.llm_config import LLMConfig from metagpt.provider.base_llm import BaseLLM from metagpt.schema import Message -from tests.metagpt.provider.req_resp_const import ( - default_resp_cont, - get_part_chat_completion, - prompt, -) -name = "GPT" +default_chat_resp = { + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": "I'am GPT", + }, + "finish_reason": "stop", + } + ] +} +prompt_msg = "who are you" +resp_content = default_chat_resp["choices"][0]["message"]["content"] class MockBaseLLM(BaseLLM): @@ -25,13 +33,16 @@ class MockBaseLLM(BaseLLM): pass def completion(self, messages: list[dict], timeout=3): - return get_part_chat_completion(name) + return default_chat_resp async def acompletion(self, messages: list[dict], timeout=3): - return get_part_chat_completion(name) + return default_chat_resp async def acompletion_text(self, messages: list[dict], stream=False, timeout=3) -> str: - return default_resp_cont + return resp_content + + async def close(self): + return default_chat_resp def test_base_llm(): @@ -75,25 +86,25 @@ def test_base_llm(): choice_text = base_llm.get_choice_text(openai_funccall_resp) assert choice_text == openai_funccall_resp["choices"][0]["message"]["content"] - # resp = base_llm.ask(prompt) - # assert resp == default_resp_cont + # resp = base_llm.ask(prompt_msg) + # assert resp == resp_content - # resp = base_llm.ask_batch([prompt]) - # assert resp == default_resp_cont + # resp = base_llm.ask_batch([prompt_msg]) + # assert resp == resp_content - # resp = base_llm.ask_code([prompt]) - # assert resp == default_resp_cont + # resp = base_llm.ask_code([prompt_msg]) + # assert resp == resp_content @pytest.mark.asyncio async def test_async_base_llm(): base_llm = MockBaseLLM() - resp = await base_llm.aask(prompt) - assert resp == default_resp_cont + resp = await base_llm.aask(prompt_msg) + assert resp == resp_content - resp = await base_llm.aask_batch([prompt]) - assert resp == default_resp_cont + resp = await base_llm.aask_batch([prompt_msg]) + assert resp == resp_content - # resp = await base_llm.aask_code([prompt]) - # assert resp == default_resp_cont + # resp = await base_llm.aask_code([prompt_msg]) + # assert resp == resp_content diff --git a/tests/metagpt/provider/test_fireworks_llm.py b/tests/metagpt/provider/test_fireworks_llm.py index 1c1aa9caa..66b55e5b2 100644 --- a/tests/metagpt/provider/test_fireworks_llm.py +++ b/tests/metagpt/provider/test_fireworks_llm.py @@ -3,7 +3,14 @@ # @Desc : the unittest of fireworks api import pytest +from openai.types.chat.chat_completion import ( + ChatCompletion, + ChatCompletionMessage, + Choice, +) from openai.types.chat.chat_completion_chunk import ChatCompletionChunk +from openai.types.chat.chat_completion_chunk import Choice as AChoice +from openai.types.chat.chat_completion_chunk import ChoiceDelta from openai.types.completion_usage import CompletionUsage from metagpt.provider.fireworks_api import ( @@ -13,19 +20,42 @@ from metagpt.provider.fireworks_api import ( ) from metagpt.utils.cost_manager import Costs from tests.metagpt.provider.mock_llm_config import mock_llm_config -from tests.metagpt.provider.req_resp_const import ( - get_openai_chat_completion, - get_openai_chat_completion_chunk, - llm_general_chat_funcs_test, - messages, - prompt, - resp_cont_tmpl, + +resp_content = "I'm fireworks" +default_resp = ChatCompletion( + id="cmpl-a6652c1bb181caae8dd19ad8", + model="accounts/fireworks/models/llama-v2-13b-chat", + object="chat.completion", + created=1703300855, + choices=[ + Choice( + finish_reason="stop", + index=0, + message=ChatCompletionMessage(role="assistant", content=resp_content), + logprobs=None, + ) + ], + usage=CompletionUsage(completion_tokens=110, prompt_tokens=92, total_tokens=202), ) -name = "fireworks" -resp_cont = resp_cont_tmpl.format(name=name) -default_resp = get_openai_chat_completion(name) -default_resp_chunk = get_openai_chat_completion_chunk(name, usage_as_dict=True) +default_resp_chunk = ChatCompletionChunk( + id=default_resp.id, + model=default_resp.model, + object="chat.completion.chunk", + created=default_resp.created, + choices=[ + AChoice( + delta=ChoiceDelta(content=resp_content, role="assistant"), + finish_reason="stop", + index=0, + logprobs=None, + ) + ], + usage=dict(default_resp.usage), +) + +prompt_msg = "who are you" +messages = [{"role": "user", "content": prompt_msg}] def test_fireworks_costmanager(): @@ -58,17 +88,27 @@ async def mock_openai_acompletions_create(self, stream: bool = False, **kwargs) async def test_fireworks_acompletion(mocker): mocker.patch("openai.resources.chat.completions.AsyncCompletions.create", mock_openai_acompletions_create) - fireworks_llm = FireworksLLM(mock_llm_config) - fireworks_llm.model = "llama-v2-13b-chat" + fireworks_gpt = FireworksLLM(mock_llm_config) + fireworks_gpt.model = "llama-v2-13b-chat" - fireworks_llm._update_costs( + fireworks_gpt._update_costs( usage=CompletionUsage(prompt_tokens=500000, completion_tokens=500000, total_tokens=1000000) ) - assert fireworks_llm.get_costs() == Costs( + assert fireworks_gpt.get_costs() == Costs( total_prompt_tokens=500000, total_completion_tokens=500000, total_cost=0.5, total_budget=0 ) - resp = await fireworks_llm.acompletion(messages) - assert resp.choices[0].message.content in resp_cont + resp = await fireworks_gpt.acompletion(messages) + assert resp.choices[0].message.content in resp_content - await llm_general_chat_funcs_test(fireworks_llm, prompt, messages, resp_cont) + resp = await fireworks_gpt.aask(prompt_msg, stream=False) + assert resp == resp_content + + resp = await fireworks_gpt.acompletion_text(messages, stream=False) + assert resp == resp_content + + resp = await fireworks_gpt.acompletion_text(messages, stream=True) + assert resp == resp_content + + resp = await fireworks_gpt.aask(prompt_msg) + assert resp == resp_content diff --git a/tests/metagpt/provider/test_google_gemini_api.py b/tests/metagpt/provider/test_google_gemini_api.py index 50c15ee19..404ae1e90 100644 --- a/tests/metagpt/provider/test_google_gemini_api.py +++ b/tests/metagpt/provider/test_google_gemini_api.py @@ -11,12 +11,6 @@ from google.generativeai.types import content_types from metagpt.provider.google_gemini_api import GeminiLLM from tests.metagpt.provider.mock_llm_config import mock_llm_config -from tests.metagpt.provider.req_resp_const import ( - gemini_messages, - llm_general_chat_funcs_test, - prompt, - resp_cont_tmpl, -) @dataclass @@ -24,8 +18,10 @@ class MockGeminiResponse(ABC): text: str -resp_cont = resp_cont_tmpl.format(name="gemini") -default_resp = MockGeminiResponse(text=resp_cont) +prompt_msg = "who are you" +messages = [{"role": "user", "parts": prompt_msg}] +resp_content = "I'm gemini from google" +default_resp = MockGeminiResponse(text=resp_content) def mock_gemini_count_tokens(self, contents: content_types.ContentsType) -> glm.CountTokensResponse: @@ -64,18 +60,28 @@ async def test_gemini_acompletion(mocker): mock_gemini_generate_content_async, ) - gemini_llm = GeminiLLM(mock_llm_config) + gemini_gpt = GeminiLLM(mock_llm_config) - assert gemini_llm._user_msg(prompt) == {"role": "user", "parts": [prompt]} - assert gemini_llm._assistant_msg(prompt) == {"role": "model", "parts": [prompt]} + assert gemini_gpt._user_msg(prompt_msg) == {"role": "user", "parts": [prompt_msg]} + assert gemini_gpt._assistant_msg(prompt_msg) == {"role": "model", "parts": [prompt_msg]} - usage = gemini_llm.get_usage(gemini_messages, resp_cont) + usage = gemini_gpt.get_usage(messages, resp_content) assert usage == {"prompt_tokens": 20, "completion_tokens": 20} - resp = gemini_llm.completion(gemini_messages) + resp = gemini_gpt.completion(messages) assert resp == default_resp - resp = await gemini_llm.acompletion(gemini_messages) + resp = await gemini_gpt.acompletion(messages) assert resp.text == default_resp.text - await llm_general_chat_funcs_test(gemini_llm, prompt, gemini_messages, resp_cont) + resp = await gemini_gpt.aask(prompt_msg, stream=False) + assert resp == resp_content + + resp = await gemini_gpt.acompletion_text(messages, stream=False) + assert resp == resp_content + + resp = await gemini_gpt.acompletion_text(messages, stream=True) + assert resp == resp_content + + resp = await gemini_gpt.aask(prompt_msg) + assert resp == resp_content diff --git a/tests/metagpt/provider/test_ollama_api.py b/tests/metagpt/provider/test_ollama_api.py index af2e929e9..5d942598b 100644 --- a/tests/metagpt/provider/test_ollama_api.py +++ b/tests/metagpt/provider/test_ollama_api.py @@ -9,15 +9,12 @@ import pytest from metagpt.provider.ollama_api import OllamaLLM from tests.metagpt.provider.mock_llm_config import mock_llm_config -from tests.metagpt.provider.req_resp_const import ( - llm_general_chat_funcs_test, - messages, - prompt, - resp_cont_tmpl, -) -resp_cont = resp_cont_tmpl.format(name="ollama") -default_resp = {"message": {"role": "assistant", "content": resp_cont}} +prompt_msg = "who are you" +messages = [{"role": "user", "content": prompt_msg}] + +resp_content = "I'm ollama" +default_resp = {"message": {"role": "assistant", "content": resp_content}} async def mock_ollama_arequest(self, stream: bool = False, **kwargs) -> Tuple[Any, Any, bool]: @@ -44,12 +41,19 @@ async def mock_ollama_arequest(self, stream: bool = False, **kwargs) -> Tuple[An async def test_gemini_acompletion(mocker): mocker.patch("metagpt.provider.general_api_requestor.GeneralAPIRequestor.arequest", mock_ollama_arequest) - ollama_llm = OllamaLLM(mock_llm_config) + ollama_gpt = OllamaLLM(mock_llm_config) - resp = await ollama_llm.acompletion(messages) + resp = await ollama_gpt.acompletion(messages) assert resp["message"]["content"] == default_resp["message"]["content"] - resp = await ollama_llm.aask(prompt, stream=False) - assert resp == resp_cont + resp = await ollama_gpt.aask(prompt_msg, stream=False) + assert resp == resp_content - await llm_general_chat_funcs_test(ollama_llm, prompt, messages, resp_cont) + resp = await ollama_gpt.acompletion_text(messages, stream=False) + assert resp == resp_content + + resp = await ollama_gpt.acompletion_text(messages, stream=True) + assert resp == resp_content + + resp = await ollama_gpt.aask(prompt_msg) + assert resp == resp_content diff --git a/tests/metagpt/provider/test_open_llm_api.py b/tests/metagpt/provider/test_open_llm_api.py index aa38b95a6..fc7b510cc 100644 --- a/tests/metagpt/provider/test_open_llm_api.py +++ b/tests/metagpt/provider/test_open_llm_api.py @@ -3,26 +3,53 @@ # @Desc : import pytest +from openai.types.chat.chat_completion import ( + ChatCompletion, + ChatCompletionMessage, + Choice, +) from openai.types.chat.chat_completion_chunk import ChatCompletionChunk +from openai.types.chat.chat_completion_chunk import Choice as AChoice +from openai.types.chat.chat_completion_chunk import ChoiceDelta from openai.types.completion_usage import CompletionUsage from metagpt.provider.open_llm_api import OpenLLM -from metagpt.utils.cost_manager import CostManager, Costs +from metagpt.utils.cost_manager import Costs from tests.metagpt.provider.mock_llm_config import mock_llm_config -from tests.metagpt.provider.req_resp_const import ( - get_openai_chat_completion, - get_openai_chat_completion_chunk, - llm_general_chat_funcs_test, - messages, - prompt, - resp_cont_tmpl, + +resp_content = "I'm llama2" +default_resp = ChatCompletion( + id="cmpl-a6652c1bb181caae8dd19ad8", + model="llama-v2-13b-chat", + object="chat.completion", + created=1703302755, + choices=[ + Choice( + finish_reason="stop", + index=0, + message=ChatCompletionMessage(role="assistant", content=resp_content), + logprobs=None, + ) + ], ) -name = "llama2-7b" -resp_cont = resp_cont_tmpl.format(name=name) -default_resp = get_openai_chat_completion(name) +default_resp_chunk = ChatCompletionChunk( + id=default_resp.id, + model=default_resp.model, + object="chat.completion.chunk", + created=default_resp.created, + choices=[ + AChoice( + delta=ChoiceDelta(content=resp_content, role="assistant"), + finish_reason="stop", + index=0, + logprobs=None, + ) + ], +) -default_resp_chunk = get_openai_chat_completion_chunk(name) +prompt_msg = "who are you" +messages = [{"role": "user", "content": prompt_msg}] async def mock_openai_acompletions_create(self, stream: bool = False, **kwargs) -> ChatCompletionChunk: @@ -41,16 +68,25 @@ async def mock_openai_acompletions_create(self, stream: bool = False, **kwargs) async def test_openllm_acompletion(mocker): mocker.patch("openai.resources.chat.completions.AsyncCompletions.create", mock_openai_acompletions_create) - openllm_llm = OpenLLM(mock_llm_config) - openllm_llm.model = "llama-v2-13b-chat" + openllm_gpt = OpenLLM(mock_llm_config) + openllm_gpt.model = "llama-v2-13b-chat" - openllm_llm.cost_manager = CostManager() - openllm_llm._update_costs(usage=CompletionUsage(prompt_tokens=100, completion_tokens=100, total_tokens=200)) - assert openllm_llm.get_costs() == Costs( + openllm_gpt._update_costs(usage=CompletionUsage(prompt_tokens=100, completion_tokens=100, total_tokens=200)) + assert openllm_gpt.get_costs() == Costs( total_prompt_tokens=100, total_completion_tokens=100, total_cost=0, total_budget=0 ) - resp = await openllm_llm.acompletion(messages) - assert resp.choices[0].message.content in resp_cont + resp = await openllm_gpt.acompletion(messages) + assert resp.choices[0].message.content in resp_content - await llm_general_chat_funcs_test(openllm_llm, prompt, messages, resp_cont) + resp = await openllm_gpt.aask(prompt_msg, stream=False) + assert resp == resp_content + + resp = await openllm_gpt.acompletion_text(messages, stream=False) + assert resp == resp_content + + resp = await openllm_gpt.acompletion_text(messages, stream=True) + assert resp == resp_content + + resp = await openllm_gpt.aask(prompt_msg) + assert resp == resp_content diff --git a/tests/metagpt/provider/test_qianfan_api.py b/tests/metagpt/provider/test_qianfan_api.py deleted file mode 100644 index 28341425c..000000000 --- a/tests/metagpt/provider/test_qianfan_api.py +++ /dev/null @@ -1,56 +0,0 @@ -#!/usr/bin/env python -# -*- coding: utf-8 -*- -# @Desc : the unittest of qianfan api - -from typing import AsyncIterator, Union - -import pytest -from qianfan.resources.typing import JsonBody, QfResponse - -from metagpt.provider.qianfan_api import QianFanLLM -from tests.metagpt.provider.mock_llm_config import mock_llm_config_qianfan -from tests.metagpt.provider.req_resp_const import ( - get_qianfan_response, - llm_general_chat_funcs_test, - messages, - prompt, - resp_cont_tmpl, -) - -name = "ERNIE-Bot-turbo" -resp_cont = resp_cont_tmpl.format(name=name) - - -def mock_qianfan_do(self, messages: list[dict], model: str, stream: bool = False, system: str = None) -> QfResponse: - return get_qianfan_response(name=name) - - -async def mock_qianfan_ado( - self, messages: list[dict], model: str, stream: bool = True, system: str = None -) -> Union[QfResponse, AsyncIterator[QfResponse]]: - resps = [get_qianfan_response(name=name)] - if stream: - - async def aresp_iterator(resps: list[JsonBody]): - for resp in resps: - yield resp - - return aresp_iterator(resps) - else: - return resps[0] - - -@pytest.mark.asyncio -async def test_qianfan_acompletion(mocker): - mocker.patch("qianfan.resources.llm.chat_completion.ChatCompletion.do", mock_qianfan_do) - mocker.patch("qianfan.resources.llm.chat_completion.ChatCompletion.ado", mock_qianfan_ado) - - qianfan_llm = QianFanLLM(mock_llm_config_qianfan) - - resp = qianfan_llm.completion(messages) - assert resp.get("result") == resp_cont - - resp = await qianfan_llm.acompletion(messages) - assert resp.get("result") == resp_cont - - await llm_general_chat_funcs_test(qianfan_llm, prompt, messages, resp_cont) diff --git a/tests/metagpt/provider/test_spark_api.py b/tests/metagpt/provider/test_spark_api.py index 9c278267d..f5a6f66fd 100644 --- a/tests/metagpt/provider/test_spark_api.py +++ b/tests/metagpt/provider/test_spark_api.py @@ -4,18 +4,12 @@ import pytest +from metagpt.config2 import Config from metagpt.provider.spark_api import GetMessageFromWeb, SparkLLM -from tests.metagpt.provider.mock_llm_config import ( - mock_llm_config, - mock_llm_config_spark, -) -from tests.metagpt.provider.req_resp_const import ( - llm_general_chat_funcs_test, - prompt, - resp_cont_tmpl, -) +from tests.metagpt.provider.mock_llm_config import mock_llm_config -resp_cont = resp_cont_tmpl.format(name="Spark") +prompt_msg = "who are you" +resp_content = "I'm Spark" class MockWebSocketApp(object): @@ -29,7 +23,7 @@ class MockWebSocketApp(object): def test_get_msg_from_web(mocker): mocker.patch("websocket.WebSocketApp", MockWebSocketApp) - get_msg_from_web = GetMessageFromWeb(prompt, mock_llm_config) + get_msg_from_web = GetMessageFromWeb(prompt_msg, mock_llm_config) assert get_msg_from_web.gen_params()["parameter"]["chat"]["domain"] == "mock_domain" ret = get_msg_from_web.run() @@ -37,26 +31,34 @@ def test_get_msg_from_web(mocker): def mock_spark_get_msg_from_web_run(self) -> str: - return resp_cont + return resp_content @pytest.mark.asyncio -async def test_spark_aask(mocker): - mocker.patch("metagpt.provider.spark_api.GetMessageFromWeb.run", mock_spark_get_msg_from_web_run) - - llm = SparkLLM(mock_llm_config_spark) +async def test_spark_aask(): + llm = SparkLLM(Config.from_home("spark.yaml").llm) resp = await llm.aask("Hello!") - assert resp == resp_cont + print(resp) @pytest.mark.asyncio async def test_spark_acompletion(mocker): mocker.patch("metagpt.provider.spark_api.GetMessageFromWeb.run", mock_spark_get_msg_from_web_run) - spark_llm = SparkLLM(mock_llm_config) + spark_gpt = SparkLLM(mock_llm_config) - resp = await spark_llm.acompletion([]) - assert resp == resp_cont + resp = await spark_gpt.acompletion([]) + assert resp == resp_content - await llm_general_chat_funcs_test(spark_llm, prompt, prompt, resp_cont) + resp = await spark_gpt.aask(prompt_msg, stream=False) + assert resp == resp_content + + resp = await spark_gpt.acompletion_text([], stream=False) + assert resp == resp_content + + resp = await spark_gpt.acompletion_text([], stream=True) + assert resp == resp_content + + resp = await spark_gpt.aask(prompt_msg) + assert resp == resp_content diff --git a/tests/metagpt/provider/test_zhipuai_api.py b/tests/metagpt/provider/test_zhipuai_api.py index c51010122..ad2ececa2 100644 --- a/tests/metagpt/provider/test_zhipuai_api.py +++ b/tests/metagpt/provider/test_zhipuai_api.py @@ -6,24 +6,22 @@ import pytest from metagpt.provider.zhipuai_api import ZhiPuAILLM from tests.metagpt.provider.mock_llm_config import mock_llm_config_zhipu -from tests.metagpt.provider.req_resp_const import ( - get_part_chat_completion, - llm_general_chat_funcs_test, - messages, - prompt, - resp_cont_tmpl, -) -name = "ChatGLM-4" -resp_cont = resp_cont_tmpl.format(name=name) -default_resp = get_part_chat_completion(name) +prompt_msg = "who are you" +messages = [{"role": "user", "content": prompt_msg}] + +resp_content = "I'm chatglm-turbo" +default_resp = { + "choices": [{"finish_reason": "stop", "index": 0, "message": {"content": resp_content, "role": "assistant"}}], + "usage": {"completion_tokens": 22, "prompt_tokens": 19, "total_tokens": 41}, +} async def mock_zhipuai_acreate_stream(self, **kwargs): class MockResponse(object): async def _aread(self): class Iterator(object): - events = [{"choices": [{"index": 0, "delta": {"content": resp_cont, "role": "assistant"}}]}] + events = [{"choices": [{"index": 0, "delta": {"content": resp_content, "role": "assistant"}}]}] async def __aiter__(self): for event in self.events: @@ -48,12 +46,22 @@ async def test_zhipuai_acompletion(mocker): mocker.patch("metagpt.provider.zhipuai.zhipu_model_api.ZhiPuModelAPI.acreate", mock_zhipuai_acreate) mocker.patch("metagpt.provider.zhipuai.zhipu_model_api.ZhiPuModelAPI.acreate_stream", mock_zhipuai_acreate_stream) - zhipu_llm = ZhiPuAILLM(mock_llm_config_zhipu) + zhipu_gpt = ZhiPuAILLM(mock_llm_config_zhipu) - resp = await zhipu_llm.acompletion(messages) - assert resp["choices"][0]["message"]["content"] == resp_cont + resp = await zhipu_gpt.acompletion(messages) + assert resp["choices"][0]["message"]["content"] == resp_content - await llm_general_chat_funcs_test(zhipu_llm, prompt, messages, resp_cont) + resp = await zhipu_gpt.aask(prompt_msg, stream=False) + assert resp == resp_content + + resp = await zhipu_gpt.acompletion_text(messages, stream=False) + assert resp == resp_content + + resp = await zhipu_gpt.acompletion_text(messages, stream=True) + assert resp == resp_content + + resp = await zhipu_gpt.aask(prompt_msg) + assert resp == resp_content def test_zhipuai_proxy(): diff --git a/tests/spark.yaml b/tests/spark.yaml new file mode 100644 index 000000000..a5bbd98bd --- /dev/null +++ b/tests/spark.yaml @@ -0,0 +1,7 @@ +llm: + api_type: "spark" + app_id: "xxx" + api_key: "xxx" + api_secret: "xxx" + domain: "generalv2" + base_url: "wss://spark-api.xf-yun.com/v3.1/chat" \ No newline at end of file

CN doc EN doc JA doc -Discord Follow License: MIT roadmap +Discord Follow Twitter Follow