mirror of
https://github.com/aljazceru/Auto-GPT.git
synced 2026-01-23 07:54:27 +01:00
* Feature/tighten up ci pipeline (#3700) * Fix docker volume mounts (#3710) Co-authored-by: Reinier van der Leer <github@pwuts.nl> Co-authored-by: Nicholas Tindle <nick@ntindle.com> * Feature/enable intuitive logs for community challenge step 1 (#3695) * Feature/enable intuitive logs summarization (#3697) * Move task_complete command out of prompt (#3663) * feat: move task_complete command out of prompt * fix: formatting fixes * Add the shutdown command to the test agents * tests: update test vcrs --------- Co-authored-by: James Collins <collijk@uw.edu> * Allow users to Disable Commands via the .env (#3667) * Document Disabling command categories (#3669) * feat: move task_complete command out of prompt * fix: formatting fixes * feat: add command disabling * docs: document how to disable command categories * Enable denylist handling for plugins (#3688) Co-authored-by: Luke Kyohere <lkyohere@mfsafrica.com> Co-authored-by: Nicholas Tindle <nick@ntindle.com> * Fix call to `plugin.post_planning` (#3414) Co-authored-by: Nicholas Tindle <nick@ntindle.com> * create information retrieval challenge a (#3770) Co-authored-by: Richard Beales <rich@richbeales.net> * fix typos (#3798) * Update run.bat (#3783) Co-authored-by: Richard Beales <rich@richbeales.net> * Update run.sh (#3752) Co-authored-by: Richard Beales <rich@richbeales.net> * ADD: Bash block in the contributing markdown (#3701) Co-authored-by: Richard Beales <rich@richbeales.net> * BUGFIX: Selenium Driver object reference was included in the browsing results for some reason (#3642) * * there is really no need to return the reference to the Selenium driver along with the text summary and list of links. * * removing unused second return value from browse_website() * * updated cassette * * updated YAML cassette for test_browse_website * * after requirements reinstall, another update YAML cassette for test_browse_website * * another update YAML cassette for test_browse_website, only as a placholder commit to trigger re-testing due to some docker TCP timeout issue * * another update YAML cassette for test_browse_website --------- Co-authored-by: batyu <batyu@localhost> * Update CONTRIBUTING.md * Self feedback Improvement (#3680) * Improved `Self-Feedback` * minor tweak * Test: Updated `test_get_self_feedback.py` * community challenges in the wiki (#3764) * Update README.md * Update PULL_REQUEST_TEMPLATE.md Added link to wiki Contributing page * Add link to wiki Contributing page * fix * Add link to wiki page on Contributing * Implement Logging of User Input in logs/Debug Folder (#3867) * Adds USER_INPUT_FILE_NAME * Update agent.py * Update agent.py Log only if console_input is not the authorise_key * Reformatting * add information retrieval challenge to the wiki (#3876) * add code owners policy (#3981) * add code owners * added @ to codeowners * switched to team ownership * Memory Challenge C (#3908) * Memory Challenge C * Working cassettes * Doc fixes * Linting and doc fix * Updated cassette * One more cassette try --------- Co-authored-by: merwanehamadi <merwanehamadi@gmail.com> * memory challenge c inconsistent (#3985) * Improve & fix memory challenge docs. (#3989) Co-authored-by: Kaan Osmanagaoglu <kaano@questps.com.au> * Feature/centralize prompt (#3990) Co-authored-by: xiao.hu <454292663@qq.com> * Use correct reference to prompt_generator in autogpt/llm/chat.py (#4011) * fix typos (#3998) Co-authored-by: Minfeng Lu <minfenglu@Minfengs-MacBook-Pro.local> Co-authored-by: Richard Beales <rich@richbeales.net> * fix typo in the getting started docs (#3997) Co-authored-by: Richard Beales <rich@richbeales.net> * Fix path to workspace directory in setup guide (#3927) Co-authored-by: Nicholas Tindle <nick@ntindle.com> * document that docker-compose 1.29.0 is minimally required (#3963) Co-authored-by: Nicholas Tindle <nick@ntindle.com> * Integrate pytest-xdist Plugin for Parallel and Concurrent Testing (#3870) * Adds pytest-parallel dependencies * Implement pytest-parallel for faster tests * Uses pytest-xdist * Auto number of workers processes * Update ci.yml --------- Co-authored-by: Nicholas Tindle <nick@ntindle.com> * explain temperature setting in env file (#4140) Co-authored-by: Richard Beales <rich@richbeales.net> * Catch JSON error in summary_memory.py (#3996) Co-authored-by: k-boikov <64261260+k-boikov@users.noreply.github.com> * Update duckduckgo dependency - min should be 2.9.5 (#4142) Co-authored-by: k-boikov <64261260+k-boikov@users.noreply.github.com> * Update Dockerfile - add missing scripts and plugins directories. (#3706) Co-authored-by: k-boikov <64261260+k-boikov@users.noreply.github.com> * Updated memory setup links (#3829) Co-authored-by: k-boikov <64261260+k-boikov@users.noreply.github.com> * Parse package versions so upgrades can be forced (#4149) * parse package versions so upgrades can be forced * better version from @collijk * fix typo in autopgt/agent/agent.py (#3747) Co-authored-by: merwanehamadi <merwanehamadi@gmail.com> Co-authored-by: Richard Beales <rich@richbeales.net> Co-authored-by: k-boikov <64261260+k-boikov@users.noreply.github.com> * Fix `milvus_memory_test.py` mock `Config` (#3424) Co-authored-by: k-boikov <64261260+k-boikov@users.noreply.github.com> * Implemented showing the number of preauthorised commands left. #1035 (#3322) Co-authored-by: mayubi <marwand@ayubi-it.de> Co-authored-by: Nicholas Tindle <nick@ntindle.com> Co-authored-by: k-boikov <64261260+k-boikov@users.noreply.github.com> * Challenge: Kubernetes and documentation (#4121) * challenge_kubes_and_readme * docs * testing * black and isort * revision * lint * comments * blackisort * docs * docs * deleting_cassette * suggestions * misspelling_errors --------- Co-authored-by: merwanehamadi <merwanehamadi@gmail.com> * Make sdwebui tests pass (when SD is running) (#3721) Co-authored-by: Nicholas Tindle <nick@ntindle.com> * Add Edge browser support using EdgeChromiumDriverManager (#3058) Co-authored-by: Nicholas Tindle <nick@ntindle.com> Co-authored-by: k-boikov <64261260+k-boikov@users.noreply.github.com> * Added --install-plugin-deps to Docker (#4151) Co-authored-by: Nicholas Tindle <nick@ntindle.com> * Feature/basic proxy (#4164) * basic proxy (#54) * basic proxy (#55) * basic proxy * basic proxy * basic proxy * basic proxy * add back double quotes * add more specific files * write file * basic proxy * Put back double quotes * test new CI (#4168) * test new CI * test new CI * remove double quotes * Feature/test new ci pipeline 2 (#4169) * test new CI * remove double quotes * make it a variable * make it a variable * Test New CI Pipeline (#4170) * introduce dummy prompt change * introduce dummy prompt change * empty commit * empty commit * empty commit * push to origin repo * add s to quote * Feature/fix rate limiting issue Step 1 (#4173) * temporarilly remove 3.11 * add back 3.11 (#4185) * Revert "Put back 3.11 until it's removed as a requirement" (#4191) --------- Co-authored-by: Reinier van der Leer <github@pwuts.nl> Co-authored-by: merwanehamadi <merwanehamadi@gmail.com> Co-authored-by: Peter Petermann <ppetermann80@googlemail.com> Co-authored-by: Nicholas Tindle <nick@ntindle.com> Co-authored-by: James Collins <collijk@uw.edu> Co-authored-by: Luke K <2609441+pr-0f3t@users.noreply.github.com> Co-authored-by: Luke Kyohere <lkyohere@mfsafrica.com> Co-authored-by: Robin Richtsfeld <robin.richtsfeld@gmail.com> Co-authored-by: RainRat <rainrat78@yahoo.ca> Co-authored-by: itsmarble <130370814+itsmarble@users.noreply.github.com> Co-authored-by: Ambuj Pawar <pawar.ambuj@gmail.com> Co-authored-by: bszollosinagy <4211175+bszollosinagy@users.noreply.github.com> Co-authored-by: batyu <batyu@localhost> Co-authored-by: Pi <sunfish7@gmail.com> Co-authored-by: AbTrax <45964236+AbTrax@users.noreply.github.com> Co-authored-by: Andres Caicedo <73312784+AndresCdo@users.noreply.github.com> Co-authored-by: Douglas Schonholtz <15002691+dschonholtz@users.noreply.github.com> Co-authored-by: Kaan <kaanixir@gmail.com> Co-authored-by: Kaan Osmanagaoglu <kaano@questps.com.au> Co-authored-by: xiao.hu <454292663@qq.com> Co-authored-by: Tomasz Kasperczyk <tomaszikasperczyk@gmail.com> Co-authored-by: minfeng-ai <42948406+minfenglu@users.noreply.github.com> Co-authored-by: Minfeng Lu <minfenglu@Minfengs-MacBook-Pro.local> Co-authored-by: Shlomi <81581678+jit-shlomi@users.noreply.github.com> Co-authored-by: Itai Steinherz <itaisteinherz@gmail.com> Co-authored-by: Boostrix <119627414+Boostrix@users.noreply.github.com> Co-authored-by: Kristian Jackson <kristian.jackson@gmail.com> Co-authored-by: k-boikov <64261260+k-boikov@users.noreply.github.com> Co-authored-by: Eduardo Salinas <edus@microsoft.com> Co-authored-by: prom3theu5 <dave@simcube.co.uk> Co-authored-by: dominic-ks <contact@bedevious.co.uk> Co-authored-by: andrey13771 <51243350+andrey13771@users.noreply.github.com> Co-authored-by: Marwand Ayubi <98717667+xhypeDE@users.noreply.github.com> Co-authored-by: mayubi <marwand@ayubi-it.de> Co-authored-by: Media <12145726+rihp@users.noreply.github.com> Co-authored-by: Cenny <cwenner@gmail.com> Co-authored-by: Abdelkarim Habouch <37211852+karimhabush@users.noreply.github.com>
276 lines
10 KiB
Python
276 lines
10 KiB
Python
"""Handles loading of plugins."""
|
|
|
|
import importlib
|
|
import json
|
|
import os
|
|
import zipfile
|
|
from pathlib import Path
|
|
from typing import List, Optional, Tuple
|
|
from urllib.parse import urlparse
|
|
from zipimport import zipimporter
|
|
|
|
import openapi_python_client
|
|
import requests
|
|
from auto_gpt_plugin_template import AutoGPTPluginTemplate
|
|
from openapi_python_client.cli import Config as OpenAPIConfig
|
|
|
|
from autogpt.config import Config
|
|
from autogpt.logs import logger
|
|
from autogpt.models.base_open_ai_plugin import BaseOpenAIPlugin
|
|
|
|
|
|
def inspect_zip_for_modules(zip_path: str, debug: bool = False) -> list[str]:
|
|
"""
|
|
Inspect a zipfile for a modules.
|
|
|
|
Args:
|
|
zip_path (str): Path to the zipfile.
|
|
debug (bool, optional): Enable debug logging. Defaults to False.
|
|
|
|
Returns:
|
|
list[str]: The list of module names found or empty list if none were found.
|
|
"""
|
|
result = []
|
|
with zipfile.ZipFile(zip_path, "r") as zfile:
|
|
for name in zfile.namelist():
|
|
if name.endswith("__init__.py") and not name.startswith("__MACOSX"):
|
|
logger.debug(f"Found module '{name}' in the zipfile at: {name}")
|
|
result.append(name)
|
|
if len(result) == 0:
|
|
logger.debug(f"Module '__init__.py' not found in the zipfile @ {zip_path}.")
|
|
return result
|
|
|
|
|
|
def write_dict_to_json_file(data: dict, file_path: str) -> None:
|
|
"""
|
|
Write a dictionary to a JSON file.
|
|
Args:
|
|
data (dict): Dictionary to write.
|
|
file_path (str): Path to the file.
|
|
"""
|
|
with open(file_path, "w") as file:
|
|
json.dump(data, file, indent=4)
|
|
|
|
|
|
def fetch_openai_plugins_manifest_and_spec(cfg: Config) -> dict:
|
|
"""
|
|
Fetch the manifest for a list of OpenAI plugins.
|
|
Args:
|
|
urls (List): List of URLs to fetch.
|
|
Returns:
|
|
dict: per url dictionary of manifest and spec.
|
|
"""
|
|
# TODO add directory scan
|
|
manifests = {}
|
|
for url in cfg.plugins_openai:
|
|
openai_plugin_client_dir = f"{cfg.plugins_dir}/openai/{urlparse(url).netloc}"
|
|
create_directory_if_not_exists(openai_plugin_client_dir)
|
|
if not os.path.exists(f"{openai_plugin_client_dir}/ai-plugin.json"):
|
|
try:
|
|
response = requests.get(f"{url}/.well-known/ai-plugin.json")
|
|
if response.status_code == 200:
|
|
manifest = response.json()
|
|
if manifest["schema_version"] != "v1":
|
|
logger.warn(
|
|
f"Unsupported manifest version: {manifest['schem_version']} for {url}"
|
|
)
|
|
continue
|
|
if manifest["api"]["type"] != "openapi":
|
|
logger.warn(
|
|
f"Unsupported API type: {manifest['api']['type']} for {url}"
|
|
)
|
|
continue
|
|
write_dict_to_json_file(
|
|
manifest, f"{openai_plugin_client_dir}/ai-plugin.json"
|
|
)
|
|
else:
|
|
logger.warn(
|
|
f"Failed to fetch manifest for {url}: {response.status_code}"
|
|
)
|
|
except requests.exceptions.RequestException as e:
|
|
logger.warn(f"Error while requesting manifest from {url}: {e}")
|
|
else:
|
|
logger.info(f"Manifest for {url} already exists")
|
|
manifest = json.load(open(f"{openai_plugin_client_dir}/ai-plugin.json"))
|
|
if not os.path.exists(f"{openai_plugin_client_dir}/openapi.json"):
|
|
openapi_spec = openapi_python_client._get_document(
|
|
url=manifest["api"]["url"], path=None, timeout=5
|
|
)
|
|
write_dict_to_json_file(
|
|
openapi_spec, f"{openai_plugin_client_dir}/openapi.json"
|
|
)
|
|
else:
|
|
logger.info(f"OpenAPI spec for {url} already exists")
|
|
openapi_spec = json.load(open(f"{openai_plugin_client_dir}/openapi.json"))
|
|
manifests[url] = {"manifest": manifest, "openapi_spec": openapi_spec}
|
|
return manifests
|
|
|
|
|
|
def create_directory_if_not_exists(directory_path: str) -> bool:
|
|
"""
|
|
Create a directory if it does not exist.
|
|
Args:
|
|
directory_path (str): Path to the directory.
|
|
Returns:
|
|
bool: True if the directory was created, else False.
|
|
"""
|
|
if not os.path.exists(directory_path):
|
|
try:
|
|
os.makedirs(directory_path)
|
|
logger.debug(f"Created directory: {directory_path}")
|
|
return True
|
|
except OSError as e:
|
|
logger.warn(f"Error creating directory {directory_path}: {e}")
|
|
return False
|
|
else:
|
|
logger.info(f"Directory {directory_path} already exists")
|
|
return True
|
|
|
|
|
|
def initialize_openai_plugins(
|
|
manifests_specs: dict, cfg: Config, debug: bool = False
|
|
) -> dict:
|
|
"""
|
|
Initialize OpenAI plugins.
|
|
Args:
|
|
manifests_specs (dict): per url dictionary of manifest and spec.
|
|
cfg (Config): Config instance including plugins config
|
|
debug (bool, optional): Enable debug logging. Defaults to False.
|
|
Returns:
|
|
dict: per url dictionary of manifest, spec and client.
|
|
"""
|
|
openai_plugins_dir = f"{cfg.plugins_dir}/openai"
|
|
if create_directory_if_not_exists(openai_plugins_dir):
|
|
for url, manifest_spec in manifests_specs.items():
|
|
openai_plugin_client_dir = f"{openai_plugins_dir}/{urlparse(url).hostname}"
|
|
_meta_option = (openapi_python_client.MetaType.SETUP,)
|
|
_config = OpenAPIConfig(
|
|
**{
|
|
"project_name_override": "client",
|
|
"package_name_override": "client",
|
|
}
|
|
)
|
|
prev_cwd = Path.cwd()
|
|
os.chdir(openai_plugin_client_dir)
|
|
Path("ai-plugin.json")
|
|
if not os.path.exists("client"):
|
|
client_results = openapi_python_client.create_new_client(
|
|
url=manifest_spec["manifest"]["api"]["url"],
|
|
path=None,
|
|
meta=_meta_option,
|
|
config=_config,
|
|
)
|
|
if client_results:
|
|
logger.warn(
|
|
f"Error creating OpenAPI client: {client_results[0].header} \n"
|
|
f" details: {client_results[0].detail}"
|
|
)
|
|
continue
|
|
spec = importlib.util.spec_from_file_location(
|
|
"client", "client/client/client.py"
|
|
)
|
|
module = importlib.util.module_from_spec(spec)
|
|
spec.loader.exec_module(module)
|
|
client = module.Client(base_url=url)
|
|
os.chdir(prev_cwd)
|
|
manifest_spec["client"] = client
|
|
return manifests_specs
|
|
|
|
|
|
def instantiate_openai_plugin_clients(
|
|
manifests_specs_clients: dict, cfg: Config, debug: bool = False
|
|
) -> dict:
|
|
"""
|
|
Instantiates BaseOpenAIPlugin instances for each OpenAI plugin.
|
|
Args:
|
|
manifests_specs_clients (dict): per url dictionary of manifest, spec and client.
|
|
cfg (Config): Config instance including plugins config
|
|
debug (bool, optional): Enable debug logging. Defaults to False.
|
|
Returns:
|
|
plugins (dict): per url dictionary of BaseOpenAIPlugin instances.
|
|
|
|
"""
|
|
plugins = {}
|
|
for url, manifest_spec_client in manifests_specs_clients.items():
|
|
plugins[url] = BaseOpenAIPlugin(manifest_spec_client)
|
|
return plugins
|
|
|
|
|
|
def scan_plugins(cfg: Config, debug: bool = False) -> List[AutoGPTPluginTemplate]:
|
|
"""Scan the plugins directory for plugins and loads them.
|
|
|
|
Args:
|
|
cfg (Config): Config instance including plugins config
|
|
debug (bool, optional): Enable debug logging. Defaults to False.
|
|
|
|
Returns:
|
|
List[Tuple[str, Path]]: List of plugins.
|
|
"""
|
|
loaded_plugins = []
|
|
# Generic plugins
|
|
plugins_path_path = Path(cfg.plugins_dir)
|
|
|
|
logger.debug(f"Allowlisted Plugins: {cfg.plugins_allowlist}")
|
|
logger.debug(f"Denylisted Plugins: {cfg.plugins_denylist}")
|
|
|
|
for plugin in plugins_path_path.glob("*.zip"):
|
|
if moduleList := inspect_zip_for_modules(str(plugin), debug):
|
|
for module in moduleList:
|
|
plugin = Path(plugin)
|
|
module = Path(module)
|
|
logger.debug(f"Plugin: {plugin} Module: {module}")
|
|
zipped_package = zipimporter(str(plugin))
|
|
zipped_module = zipped_package.load_module(str(module.parent))
|
|
for key in dir(zipped_module):
|
|
if key.startswith("__"):
|
|
continue
|
|
a_module = getattr(zipped_module, key)
|
|
a_keys = dir(a_module)
|
|
if (
|
|
"_abc_impl" in a_keys
|
|
and a_module.__name__ != "AutoGPTPluginTemplate"
|
|
and denylist_allowlist_check(a_module.__name__, cfg)
|
|
):
|
|
loaded_plugins.append(a_module())
|
|
# OpenAI plugins
|
|
if cfg.plugins_openai:
|
|
manifests_specs = fetch_openai_plugins_manifest_and_spec(cfg)
|
|
if manifests_specs.keys():
|
|
manifests_specs_clients = initialize_openai_plugins(
|
|
manifests_specs, cfg, debug
|
|
)
|
|
for url, openai_plugin_meta in manifests_specs_clients.items():
|
|
if denylist_allowlist_check(url, cfg):
|
|
plugin = BaseOpenAIPlugin(openai_plugin_meta)
|
|
loaded_plugins.append(plugin)
|
|
|
|
if loaded_plugins:
|
|
logger.info(f"\nPlugins found: {len(loaded_plugins)}\n" "--------------------")
|
|
for plugin in loaded_plugins:
|
|
logger.info(f"{plugin._name}: {plugin._version} - {plugin._description}")
|
|
return loaded_plugins
|
|
|
|
|
|
def denylist_allowlist_check(plugin_name: str, cfg: Config) -> bool:
|
|
"""Check if the plugin is in the allowlist or denylist.
|
|
|
|
Args:
|
|
plugin_name (str): Name of the plugin.
|
|
cfg (Config): Config object.
|
|
|
|
Returns:
|
|
True or False
|
|
"""
|
|
logger.debug(f"Checking if plugin {plugin_name} should be loaded")
|
|
if plugin_name in cfg.plugins_denylist:
|
|
logger.debug(f"Not loading plugin {plugin_name} as it was in the denylist.")
|
|
return False
|
|
if plugin_name in cfg.plugins_allowlist:
|
|
logger.debug(f"Loading plugin {plugin_name} as it was in the allowlist.")
|
|
return True
|
|
ack = input(
|
|
f"WARNING: Plugin {plugin_name} found. But not in the"
|
|
f" allowlist... Load? ({cfg.authorise_key}/{cfg.exit_key}): "
|
|
)
|
|
return ack.lower() == cfg.authorise_key
|