# What is this? ## Unit Tests for OpenAI Assistants API import json import os import sys import traceback from dotenv import load_dotenv load_dotenv() sys.path.insert( 0, os.path.abspath("../..") ) # Adds the parent directory to the system path import asyncio import logging import pytest from openai.types.beta.assistant import Assistant from typing_extensions import override import litellm from litellm import create_thread, get_thread from litellm.llms.openai.openai import ( AssistantEventHandler, AsyncAssistantEventHandler, AsyncCursorPage, MessageData, OpenAIAssistantsAPI, ) from litellm.llms.openai.openai import OpenAIMessage as Message from litellm.llms.openai.openai import SyncCursorPage, Thread """ V0 Scope: - Add Message -> `/v1/threads/{thread_id}/messages` - Run Thread -> `/v1/threads/{thread_id}/run` """ @pytest.mark.parametrize("provider", ["openai", "azure"]) @pytest.mark.parametrize( "sync_mode", [True, False], ) @pytest.mark.asyncio async def test_get_assistants(provider, sync_mode): data = { "custom_llm_provider": provider, } if provider == "azure": data["api_version"] = "2024-02-15-preview" if sync_mode == True: assistants = litellm.get_assistants(**data) assert isinstance(assistants, SyncCursorPage) else: assistants = await litellm.aget_assistants(**data) assert isinstance(assistants, AsyncCursorPage) @pytest.mark.parametrize("provider", ["azure", "openai"]) @pytest.mark.parametrize( "sync_mode", [True, False], ) @pytest.mark.asyncio() @pytest.mark.flaky(retries=3, delay=1) async def test_create_delete_assistants(provider, sync_mode): litellm.ssl_verify = False model = "gpt-4-turbo" if provider == "azure": os.environ["AZURE_API_VERSION"] = "2024-05-01-preview" model = "chatgpt-v-2" if sync_mode == True: assistant = litellm.create_assistants( custom_llm_provider=provider, model=model, instructions="You are a personal math tutor. When asked a question, write and run Python code to answer the question.", name="Math Tutor", tools=[{"type": "code_interpreter"}], ) print("New assistants", assistant) assert isinstance(assistant, Assistant) assert ( assistant.instructions == "You are a personal math tutor. When asked a question, write and run Python code to answer the question." ) assert assistant.id is not None # delete the created assistant response = litellm.delete_assistant( custom_llm_provider=provider, assistant_id=assistant.id ) print("Response deleting assistant", response) assert response.id == assistant.id else: assistant = await litellm.acreate_assistants( custom_llm_provider=provider, model=model, instructions="You are a personal math tutor. When asked a question, write and run Python code to answer the question.", name="Math Tutor", tools=[{"type": "code_interpreter"}], ) print("New assistants", assistant) assert isinstance(assistant, Assistant) assert ( assistant.instructions == "You are a personal math tutor. When asked a question, write and run Python code to answer the question." ) assert assistant.id is not None response = await litellm.adelete_assistant( custom_llm_provider=provider, assistant_id=assistant.id ) print("Response deleting assistant", response) assert response.id == assistant.id @pytest.mark.parametrize("provider", ["openai", "azure"]) @pytest.mark.parametrize("sync_mode", [True, False]) @pytest.mark.asyncio async def test_create_thread_litellm(sync_mode, provider) -> Thread: message: MessageData = {"role": "user", "content": "Hey, how's it going?"} # type: ignore data = { "custom_llm_provider": provider, "message": [message], } if provider == "azure": data["api_version"] = "2024-02-15-preview" if sync_mode: new_thread = create_thread(**data) else: new_thread = await litellm.acreate_thread(**data) assert isinstance( new_thread, Thread ), f"type of thread={type(new_thread)}. Expected Thread-type" return new_thread @pytest.mark.parametrize("provider", ["openai", "azure"]) @pytest.mark.parametrize("sync_mode", [True, False]) @pytest.mark.asyncio async def test_get_thread_litellm(provider, sync_mode): new_thread = test_create_thread_litellm(sync_mode, provider) if asyncio.iscoroutine(new_thread): _new_thread = await new_thread else: _new_thread = new_thread data = { "custom_llm_provider": provider, "thread_id": _new_thread.id, } if provider == "azure": data["api_version"] = "2024-02-15-preview" if sync_mode: received_thread = get_thread(**data) else: received_thread = await litellm.aget_thread(**data) assert isinstance( received_thread, Thread ), f"type of thread={type(received_thread)}. Expected Thread-type" return new_thread @pytest.mark.parametrize("provider", ["openai", "azure"]) @pytest.mark.parametrize("sync_mode", [True, False]) @pytest.mark.asyncio async def test_add_message_litellm(sync_mode, provider): message: MessageData = {"role": "user", "content": "Hey, how's it going?"} # type: ignore new_thread = test_create_thread_litellm(sync_mode, provider) if asyncio.iscoroutine(new_thread): _new_thread = await new_thread else: _new_thread = new_thread # add message to thread message: MessageData = {"role": "user", "content": "Hey, how's it going?"} # type: ignore data = {"custom_llm_provider": provider, "thread_id": _new_thread.id, **message} if provider == "azure": data["api_version"] = "2024-02-15-preview" if sync_mode: added_message = litellm.add_message(**data) else: added_message = await litellm.a_add_message(**data) print(f"added message: {added_message}") assert isinstance(added_message, Message) @pytest.mark.parametrize( "provider", [ "azure", "openai", ], ) # @pytest.mark.parametrize( "sync_mode", [ True, False, ], ) @pytest.mark.parametrize( "is_streaming", [True, False], ) # @pytest.mark.asyncio @pytest.mark.flaky(retries=3, delay=1) async def test_aarun_thread_litellm(sync_mode, provider, is_streaming): """ - Get Assistants - Create thread - Create run w/ Assistants + Thread """ import openai try: if sync_mode: assistants = litellm.get_assistants(custom_llm_provider=provider) else: assistants = await litellm.aget_assistants(custom_llm_provider=provider) ## get the first assistant ### try: assistant_id = assistants.data[0].id except IndexError: pytest.skip("No assistants found") new_thread = test_create_thread_litellm(sync_mode=sync_mode, provider=provider) if asyncio.iscoroutine(new_thread): _new_thread = await new_thread else: _new_thread = new_thread thread_id = _new_thread.id # add message to thread message: MessageData = {"role": "user", "content": "Hey, how's it going?"} # type: ignore data = {"custom_llm_provider": provider, "thread_id": _new_thread.id, **message} if sync_mode: added_message = litellm.add_message(**data) if is_streaming: run = litellm.run_thread_stream(assistant_id=assistant_id, **data) with run as run: assert isinstance(run, AssistantEventHandler) print(run) run.until_done() else: run = litellm.run_thread( assistant_id=assistant_id, stream=is_streaming, **data ) if run.status == "completed": messages = litellm.get_messages( thread_id=_new_thread.id, custom_llm_provider=provider ) assert isinstance(messages.data[0], Message) else: pytest.fail( "An unexpected error occurred when running the thread, {}".format( run ) ) else: added_message = await litellm.a_add_message(**data) if is_streaming: run = litellm.arun_thread_stream(assistant_id=assistant_id, **data) async with run as run: print(f"run: {run}") assert isinstance( run, AsyncAssistantEventHandler, ) print(run) await run.until_done() else: run = await litellm.arun_thread( custom_llm_provider=provider, thread_id=thread_id, assistant_id=assistant_id, ) if run.status == "completed": messages = await litellm.aget_messages( thread_id=_new_thread.id, custom_llm_provider=provider ) assert isinstance(messages.data[0], Message) else: pytest.fail( "An unexpected error occurred when running the thread, {}".format( run ) ) except openai.APIError as e: pass