From a150add41f4c7b339baba4399ee28e5d58c62b28 Mon Sep 17 00:00:00 2001 From: Travis Dent Date: Thu, 14 Nov 2024 15:28:56 -0800 Subject: [PATCH 1/7] Implement tool removal command --- agentstack/generation/__init__.py | 2 +- agentstack/generation/tool_generation.py | 84 +++++++++++++++++------- agentstack/main.py | 6 ++ 3 files changed, 69 insertions(+), 23 deletions(-) diff --git a/agentstack/generation/__init__.py b/agentstack/generation/__init__.py index 41bbc9c..b9aa094 100644 --- a/agentstack/generation/__init__.py +++ b/agentstack/generation/__init__.py @@ -1,3 +1,3 @@ from .agent_generation import generate_agent from .task_generation import generate_task -from .tool_generation import add_tool +from .tool_generation import add_tool, remove_tool diff --git a/agentstack/generation/tool_generation.py b/agentstack/generation/tool_generation.py index c50c48b..e2c7997 100644 --- a/agentstack/generation/tool_generation.py +++ b/agentstack/generation/tool_generation.py @@ -9,30 +9,41 @@ import shutil import fileinput +TOOL_INIT_FILENAME = "src/tools/__init__.py" +AGENTSTACK_JSON_FILENAME = "agentstack.json" + def add_tool(tool_name: str, path: Optional[str] = None): + if path: + path = path.endswith('/') and path or path + '/' + else: + path = './' with importlib.resources.path(f'agentstack.tools', 'tools.json') as tools_data_path: tools = open_json_file(tools_data_path) framework = get_framework(path) assert_tool_exists(tool_name, tools) + agentstack_json = open_json_file(f'{path}{AGENTSTACK_JSON_FILENAME}') + + if tool_name in agentstack_json.get('tools', []): + print(f"\033[31mTool '{tool_name}' is already installed\033[0m") + sys.exit(1) with importlib.resources.path(f'agentstack.tools', f"{tool_name}.json") as tool_data_path: tool_data = open_json_file(tool_data_path) with importlib.resources.path(f'agentstack.templates.{framework}.tools', f"{tool_name}_tool.py") as tool_file_path: os.system(tool_data['package']) # Install package - shutil.copy(tool_file_path, f'{path + "/" if path else ""}src/tools/{tool_name}_tool.py') # Move tool from package to project + shutil.copy(tool_file_path, f'{path}src/tools/{tool_name}_tool.py') # Move tool from package to project add_tool_to_tools_init(tool_data, path) # Export tool from tools dir - add_tool_to_agent_definition(framework, tool_data, path) - insert_code_after_tag(f'{path + "/" if path else ""}.env', '# Tools', [tool_data['env']], next_line=True) # Add env var - insert_code_after_tag(f'{path + "/" if path else ""}.env.example', '# Tools', [tool_data['env']], next_line=True) # Add env var - - agentstack_json = open_json_file(f'{path + "/" if path else ""}agentstack.json') + # TODO If the tool is reinstalled .env variables will be duplicated. + insert_code_after_tag(f'{path}.env', '# Tools', [tool_data['env']], next_line=True) # Add env var + insert_code_after_tag(f'{path}.env.example', '# Tools', [tool_data['env']], next_line=True) # Add env var + if not agentstack_json.get('tools'): agentstack_json['tools'] = [] agentstack_json['tools'].append(tool_name) - with open(f'{path + "/" if path else ""}agentstack.json', 'w') as f: + with open(f'{path}{AGENTSTACK_JSON_FILENAME}', 'w') as f: json.dump(agentstack_json, f, indent=4) print(term_color(f'🔨 Tool {tool_name} added to agentstack project successfully', 'green')) @@ -40,26 +51,55 @@ def add_tool(tool_name: str, path: Optional[str] = None): print(term_color(f'đŸĒŠ {tool_data["cta"]}', 'blue')) -def add_tool_to_tools_init(tool_data: dict, path: Optional[str] = None): - file_path = f'{path + "/" if path else ""}src/tools/__init__.py' - tag = '# tool import' - code_to_insert = [ - f"from .{tool_data['name']}_tool import {', '.join([tool_name for tool_name in tool_data['tools']])}" - ] - insert_code_after_tag(file_path, tag, code_to_insert, next_line=True) +def remove_tool(tool_name: str, path: Optional[str] = None): + if path: + path = path.endswith('/') and path or path + '/' + else: + path = './' + with importlib.resources.path(f'agentstack.tools', 'tools.json') as tools_data_path: + tools = open_json_file(tools_data_path) + framework = get_framework() + assert_tool_exists(tool_name, tools) + agentstack_json = open_json_file(f'{path}{AGENTSTACK_JSON_FILENAME}') + + if not tool_name in agentstack_json.get('tools', []): + print(f"\033[31mTool '{tool_name}' is not installed\033[0m") + sys.exit(1) + with importlib.resources.path(f'agentstack.tools', f"{tool_name}.json") as tool_data_path: + tool_data = open_json_file(tool_data_path) + os.remove(f'{path}src/tools/{tool_name}_tool.py') + remove_tool_from_tools_init(tool_data, path) + # We don't remove the .env variables to preserve user data. + + agentstack_json['tools'].remove(tool_name) + with open(f'{path}{AGENTSTACK_JSON_FILENAME}', 'w') as f: + json.dump(agentstack_json, f, indent=4) + + print(term_color(f'🔨 Tool {tool_name} removed from agentstack project successfully', 'green')) + # TODO If we add an uninstall command to the tool_config json we can run it instead. + print(term_color(f' To uninstall dependencies, reverse the install command: {tool_data["package"]}', 'blue')) -def add_tool_to_agent_definition(framework: str, tool_data: dict, path: Optional[str] = None): - filename = '' - if framework == 'crewai': - filename = 'src/crew.py' - if path: - filename = f'{path}/{filename}' +def _format_tool_import_statement(tool_data: dict): + return f"from .{tool_data['name']}_tool import {', '.join([tool_name for tool_name in tool_data['tools']])}" + + +def add_tool_to_tools_init(tool_data: dict, path: str = ''): + file_path = f'{path}{TOOL_INIT_FILENAME}' + tag = '# tool import' + code_to_insert = [_format_tool_import_statement(tool_data), ] + insert_code_after_tag(file_path, tag, code_to_insert, next_line=True) + - with fileinput.input(files=filename, inplace=True) as f: +def remove_tool_from_tools_init(tool_data: dict, path: str = ''): + """Search for the import statement in the init and remove it.""" + file_path = f'{path}{TOOL_INIT_FILENAME}' + import_statement = _format_tool_import_statement(tool_data) + with fileinput.input(files=file_path, inplace=True) as f: for line in f: - print(line.replace('tools=[', f'tools=[{"*" if tool_data.get("tools_bundled") else ""}tools.{", tools.".join([tool_name for tool_name in tool_data["tools"]])}, '), end='') + if line.strip() != import_statement: + print(line, end='') def assert_tool_exists(tool_name: str, tools: dict): diff --git a/agentstack/main.py b/agentstack/main.py index 09c7d41..fdc6180 100644 --- a/agentstack/main.py +++ b/agentstack/main.py @@ -62,6 +62,10 @@ def main(): tools_add_parser = tools_subparsers.add_parser('add', aliases=['a'], help='Add a new tool') tools_add_parser.add_argument('name', help='Name of the tool to add') + # 'remove' command under 'tools' + tools_remove_parser = tools_subparsers.add_parser('remove', aliases=['r'], help='Remove a tool') + tools_remove_parser.add_argument('name', help='Name of the tool to remove') + # Parse arguments args = parser.parse_args() @@ -89,6 +93,8 @@ def main(): list_tools() elif args.tools_command in ['add', 'a']: generation.add_tool(args.name) + elif args.tools_command in ['remove', 'r']: + generation.remove_tool(args.name) else: tools_parser.print_help() else: From 151e4164223b7fc9b29892a3c85df079b9d4cd18 Mon Sep 17 00:00:00 2001 From: Travis Dent Date: Fri, 15 Nov 2024 08:41:39 -0800 Subject: [PATCH 2/7] Restore agent tool definition, implement removing agent tool definition --- agentstack/generation/tool_generation.py | 32 +++++++++++++++++++++--- 1 file changed, 28 insertions(+), 4 deletions(-) diff --git a/agentstack/generation/tool_generation.py b/agentstack/generation/tool_generation.py index e2c7997..af6f3a8 100644 --- a/agentstack/generation/tool_generation.py +++ b/agentstack/generation/tool_generation.py @@ -25,7 +25,7 @@ def add_tool(tool_name: str, path: Optional[str] = None): agentstack_json = open_json_file(f'{path}{AGENTSTACK_JSON_FILENAME}') if tool_name in agentstack_json.get('tools', []): - print(f"\033[31mTool '{tool_name}' is already installed\033[0m") + print(term_color(f'Tool {tool_name} is already installed', 'red')) sys.exit(1) with importlib.resources.path(f'agentstack.tools', f"{tool_name}.json") as tool_data_path: @@ -35,6 +35,7 @@ def add_tool(tool_name: str, path: Optional[str] = None): os.system(tool_data['package']) # Install package shutil.copy(tool_file_path, f'{path}src/tools/{tool_name}_tool.py') # Move tool from package to project add_tool_to_tools_init(tool_data, path) # Export tool from tools dir + add_tool_to_agent_definition(framework, tool_data, path) # Add tool to agent definition # TODO If the tool is reinstalled .env variables will be duplicated. insert_code_after_tag(f'{path}.env', '# Tools', [tool_data['env']], next_line=True) # Add env var insert_code_after_tag(f'{path}.env.example', '# Tools', [tool_data['env']], next_line=True) # Add env var @@ -63,13 +64,14 @@ def remove_tool(tool_name: str, path: Optional[str] = None): agentstack_json = open_json_file(f'{path}{AGENTSTACK_JSON_FILENAME}') if not tool_name in agentstack_json.get('tools', []): - print(f"\033[31mTool '{tool_name}' is not installed\033[0m") + print(term_color(f'Tool {tool_name} is not installed', 'red')) sys.exit(1) with importlib.resources.path(f'agentstack.tools', f"{tool_name}.json") as tool_data_path: tool_data = open_json_file(tool_data_path) os.remove(f'{path}src/tools/{tool_name}_tool.py') remove_tool_from_tools_init(tool_data, path) + remove_tool_from_agent_definition(framework, tool_data, path) # We don't remove the .env variables to preserve user data. agentstack_json['tools'].remove(tool_name) @@ -78,7 +80,7 @@ def remove_tool(tool_name: str, path: Optional[str] = None): print(term_color(f'🔨 Tool {tool_name} removed from agentstack project successfully', 'green')) # TODO If we add an uninstall command to the tool_config json we can run it instead. - print(term_color(f' To uninstall dependencies, reverse the install command: {tool_data["package"]}', 'blue')) + print(term_color(f'To uninstall dependencies, reverse the install command: {tool_data["package"]}', 'blue')) def _format_tool_import_statement(tool_data: dict): @@ -102,12 +104,34 @@ def remove_tool_from_tools_init(tool_data: dict, path: str = ''): print(line, end='') +def _framework_filename(framework: str, path: str = ''): + if framework == 'crewai': + return f'{path}src/crew.py' + + print(term_color(f'Unknown framework: {framework}', 'red')) + sys.exit(1) + + +def add_tool_to_agent_definition(framework: str, tool_data: dict, path: str = ''): + filename = _framework_filename(framework, path) + with fileinput.input(files=filename, inplace=True) as f: + for line in f: + print(line.replace('tools=[', f'tools=[{"*" if tool_data.get("tools_bundled") else ""}tools.{", tools.".join([tool_name for tool_name in tool_data["tools"]])}, '), end='') + + +def remove_tool_from_agent_definition(framework: str, tool_data: dict, path: str = ''): + filename = _framework_filename(framework, path) + with fileinput.input(files=filename, inplace=True) as f: + for line in f: + print(line.replace(f'{", ".join([f"tools.{tool_name}" for tool_name in tool_data["tools"]])}, ', ''), end='') + + def assert_tool_exists(tool_name: str, tools: dict): for cat in tools.keys(): for tool_dict in tools[cat]: if tool_dict['name'] == tool_name: return - print(f"\033[31mNo known AgentStack tool: '{tool_name}'\033[0m") + print(term_color(f'No known agentstack tool: {tool_name}', 'red')) sys.exit(1) From d682269591b0f9a88683970f79db1f5eec2bd935 Mon Sep 17 00:00:00 2001 From: Travis Dent Date: Fri, 15 Nov 2024 09:02:23 -0800 Subject: [PATCH 3/7] Support adding & removing poetry packages --- agentstack/generation/tool_generation.py | 7 ++++--- agentstack/tools/agent_connect.json | 2 +- agentstack/tools/browserbase.json | 2 +- agentstack/tools/code_interpreter.json | 2 +- agentstack/tools/composio.json | 2 +- agentstack/tools/directory_search.json | 2 +- agentstack/tools/exa.json | 2 +- agentstack/tools/file_read.json | 2 +- agentstack/tools/firecrawl.json | 2 +- agentstack/tools/ftp.json | 2 +- agentstack/tools/mem0.json | 2 +- agentstack/tools/open_interpreter.json | 2 +- agentstack/tools/perplexity.json | 2 +- agentstack/tools/vision.json | 2 +- 14 files changed, 17 insertions(+), 16 deletions(-) diff --git a/agentstack/generation/tool_generation.py b/agentstack/generation/tool_generation.py index af6f3a8..c467eaa 100644 --- a/agentstack/generation/tool_generation.py +++ b/agentstack/generation/tool_generation.py @@ -32,7 +32,8 @@ def add_tool(tool_name: str, path: Optional[str] = None): tool_data = open_json_file(tool_data_path) with importlib.resources.path(f'agentstack.templates.{framework}.tools', f"{tool_name}_tool.py") as tool_file_path: - os.system(tool_data['package']) # Install package + if tool_data.get('packages'): + os.system(f'poetry add {' '.join(tool_data["packages"])}') # Install packages shutil.copy(tool_file_path, f'{path}src/tools/{tool_name}_tool.py') # Move tool from package to project add_tool_to_tools_init(tool_data, path) # Export tool from tools dir add_tool_to_agent_definition(framework, tool_data, path) # Add tool to agent definition @@ -69,6 +70,8 @@ def remove_tool(tool_name: str, path: Optional[str] = None): with importlib.resources.path(f'agentstack.tools', f"{tool_name}.json") as tool_data_path: tool_data = open_json_file(tool_data_path) + if tool_data.get('packages'): + os.system(f'poetry remove {" ".join(tool_data["packages"])}') # Uninstall packages os.remove(f'{path}src/tools/{tool_name}_tool.py') remove_tool_from_tools_init(tool_data, path) remove_tool_from_agent_definition(framework, tool_data, path) @@ -79,8 +82,6 @@ def remove_tool(tool_name: str, path: Optional[str] = None): json.dump(agentstack_json, f, indent=4) print(term_color(f'🔨 Tool {tool_name} removed from agentstack project successfully', 'green')) - # TODO If we add an uninstall command to the tool_config json we can run it instead. - print(term_color(f'To uninstall dependencies, reverse the install command: {tool_data["package"]}', 'blue')) def _format_tool_import_statement(tool_data: dict): diff --git a/agentstack/tools/agent_connect.json b/agentstack/tools/agent_connect.json index 43c87b4..34922eb 100644 --- a/agentstack/tools/agent_connect.json +++ b/agentstack/tools/agent_connect.json @@ -1,6 +1,6 @@ { "name": "agent-connect", - "package": "poetry add agent-connect", + "packages": ["agent-connect"], "env": "HOST_DOMAIN=...\nHOST_PORT=\"80\"\nHOST_WS_PATH=\"/ws\"\nDID_DOCUMENT_PATH=...\nSSL_CERT_PATH=...\nSSL_KEY_PATH=...", "tools": ["send_message", "receive_message"] } diff --git a/agentstack/tools/browserbase.json b/agentstack/tools/browserbase.json index 4ffb1a0..9e75455 100644 --- a/agentstack/tools/browserbase.json +++ b/agentstack/tools/browserbase.json @@ -1,6 +1,6 @@ { "name": "browserbase", - "package": "poetry add browserbase playwright", + "packages": ["browserbase", "playwright"], "env": "BROWSERBASE_API_KEY=...\nBROWSERBASE_PROJECT_ID=...", "tools": ["browserbase"], "cta": "Create an API key at https://www.browserbase.com/" diff --git a/agentstack/tools/code_interpreter.json b/agentstack/tools/code_interpreter.json index 31271db..8c934fd 100644 --- a/agentstack/tools/code_interpreter.json +++ b/agentstack/tools/code_interpreter.json @@ -1,6 +1,6 @@ { "name": "code_interpreter", - "package": "poetry add crewai-tools", + "packages": ["crewai-tools"], "env": "", "tools": ["code_interpreter"] } \ No newline at end of file diff --git a/agentstack/tools/composio.json b/agentstack/tools/composio.json index 2a80bc6..c89d699 100644 --- a/agentstack/tools/composio.json +++ b/agentstack/tools/composio.json @@ -1,6 +1,6 @@ { "name": "composio", - "package": "poetry add composio-crewai", + "packages": ["composio-crewai"], "env": "COMPOSIO_API_KEY=...", "tools": ["composio_tools"], "tools_bundled": true, diff --git a/agentstack/tools/directory_search.json b/agentstack/tools/directory_search.json index 5ac6648..f96d332 100644 --- a/agentstack/tools/directory_search.json +++ b/agentstack/tools/directory_search.json @@ -1,6 +1,6 @@ { "name": "dir_search_tool", - "package": "poetry add crewai-tools", + "packages": ["crewai-tools"], "env": "", "tools": ["dir_search_tool"] } \ No newline at end of file diff --git a/agentstack/tools/exa.json b/agentstack/tools/exa.json index 45690ac..fbcec41 100644 --- a/agentstack/tools/exa.json +++ b/agentstack/tools/exa.json @@ -1,6 +1,6 @@ { "name": "exa", - "package": "poetry add exa_py", + "packages": ["exa_py"], "env": "EXA_API_KEY=...", "tools": ["search_and_contents"], "cta": "Get your Exa API key at https://dashboard.exa.ai/api-keys" diff --git a/agentstack/tools/file_read.json b/agentstack/tools/file_read.json index 04be812..98620c3 100644 --- a/agentstack/tools/file_read.json +++ b/agentstack/tools/file_read.json @@ -1,6 +1,6 @@ { "name": "file_read_tool", - "package": "poetry add crewai-tools", + "packages": [], "env": "", "tools": ["file_read_tool"] } \ No newline at end of file diff --git a/agentstack/tools/firecrawl.json b/agentstack/tools/firecrawl.json index e16d742..2fcdaae 100644 --- a/agentstack/tools/firecrawl.json +++ b/agentstack/tools/firecrawl.json @@ -1,6 +1,6 @@ { "name": "firecrawl", - "package": "poetry add firecrawl-py", + "packages": ["firecrawl-py"], "env": "FIRECRAWL_API_KEY=...", "tools": ["web_scrape", "web_crawl", "retrieve_web_crawl"], "cta": "Create an API key at https://www.firecrawl.dev/" diff --git a/agentstack/tools/ftp.json b/agentstack/tools/ftp.json index 7a3fd1c..a828de9 100644 --- a/agentstack/tools/ftp.json +++ b/agentstack/tools/ftp.json @@ -1,6 +1,6 @@ { "name": "ftp", - "package": "", + "packages": [], "env": "FTP_HOST=...\nFTP_USER=...\nFTP_PASSWORD=...", "tools": ["upload_files"], "cta": "Be sure to add your FTP credentials to .env" diff --git a/agentstack/tools/mem0.json b/agentstack/tools/mem0.json index 3ccb2b8..a273bc4 100644 --- a/agentstack/tools/mem0.json +++ b/agentstack/tools/mem0.json @@ -1,6 +1,6 @@ { "name": "mem0", - "package": "poetry add mem0ai", + "packages": ["mem0ai"], "env": "MEM0_API_KEY=...", "tools": ["write_to_memory", "read_from_memory"], "cta": "Create your mem0 API key at https://mem0.ai/" diff --git a/agentstack/tools/open_interpreter.json b/agentstack/tools/open_interpreter.json index c709940..a0bdbc8 100644 --- a/agentstack/tools/open_interpreter.json +++ b/agentstack/tools/open_interpreter.json @@ -1,6 +1,6 @@ { "name": "open_interpreter", - "package": "poetry add open-interpreter", + "packages": ["open-interpreter"], "env": "", "tools": ["execute_code"] } \ No newline at end of file diff --git a/agentstack/tools/perplexity.json b/agentstack/tools/perplexity.json index 8e75110..ea35bb3 100644 --- a/agentstack/tools/perplexity.json +++ b/agentstack/tools/perplexity.json @@ -1,6 +1,6 @@ { "name": "perplexity", - "package": "", + "packages": [], "env": "PERPLEXITY_API_KEY=pplx-...", "tools": ["query_perplexity"] } \ No newline at end of file diff --git a/agentstack/tools/vision.json b/agentstack/tools/vision.json index 528dc8e..3e27ea9 100644 --- a/agentstack/tools/vision.json +++ b/agentstack/tools/vision.json @@ -1,6 +1,6 @@ { "name": "vision", - "package": "poetry add crewai-tools", + "packages": ["crewai-tools"], "env": "", "tools": ["vision_tool"] } \ No newline at end of file From 1b3bb96e37b466c1f7a9c95c4f61a9bd1ca18161 Mon Sep 17 00:00:00 2001 From: Travis Dent Date: Fri, 15 Nov 2024 09:04:32 -0800 Subject: [PATCH 4/7] crewai-tools is already part of pyproject.toml, so exclude it from tool packages --- agentstack/tools/code_interpreter.json | 2 +- agentstack/tools/directory_search.json | 2 +- agentstack/tools/vision.json | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/agentstack/tools/code_interpreter.json b/agentstack/tools/code_interpreter.json index 8c934fd..a746ff1 100644 --- a/agentstack/tools/code_interpreter.json +++ b/agentstack/tools/code_interpreter.json @@ -1,6 +1,6 @@ { "name": "code_interpreter", - "packages": ["crewai-tools"], + "packages": [], "env": "", "tools": ["code_interpreter"] } \ No newline at end of file diff --git a/agentstack/tools/directory_search.json b/agentstack/tools/directory_search.json index f96d332..59fb371 100644 --- a/agentstack/tools/directory_search.json +++ b/agentstack/tools/directory_search.json @@ -1,6 +1,6 @@ { "name": "dir_search_tool", - "packages": ["crewai-tools"], + "packages": [], "env": "", "tools": ["dir_search_tool"] } \ No newline at end of file diff --git a/agentstack/tools/vision.json b/agentstack/tools/vision.json index 3e27ea9..c622f08 100644 --- a/agentstack/tools/vision.json +++ b/agentstack/tools/vision.json @@ -1,6 +1,6 @@ { "name": "vision", - "packages": ["crewai-tools"], + "packages": [], "env": "", "tools": ["vision_tool"] } \ No newline at end of file From 2a736bc58666f3942a6fd6421d96877727c1384f Mon Sep 17 00:00:00 2001 From: Travis Dent Date: Fri, 15 Nov 2024 10:54:07 -0800 Subject: [PATCH 5/7] Conditionals for adding env vars, python <3.12 compat --- agentstack/generation/gen_utils.py | 6 ++++++ agentstack/generation/tool_generation.py | 17 +++++++++++------ 2 files changed, 17 insertions(+), 6 deletions(-) diff --git a/agentstack/generation/gen_utils.py b/agentstack/generation/gen_utils.py index 3faf78c..d0aabfa 100644 --- a/agentstack/generation/gen_utils.py +++ b/agentstack/generation/gen_utils.py @@ -59,3 +59,9 @@ def insert_after_tasks(file_path, code_to_insert): return True return False + +def string_in_file(file_path: str, str_to_match: str) -> bool: + with open(file_path, 'r') as file: + file_content = file.read() + return str_to_match in file_content + diff --git a/agentstack/generation/tool_generation.py b/agentstack/generation/tool_generation.py index c467eaa..c8acccb 100644 --- a/agentstack/generation/tool_generation.py +++ b/agentstack/generation/tool_generation.py @@ -3,7 +3,7 @@ import sys from typing import Optional -from .gen_utils import insert_code_after_tag +from .gen_utils import insert_code_after_tag, string_in_file from ..utils import open_json_file, get_framework, term_color import os import shutil @@ -33,13 +33,17 @@ def add_tool(tool_name: str, path: Optional[str] = None): with importlib.resources.path(f'agentstack.templates.{framework}.tools', f"{tool_name}_tool.py") as tool_file_path: if tool_data.get('packages'): - os.system(f'poetry add {' '.join(tool_data["packages"])}') # Install packages + packages = ' '.join(tool_data['packages']) + os.system(f'poetry add {packages}') # Install packages shutil.copy(tool_file_path, f'{path}src/tools/{tool_name}_tool.py') # Move tool from package to project add_tool_to_tools_init(tool_data, path) # Export tool from tools dir add_tool_to_agent_definition(framework, tool_data, path) # Add tool to agent definition - # TODO If the tool is reinstalled .env variables will be duplicated. - insert_code_after_tag(f'{path}.env', '# Tools', [tool_data['env']], next_line=True) # Add env var - insert_code_after_tag(f'{path}.env.example', '# Tools', [tool_data['env']], next_line=True) # Add env var + if tool_data.get('env'): # if the env vars aren't in the .env files, add them + first_var_name = tool_data['env'].split('=')[0] + if not string_in_file(f'{path}.env', first_var_name): + insert_code_after_tag(f'{path}.env', '# Tools', [tool_data['env']], next_line=True) # Add env var + if not string_in_file(f'{path}.env.example', first_var_name): + insert_code_after_tag(f'{path}.env.example', '# Tools', [tool_data['env']], next_line=True) # Add env var if not agentstack_json.get('tools'): agentstack_json['tools'] = [] @@ -71,7 +75,8 @@ def remove_tool(tool_name: str, path: Optional[str] = None): with importlib.resources.path(f'agentstack.tools', f"{tool_name}.json") as tool_data_path: tool_data = open_json_file(tool_data_path) if tool_data.get('packages'): - os.system(f'poetry remove {" ".join(tool_data["packages"])}') # Uninstall packages + packages = ' '.join(tool_data['packages']) + os.system(f'poetry remove {packages}') # Uninstall packages os.remove(f'{path}src/tools/{tool_name}_tool.py') remove_tool_from_tools_init(tool_data, path) remove_tool_from_agent_definition(framework, tool_data, path) From c3f716a3f558a22fc83f990539805fa26bda8fb8 Mon Sep 17 00:00:00 2001 From: Travis Dent Date: Fri, 15 Nov 2024 10:58:11 -0800 Subject: [PATCH 6/7] Fancy term color on remove statement --- agentstack/generation/tool_generation.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/agentstack/generation/tool_generation.py b/agentstack/generation/tool_generation.py index c8acccb..944bbeb 100644 --- a/agentstack/generation/tool_generation.py +++ b/agentstack/generation/tool_generation.py @@ -86,7 +86,7 @@ def remove_tool(tool_name: str, path: Optional[str] = None): with open(f'{path}{AGENTSTACK_JSON_FILENAME}', 'w') as f: json.dump(agentstack_json, f, indent=4) - print(term_color(f'🔨 Tool {tool_name} removed from agentstack project successfully', 'green')) + print(term_color(f'🔨 Tool {tool_name}', 'green'), term_color('removed', 'red'), term_color('from agentstack project successfully', 'green')) def _format_tool_import_statement(tool_data: dict): From 5d972fa090d62a9e50db2139a97dbebd2d8446ff Mon Sep 17 00:00:00 2001 From: Travis Dent Date: Fri, 15 Nov 2024 11:02:34 -0800 Subject: [PATCH 7/7] put join inside format string with correct quotes --- agentstack/generation/tool_generation.py | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/agentstack/generation/tool_generation.py b/agentstack/generation/tool_generation.py index 944bbeb..f60b3f1 100644 --- a/agentstack/generation/tool_generation.py +++ b/agentstack/generation/tool_generation.py @@ -33,8 +33,7 @@ def add_tool(tool_name: str, path: Optional[str] = None): with importlib.resources.path(f'agentstack.templates.{framework}.tools', f"{tool_name}_tool.py") as tool_file_path: if tool_data.get('packages'): - packages = ' '.join(tool_data['packages']) - os.system(f'poetry add {packages}') # Install packages + os.system(f"poetry add {' '.join(tool_data['packages'])}") # Install packages shutil.copy(tool_file_path, f'{path}src/tools/{tool_name}_tool.py') # Move tool from package to project add_tool_to_tools_init(tool_data, path) # Export tool from tools dir add_tool_to_agent_definition(framework, tool_data, path) # Add tool to agent definition @@ -75,8 +74,7 @@ def remove_tool(tool_name: str, path: Optional[str] = None): with importlib.resources.path(f'agentstack.tools', f"{tool_name}.json") as tool_data_path: tool_data = open_json_file(tool_data_path) if tool_data.get('packages'): - packages = ' '.join(tool_data['packages']) - os.system(f'poetry remove {packages}') # Uninstall packages + os.system(f"poetry remove {' '.join(tool_data['packages'])}") # Uninstall packages os.remove(f'{path}src/tools/{tool_name}_tool.py') remove_tool_from_tools_init(tool_data, path) remove_tool_from_agent_definition(framework, tool_data, path)