From e660e693ff5334e7d83670f434348f59956b25cc Mon Sep 17 00:00:00 2001 From: gagb Date: Sat, 11 Nov 2023 20:21:50 +0000 Subject: [PATCH 01/12] Refactor GPTAssistantAgent constructor to handle instructions and overwrite_instructions flag - Ensure that `system_message` is always consistent with `instructions` - Ensure provided instructions are always used - Add option to permanently modify the instructions of the assistant --- .../agentchat/contrib/gpt_assistant_agent.py | 47 +++++++++++++++---- 1 file changed, 37 insertions(+), 10 deletions(-) diff --git a/autogen/agentchat/contrib/gpt_assistant_agent.py b/autogen/agentchat/contrib/gpt_assistant_agent.py index 50f7d4799a46..d40b5da97e39 100644 --- a/autogen/agentchat/contrib/gpt_assistant_agent.py +++ b/autogen/agentchat/contrib/gpt_assistant_agent.py @@ -21,33 +21,29 @@ class GPTAssistantAgent(ConversableAgent): def __init__( self, name="GPT Assistant", - instructions: Optional[str] = "You are a helpful GPT Assistant.", + instructions: Optional[str] = None, llm_config: Optional[Union[Dict, bool]] = None, + overwrite_instructions: bool = False, ): """ Args: name (str): name of the agent. instructions (str): instructions for the OpenAI assistant configuration. + the system message of the agent will be set to the instructions and used in the assistant run. + (irrespective of the overwrite_instructions flag) llm_config (dict or False): llm inference configuration. - model: Model to use for the assistant (gpt-4-1106-preview, gpt-3.5-turbo-1106). - check_every_ms: check thread run status interval - tools: Give Assistants access to OpenAI-hosted tools like Code Interpreter and Knowledge Retrieval, or build your own tools using Function calling. ref https://platform.openai.com/docs/assistants/tools - file_ids: files used by retrieval in run + overwrite_instructions (bool): whether to overwrite the instructions of an existing assistant. """ - super().__init__( - name=name, - system_message=instructions, - human_input_mode="NEVER", - llm_config=llm_config, - ) - # Use AutoGen OpenAIWrapper to create a client - oai_wrapper = OpenAIWrapper(**self.llm_config) + oai_wrapper = OpenAIWrapper(**llm_config) if len(oai_wrapper._clients) > 1: logger.warning("GPT Assistant only supports one OpenAI client. Using the first client in the list.") self._openai_client = oai_wrapper._clients[0] - openai_assistant_id = llm_config.get("assistant_id", None) if openai_assistant_id is None: # create a new assistant @@ -60,6 +56,31 @@ def __init__( else: # retrieve an existing assistant self._openai_assistant = self._openai_client.beta.assistants.retrieve(openai_assistant_id) + # if no instructions are provided, set the instructions to the existing instructions + if instructions is None: + logger.warning( + "No instructions were provided for given assistant. Using existing instructions from assistant API." + ) + instructions = self.get_assistant_instructions() + elif instructions is not None and overwrite_instructions: + logger.warning( + "overwrite_instructions is True. Provided instructions will be used and will modify the assistant in the API" + ) + self._openai_assistant = self._openai_client.beta.assistants.update( + assistant_id=openai_assistant_id, + instructions=instructions, + ) + else: + logger.warning( + "overwrite_instructions is False. Provided instructions will be used without permanently modifying the assistant in the API." + ) + + super().__init__( + name=name, + system_message=instructions, + human_input_mode="NEVER", + llm_config=llm_config, + ) # lazly create thread self._openai_threads = {} @@ -107,6 +128,8 @@ def _invoke_assistant( run = self._openai_client.beta.threads.runs.create( thread_id=assistant_thread.id, assistant_id=self._openai_assistant.id, + # pass the latest system message as instructions + instructions=self.system_message, ) run_response_messages = self._get_run_response(assistant_thread, run) @@ -300,3 +323,7 @@ def pretty_print_thread(self, thread): def oai_threads(self) -> Dict[Agent, Any]: """Return the threads of the agent.""" return self._openai_threads + + def get_assistant_instructions(self): + """Return the assistant instructions from OAI assistant API""" + return self._openai_assistant.instructions From 0b8e2333fc9e9c3b4540d06ac43864eb3de00c8c Mon Sep 17 00:00:00 2001 From: gagb Date: Sat, 11 Nov 2023 22:32:38 +0000 Subject: [PATCH 02/12] Improve default behavior --- .../agentchat/contrib/gpt_assistant_agent.py | 21 ++++++++++++++----- 1 file changed, 16 insertions(+), 5 deletions(-) diff --git a/autogen/agentchat/contrib/gpt_assistant_agent.py b/autogen/agentchat/contrib/gpt_assistant_agent.py index d40b5da97e39..2e0d77ef9622 100644 --- a/autogen/agentchat/contrib/gpt_assistant_agent.py +++ b/autogen/agentchat/contrib/gpt_assistant_agent.py @@ -7,6 +7,7 @@ from autogen import OpenAIWrapper from autogen.agentchat.agent import Agent from autogen.agentchat.assistant_agent import ConversableAgent +from autogen.agentchat.assistant_agent import AssistantAgent from typing import Dict, Optional, Union, List, Tuple, Any logger = logging.getLogger(__name__) @@ -29,9 +30,14 @@ def __init__( Args: name (str): name of the agent. instructions (str): instructions for the OpenAI assistant configuration. - the system message of the agent will be set to the instructions and used in the assistant run. - (irrespective of the overwrite_instructions flag) + When instructions is not None, the system message of the agent will be + set to the provided instructions and used in the assistant run, irrespective + of the overwrite_instructions flag. But when instructions is None, + and the assistant does not exist, the system message will be set to + AssistantAgent.DEFAULT_SYSTEM_MESSAGE. If the assistant exists, the + system message will be set to the existing assistant instructions. llm_config (dict or False): llm inference configuration. + - assistant_id: ID of the assistant to use. If None, a new assistant will be created. - model: Model to use for the assistant (gpt-4-1106-preview, gpt-3.5-turbo-1106). - check_every_ms: check thread run status interval - tools: Give Assistants access to OpenAI-hosted tools like Code Interpreter and Knowledge Retrieval, @@ -47,11 +53,16 @@ def __init__( openai_assistant_id = llm_config.get("assistant_id", None) if openai_assistant_id is None: # create a new assistant + if instructions is None: + logger.warning( + "No instructions were provided for new assistant. Using default instructions from AssistantAgent.DEFAULT_SYSTEM_MESSAGE." + ) + instructions = AssistantAgent.DEFAULT_SYSTEM_MESSAGE self._openai_assistant = self._openai_client.beta.assistants.create( name=name, instructions=instructions, - tools=self.llm_config.get("tools", []), - model=self.llm_config.get("model", "gpt-4-1106-preview"), + tools=llm_config.get("tools", []), + model=llm_config.get("model", "gpt-4-1106-preview"), ) else: # retrieve an existing assistant @@ -62,7 +73,7 @@ def __init__( "No instructions were provided for given assistant. Using existing instructions from assistant API." ) instructions = self.get_assistant_instructions() - elif instructions is not None and overwrite_instructions: + elif overwrite_instructions is True: logger.warning( "overwrite_instructions is True. Provided instructions will be used and will modify the assistant in the API" ) From d7cf4812cd91c3b74791ae0bcab75d59db394dd3 Mon Sep 17 00:00:00 2001 From: gagb Date: Sat, 11 Nov 2023 23:34:24 +0000 Subject: [PATCH 03/12] Add a test; add method to delete assistant --- .../agentchat/contrib/gpt_assistant_agent.py | 9 +++++++ test/agentchat/contrib/test_gpt_assistant.py | 27 ++++++++++++++++++- 2 files changed, 35 insertions(+), 1 deletion(-) diff --git a/autogen/agentchat/contrib/gpt_assistant_agent.py b/autogen/agentchat/contrib/gpt_assistant_agent.py index 2e0d77ef9622..d226e8ed01d2 100644 --- a/autogen/agentchat/contrib/gpt_assistant_agent.py +++ b/autogen/agentchat/contrib/gpt_assistant_agent.py @@ -335,6 +335,15 @@ def oai_threads(self) -> Dict[Agent, Any]: """Return the threads of the agent.""" return self._openai_threads + @property + def assistant_id(self): + """Return the assistant id""" + return self._openai_assistant.id + def get_assistant_instructions(self): """Return the assistant instructions from OAI assistant API""" return self._openai_assistant.instructions + + def delete_assistant(self): + """Delete the assistant from OAI assistant API""" + self._openai_client.beta.assistants.delete(self.assistant_id) diff --git a/test/agentchat/contrib/test_gpt_assistant.py b/test/agentchat/contrib/test_gpt_assistant.py index bf8d2e227b7f..8bb16f713ba1 100644 --- a/test/agentchat/contrib/test_gpt_assistant.py +++ b/test/agentchat/contrib/test_gpt_assistant.py @@ -38,9 +38,10 @@ def test_gpt_assistant_chat(): "description": "This is an API endpoint allowing users (analysts) to input question about GitHub in text format to retrieve the realted and structured data.", } + config_list = autogen.config_list_from_json(OAI_CONFIG_LIST, file_location=KEY_LOC) analyst = GPTAssistantAgent( name="Open_Source_Project_Analyst", - llm_config={"tools": [{"type": "function", "function": ossinsight_api_schema}]}, + llm_config={"tools": [{"type": "function", "function": ossinsight_api_schema}], "config_list": config_list}, instructions="Hello, Open Source Project Analyst. You'll conduct comprehensive evaluations of open source projects or organizations on the GitHub platform", ) analyst.register_function( @@ -62,5 +63,29 @@ def test_gpt_assistant_chat(): assert len(analyst._openai_threads) == 0 +def test_get_assistant_instructions(): + """ + Create a new assistant + Set its instructions + Retrieve the instructions + assert that the retrieved instructions match the set instructions + """ + + config_list = autogen.config_list_from_json(OAI_CONFIG_LIST, file_location=KEY_LOC) + assistant = GPTAssistantAgent( + "assistant", + instructions="This is a test", + llm_config={ + "config_list": config_list, + }, + ) + + instruction_match = assistant.get_assistant_instructions() == "This is a test" + assistant.delete_assistant() + + assert instruction_match is True + + if __name__ == "__main__": test_gpt_assistant_chat() + test_get_assistant_instructions() From f3db6099e0eba03e106424dbda4812123baa926a Mon Sep 17 00:00:00 2001 From: gagb Date: Sat, 11 Nov 2023 23:49:14 +0000 Subject: [PATCH 04/12] Add a new test for overwriting instructions --- test/agentchat/contrib/test_gpt_assistant.py | 48 ++++++++++++++++++-- 1 file changed, 44 insertions(+), 4 deletions(-) diff --git a/test/agentchat/contrib/test_gpt_assistant.py b/test/agentchat/contrib/test_gpt_assistant.py index 8bb16f713ba1..22f42d94c99b 100644 --- a/test/agentchat/contrib/test_gpt_assistant.py +++ b/test/agentchat/contrib/test_gpt_assistant.py @@ -65,10 +65,8 @@ def test_gpt_assistant_chat(): def test_get_assistant_instructions(): """ - Create a new assistant - Set its instructions - Retrieve the instructions - assert that the retrieved instructions match the set instructions + Test function to create a new GPTAssistantAgent, set its instructions, retrieve the instructions, + and assert that the retrieved instructions match the set instructions. """ config_list = autogen.config_list_from_json(OAI_CONFIG_LIST, file_location=KEY_LOC) @@ -86,6 +84,48 @@ def test_get_assistant_instructions(): assert instruction_match is True +def test_gpt_assistant_instructions_overwrite(): + """ + Test that the instructions of a GPTAssistantAgent can be overwritten or not depending on the value of the + `overwrite_instructions` parameter when creating a new assistant with the same ID. + + Steps: + 1. Create a new GPTAssistantAgent with some instructions. + 2. Get the ID of the assistant. + 3. Create a new GPTAssistantAgent with the same ID but different instructions and `overwrite_instructions=True`. + 4. Check that the instructions of the assistant have been overwritten with the new ones. + """ + + instructions1 = "This is a test #1" + instructions2 = "This is a test #2" + + config_list = autogen.config_list_from_json(OAI_CONFIG_LIST, file_location=KEY_LOC) + assistant = GPTAssistantAgent( + "assistant", + instructions=instructions1, + llm_config={ + "config_list": config_list, + }, + ) + + assistant_id = assistant.assistant_id + assistant = GPTAssistantAgent( + "assistant", + instructions=instructions2, + llm_config={ + "config_list": config_list, + "assistant_id": assistant_id, + }, + overwrite_instructions=True, + ) + + instruction_match = assistant.get_assistant_instructions() == instructions2 + assistant.delete_assistant() + + assert instruction_match is True + + if __name__ == "__main__": test_gpt_assistant_chat() test_get_assistant_instructions() + test_gpt_assistant_instructions_overwrite() From 760dd4293f4c52063096dc24635c3311ace0c311 Mon Sep 17 00:00:00 2001 From: gagb Date: Sat, 11 Nov 2023 23:53:41 +0000 Subject: [PATCH 05/12] Add test case for when no instructions are given for existing assistant --- test/agentchat/contrib/test_gpt_assistant.py | 33 ++++++++++++++++++++ 1 file changed, 33 insertions(+) diff --git a/test/agentchat/contrib/test_gpt_assistant.py b/test/agentchat/contrib/test_gpt_assistant.py index 22f42d94c99b..83c8e58494ae 100644 --- a/test/agentchat/contrib/test_gpt_assistant.py +++ b/test/agentchat/contrib/test_gpt_assistant.py @@ -125,7 +125,40 @@ def test_gpt_assistant_instructions_overwrite(): assert instruction_match is True +def test_gpt_assistant_existing_no_instructions(): + """ + Test function to check if the GPTAssistantAgent can retrieve instructions for an existing assistant + even if the assistant was created with no instructions initially. + """ + instructions = "This is a test #1" + + config_list = autogen.config_list_from_json(OAI_CONFIG_LIST, file_location=KEY_LOC) + assistant = GPTAssistantAgent( + "assistant", + instructions=instructions, + llm_config={ + "config_list": config_list, + }, + ) + + assistant_id = assistant.assistant_id + + # create a new assistant with the same ID but no instructions + assistant = GPTAssistantAgent( + "assistant", + llm_config={ + "config_list": config_list, + "assistant_id": assistant_id, + }, + ) + + instruction_match = assistant.get_assistant_instructions() == instructions + assistant.delete_assistant() + assert instruction_match is True + + if __name__ == "__main__": test_gpt_assistant_chat() test_get_assistant_instructions() test_gpt_assistant_instructions_overwrite() + test_gpt_assistant_existing_no_instructions() From 211fbda3f47ab85a13b940f87a5f906cc57e1fe4 Mon Sep 17 00:00:00 2001 From: gagb Date: Sun, 12 Nov 2023 02:37:28 +0000 Subject: [PATCH 06/12] Add pytest markers to test_gpt_assistant.py --- test/agentchat/contrib/test_gpt_assistant.py | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/test/agentchat/contrib/test_gpt_assistant.py b/test/agentchat/contrib/test_gpt_assistant.py index 83c8e58494ae..fab3fbb77db2 100644 --- a/test/agentchat/contrib/test_gpt_assistant.py +++ b/test/agentchat/contrib/test_gpt_assistant.py @@ -63,6 +63,10 @@ def test_gpt_assistant_chat(): assert len(analyst._openai_threads) == 0 +@pytest.mark.skipif( + sys.platform in ["darwin", "win32"] or skip_test, + reason="do not run on MacOS or windows or dependency is not installed", +) def test_get_assistant_instructions(): """ Test function to create a new GPTAssistantAgent, set its instructions, retrieve the instructions, @@ -84,6 +88,10 @@ def test_get_assistant_instructions(): assert instruction_match is True +@pytest.mark.skipif( + sys.platform in ["darwin", "win32"] or skip_test, + reason="do not run on MacOS or windows or dependency is not installed", +) def test_gpt_assistant_instructions_overwrite(): """ Test that the instructions of a GPTAssistantAgent can be overwritten or not depending on the value of the @@ -125,6 +133,10 @@ def test_gpt_assistant_instructions_overwrite(): assert instruction_match is True +@pytest.mark.skipif( + sys.platform in ["darwin", "win32"] or skip_test, + reason="do not run on MacOS or windows or dependency is not installed", +) def test_gpt_assistant_existing_no_instructions(): """ Test function to check if the GPTAssistantAgent can retrieve instructions for an existing assistant From 5d83da5a3f28ac0eadebf9468492108c9b2d4c70 Mon Sep 17 00:00:00 2001 From: kevin666aa Date: Sat, 11 Nov 2023 23:36:50 -0500 Subject: [PATCH 07/12] add test in workflow --- .github/workflows/contrib-openai.yml | 41 ++++++++++++++++++++++++++++ .github/workflows/contrib-tests.yml | 26 ++++++++++++++++++ 2 files changed, 67 insertions(+) diff --git a/.github/workflows/contrib-openai.yml b/.github/workflows/contrib-openai.yml index c4f19b78f526..46335517987f 100644 --- a/.github/workflows/contrib-openai.yml +++ b/.github/workflows/contrib-openai.yml @@ -97,3 +97,44 @@ jobs: with: file: ./coverage.xml flags: unittests + GPTAssistantAgent: + strategy: + matrix: + os: [ubuntu-latest] + python-version: ["3.11"] + runs-on: ${{ matrix.os }} + environment: openai1 + steps: + # checkout to pr branch + - name: Checkout + uses: actions/checkout@v3 + 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 }} + - name: Install packages and dependencies + run: | + docker --version + python -m pip install --upgrade pip wheel + pip install -e . + python -c "import autogen" + pip install coverage pytest-asyncio + - name: Install packages for test when needed + run: | + pip install docker + - name: Coverage + env: + OPENAI_API_KEY: ${{ secrets.OPENAI_API_KEY }} + AZURE_OPENAI_API_KEY: ${{ secrets.AZURE_OPENAI_API_KEY }} + AZURE_OPENAI_API_BASE: ${{ secrets.AZURE_OPENAI_API_BASE }} + OAI_CONFIG_LIST: ${{ secrets.OAI_CONFIG_LIST }} + run: | + coverage run -a -m pytest test/agentchat/contrib/gpt_assistant_agent.py + coverage xml + - name: Upload coverage to Codecov + uses: codecov/codecov-action@v3 + with: + file: ./coverage.xml + flags: unittests diff --git a/.github/workflows/contrib-tests.yml b/.github/workflows/contrib-tests.yml index 70d41de30c27..21cdffb4761d 100644 --- a/.github/workflows/contrib-tests.yml +++ b/.github/workflows/contrib-tests.yml @@ -82,3 +82,29 @@ jobs: if: matrix.python-version != '3.10' run: | pytest test/agentchat/contrib/test_compressible_agent.py + + GPTAssistantAgent: + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: [ubuntu-latest, macos-latest, windows-2019] + python-version: ["3.8", "3.9", "3.10", "3.11"] + steps: + - uses: actions/checkout@v3 + - name: Set up Python ${{ matrix.python-version }} + uses: actions/setup-python@v4 + with: + python-version: ${{ matrix.python-version }} + - name: Install packages and dependencies for all tests + run: | + python -m pip install --upgrade pip wheel + pip install pytest + - name: Install packages and dependencies for GPTAssistantAgent + run: | + pip install -e . + pip uninstall -y openai + - name: Test GPTAssistantAgent + if: matrix.python-version != '3.10' + run: | + pytest test/agentchat/contrib/gpt_assistant_agent.py From 5e3f39f4357983b8936a06e44b6633054e8d144d Mon Sep 17 00:00:00 2001 From: kevin666aa Date: Sat, 11 Nov 2023 23:40:32 -0500 Subject: [PATCH 08/12] update --- .github/workflows/contrib-openai.yml | 2 +- .github/workflows/contrib-tests.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/contrib-openai.yml b/.github/workflows/contrib-openai.yml index 46335517987f..483f43c48721 100644 --- a/.github/workflows/contrib-openai.yml +++ b/.github/workflows/contrib-openai.yml @@ -131,7 +131,7 @@ jobs: AZURE_OPENAI_API_BASE: ${{ secrets.AZURE_OPENAI_API_BASE }} OAI_CONFIG_LIST: ${{ secrets.OAI_CONFIG_LIST }} run: | - coverage run -a -m pytest test/agentchat/contrib/gpt_assistant_agent.py + coverage run -a -m pytest test/agentchat/contrib/test_gpt_assistant.py coverage xml - name: Upload coverage to Codecov uses: codecov/codecov-action@v3 diff --git a/.github/workflows/contrib-tests.yml b/.github/workflows/contrib-tests.yml index 21cdffb4761d..17b43181efa6 100644 --- a/.github/workflows/contrib-tests.yml +++ b/.github/workflows/contrib-tests.yml @@ -107,4 +107,4 @@ jobs: - name: Test GPTAssistantAgent if: matrix.python-version != '3.10' run: | - pytest test/agentchat/contrib/gpt_assistant_agent.py + pytest test/agentchat/contrib/test_gpt_assistant.py From 23ec00fedc255cb76209de757a84d16872c668ac Mon Sep 17 00:00:00 2001 From: kevin666aa Date: Sat, 11 Nov 2023 23:46:00 -0500 Subject: [PATCH 09/12] fix test_client_stream --- test/oai/test_client_stream.py | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/test/oai/test_client_stream.py b/test/oai/test_client_stream.py index a6081a193ac4..5bc3ee3bc587 100644 --- a/test/oai/test_client_stream.py +++ b/test/oai/test_client_stream.py @@ -18,7 +18,7 @@ def test_aoai_chat_completion_stream(): filter_dict={"api_type": ["azure"], "model": ["gpt-3.5-turbo"]}, ) client = OpenAIWrapper(config_list=config_list) - response = client.create(messages=[{"role": "user", "content": "2+2="}], seed=None, stream=True) + response = client.create(messages=[{"role": "user", "content": "2+2="}], stream=True) print(response) print(client.extract_text_or_function_call(response)) @@ -31,7 +31,7 @@ def test_chat_completion_stream(): filter_dict={"model": ["gpt-3.5-turbo"]}, ) client = OpenAIWrapper(config_list=config_list) - response = client.create(messages=[{"role": "user", "content": "1+1="}], seed=None, stream=True) + response = client.create(messages=[{"role": "user", "content": "1+1="}], stream=True) print(response) print(client.extract_text_or_function_call(response)) @@ -63,7 +63,6 @@ def test_chat_functions_stream(): response = client.create( messages=[{"role": "user", "content": "What's the weather like today in San Francisco?"}], functions=functions, - seed=None, stream=True, ) print(response) @@ -74,7 +73,7 @@ def test_chat_functions_stream(): def test_completion_stream(): config_list = config_list_openai_aoai(KEY_LOC) client = OpenAIWrapper(config_list=config_list) - response = client.create(prompt="1+1=", model="gpt-3.5-turbo-instruct", seed=None, stream=True) + response = client.create(prompt="1+1=", model="gpt-3.5-turbo-instruct", stream=True) print(response) print(client.extract_text_or_function_call(response)) From 8ecea15457569ed50f33b2912275be070e9a46cc Mon Sep 17 00:00:00 2001 From: kevin666aa Date: Sun, 12 Nov 2023 00:15:06 -0500 Subject: [PATCH 10/12] comment out test_hierarchy_ --- test/test_notebook.py | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/test/test_notebook.py b/test/test_notebook.py index 014724c1db0d..5d6bbccf8767 100644 --- a/test/test_notebook.py +++ b/test/test_notebook.py @@ -84,12 +84,12 @@ def _test_oai_chatgpt_gpt4(save=False): run_notebook("oai_chatgpt_gpt4.ipynb", save=save) -@pytest.mark.skipif( - skip or not sys.version.startswith("3.10"), - reason="do not run if openai is not installed or py!=3.10", -) -def test_hierarchy_flow_using_select_speaker(save=False): - run_notebook("agentchat_hierarchy_flow_using_select_speaker.ipynb", save=save) +# @pytest.mark.skipif( +# skip or not sys.version.startswith("3.10"), +# reason="do not run if openai is not installed or py!=3.10", +# ) +# def test_hierarchy_flow_using_select_speaker(save=False): +# run_notebook("agentchat_hierarchy_flow_using_select_speaker.ipynb", save=save) if __name__ == "__main__": From 955cf9734784c1d5900124b9afa7376bc0d3fc7d Mon Sep 17 00:00:00 2001 From: gagb Date: Sun, 12 Nov 2023 07:01:45 +0000 Subject: [PATCH 11/12] Add basic gptassistant notebook - also improve logging in gpt assistant --- .../agentchat/contrib/gpt_assistant_agent.py | 2 + ...ntchat_oai_assistant_twoagents_basic.ipynb | 214 ++++++++++++++++++ 2 files changed, 216 insertions(+) create mode 100644 notebook/agentchat_oai_assistant_twoagents_basic.ipynb diff --git a/autogen/agentchat/contrib/gpt_assistant_agent.py b/autogen/agentchat/contrib/gpt_assistant_agent.py index d226e8ed01d2..5d1e58b52369 100644 --- a/autogen/agentchat/contrib/gpt_assistant_agent.py +++ b/autogen/agentchat/contrib/gpt_assistant_agent.py @@ -52,6 +52,7 @@ def __init__( self._openai_client = oai_wrapper._clients[0] openai_assistant_id = llm_config.get("assistant_id", None) if openai_assistant_id is None: + logger.warning("assistant_id was None, creating a new assistant") # create a new assistant if instructions is None: logger.warning( @@ -346,4 +347,5 @@ def get_assistant_instructions(self): def delete_assistant(self): """Delete the assistant from OAI assistant API""" + logger.warning("Permanently deleting assistant...") self._openai_client.beta.assistants.delete(self.assistant_id) diff --git a/notebook/agentchat_oai_assistant_twoagents_basic.ipynb b/notebook/agentchat_oai_assistant_twoagents_basic.ipynb new file mode 100644 index 000000000000..bb9a2ec14714 --- /dev/null +++ b/notebook/agentchat_oai_assistant_twoagents_basic.ipynb @@ -0,0 +1,214 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## OpenAI Assistants in AutoGen\n", + "\n", + "This notebook shows a very basic example of the `GPTAssistantAgent` working with\n", + "`UserProxyAgent` in AutoGen." + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "assistant_id was None, creating a new assistant\n" + ] + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\u001b[33muser_proxy\u001b[0m (to assistant):\n", + "\n", + "Print hello world\n", + "\n", + "--------------------------------------------------------------------------------\n", + "\u001b[33massistant\u001b[0m (to user_proxy):\n", + "\n", + "```python\n", + "print(\"Hello, World!\")\n", + "```\n", + "\n", + "Please run this Python code to print \"Hello, World!\" to the console.\n", + "\n", + "\n", + "--------------------------------------------------------------------------------\n", + "\u001b[31m\n", + ">>>>>>>> EXECUTING CODE BLOCK 0 (inferred language is python)...\u001b[0m\n" + ] + }, + { + "name": "stderr", + "output_type": "stream", + "text": [ + "execute_code was called without specifying a value for use_docker. Since the python docker package is not available, code will be run natively. Note: this fallback behavior is subject to change\n" + ] + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\u001b[33muser_proxy\u001b[0m (to assistant):\n", + "\n", + "exitcode: 0 (execution succeeded)\n", + "Code output: \n", + "Hello, World!\n", + "\n", + "\n", + "--------------------------------------------------------------------------------\n", + "\u001b[33massistant\u001b[0m (to user_proxy):\n", + "\n", + "The code executed successfully and printed \"Hello, World!\" as expected.\n", + "\n", + "TERMINATE\n", + "\n", + "\n", + "--------------------------------------------------------------------------------\n" + ] + } + ], + "source": [ + "import logging\n", + "import os\n", + " \n", + "from autogen import config_list_from_json\n", + "from autogen import AssistantAgent\n", + "from autogen.agentchat.contrib.gpt_assistant_agent import GPTAssistantAgent\n", + "from autogen import UserProxyAgent\n", + "\n", + "logger = logging.getLogger(__name__)\n", + "logger.setLevel(logging.WARNING)\n", + "\n", + "assistant_id = os.environ.get(\"ASSISTANT_ID\", None)\n", + "\n", + "config_list = config_list_from_json(\"OAI_CONFIG_LIST\")\n", + "llm_config = {\n", + " \"config_list\": config_list,\n", + " \"assistant_id\": assistant_id\n", + "}\n", + "\n", + "gpt_assistant = GPTAssistantAgent(name=\"assistant\",\n", + " instructions=AssistantAgent.DEFAULT_SYSTEM_MESSAGE,\n", + " llm_config=llm_config)\n", + "\n", + "user_proxy = UserProxyAgent(name=\"user_proxy\",\n", + " code_execution_config={\n", + " \"work_dir\": \"coding\"\n", + " },\n", + " is_termination_msg=lambda msg: \"TERMINATE\" in msg[\"content\"],\n", + " human_input_mode=\"NEVER\",\n", + " max_consecutive_auto_reply=1)\n", + "user_proxy.initiate_chat(gpt_assistant, message=\"Print hello world\")" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\u001b[33muser_proxy\u001b[0m (to assistant):\n", + "\n", + "Write py code to eval 2 + 2\n", + "\n", + "--------------------------------------------------------------------------------\n", + "\u001b[33massistant\u001b[0m (to user_proxy):\n", + "\n", + "```python\n", + "# Let's write a simple Python code to evaluate 2 + 2 and print the result.\n", + "\n", + "result = 2 + 2\n", + "print(result)\n", + "```\n", + "\n", + "\n", + "--------------------------------------------------------------------------------\n", + "\u001b[31m\n", + ">>>>>>>> EXECUTING CODE BLOCK 0 (inferred language is python)...\u001b[0m\n" + ] + }, + { + "name": "stderr", + "output_type": "stream", + "text": [ + "execute_code was called without specifying a value for use_docker. Since the python docker package is not available, code will be run natively. Note: this fallback behavior is subject to change\n" + ] + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\u001b[33muser_proxy\u001b[0m (to assistant):\n", + "\n", + "exitcode: 0 (execution succeeded)\n", + "Code output: \n", + "4\n", + "\n", + "\n", + "--------------------------------------------------------------------------------\n", + "\u001b[33massistant\u001b[0m (to user_proxy):\n", + "\n", + "The Python code was executed successfully and the result of evaluating 2 + 2 is 4.\n", + "\n", + "TERMINATE\n", + "\n", + "\n", + "--------------------------------------------------------------------------------\n" + ] + } + ], + "source": [ + "user_proxy.initiate_chat(gpt_assistant, message=\"Write py code to eval 2 + 2\", clear_history=True)" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Permanently deleting assistant...\n" + ] + } + ], + "source": [ + "gpt_assistant.delete_assistant()" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.10.12" + } + }, + "nbformat": 4, + "nbformat_minor": 2 +} From 8f13477f776d516199653233205d0a9c24ac8eaf Mon Sep 17 00:00:00 2001 From: Chi Wang Date: Sun, 12 Nov 2023 05:20:21 -0800 Subject: [PATCH 12/12] Update notebook/agentchat_oai_assistant_twoagents_basic.ipynb Co-authored-by: Qingyun Wu --- notebook/agentchat_oai_assistant_twoagents_basic.ipynb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/notebook/agentchat_oai_assistant_twoagents_basic.ipynb b/notebook/agentchat_oai_assistant_twoagents_basic.ipynb index bb9a2ec14714..44660bf960a7 100644 --- a/notebook/agentchat_oai_assistant_twoagents_basic.ipynb +++ b/notebook/agentchat_oai_assistant_twoagents_basic.ipynb @@ -6,7 +6,7 @@ "source": [ "## OpenAI Assistants in AutoGen\n", "\n", - "This notebook shows a very basic example of the `GPTAssistantAgent` working with\n", + "This notebook shows a very basic example of the [`GPTAssistantAgent`](https://github.com/microsoft/autogen/blob/main/autogen/agentchat/contrib/gpt_assistant_agent.py#L16C43-L16C43), which is an experimental AutoGen agent class that leverages the [OpenAI Assistant API](https://platform.openai.com/docs/assistants/overview) for conversational capabilities, working with\n", "`UserProxyAgent` in AutoGen." ] },