-
-
Notifications
You must be signed in to change notification settings - Fork 406
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
1da3848
commit 5ff27fe
Showing
1 changed file
with
109 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,109 @@ | ||
import asyncio | ||
import json | ||
from json import JSONDecodeError | ||
from typing import List, Union | ||
|
||
from langchain_core.agents import AgentAction, AgentActionMessageLog, AgentFinish | ||
from langchain_core.exceptions import OutputParserException | ||
from langchain_core.messages import ( | ||
AIMessage, | ||
BaseMessage, | ||
) | ||
from langchain_core.outputs import ChatGeneration, Generation | ||
|
||
from langchain.agents.agent import AgentOutputParser | ||
|
||
|
||
class OpenAIFunctionsAgentOutputParser(AgentOutputParser): | ||
"""Parses a message into agent action/finish. | ||
Is meant to be used with OpenAI models, as it relies on the specific | ||
function_call parameter from OpenAI to convey what tools to use. | ||
If a function_call parameter is passed, then that is used to get | ||
the tool and tool input. | ||
If one is not passed, then the AIMessage is assumed to be the final output. | ||
""" | ||
|
||
@property | ||
def _type(self) -> str: | ||
return "openai-functions-agent" | ||
|
||
@staticmethod | ||
def _parse_ai_message(message: BaseMessage) -> Union[AgentAction, AgentFinish]: | ||
"""Parse an AI message.""" | ||
if not isinstance(message, AIMessage): | ||
raise TypeError(f"Expected an AI message got {type(message)}") | ||
|
||
function_call = message.additional_kwargs.get("function_call", {}) | ||
|
||
if function_call: | ||
function_name = function_call["name"] | ||
try: | ||
if len(function_call["arguments"].strip()) == 0: | ||
# OpenAI returns an empty string for functions containing no args | ||
_tool_input = {} | ||
else: | ||
# otherwise it returns a json object | ||
_tool_input = json.loads(function_call["arguments"]) | ||
except JSONDecodeError: | ||
if function_name == "python": | ||
code = function_call["arguments"] | ||
_tool_input = { | ||
"code": code, | ||
} | ||
else: | ||
raise OutputParserException( | ||
f"Could not parse tool input: {function_call} because " | ||
f"the `arguments` is not valid JSON." | ||
) | ||
|
||
# HACK HACK HACK: | ||
# The code that encodes tool input into Open AI uses a special variable | ||
# name called `__arg1` to handle old style tools that do not expose a | ||
# schema and expect a single string argument as an input. | ||
# We unpack the argument here if it exists. | ||
# Open AI does not support passing in a JSON array as an argument. | ||
if "__arg1" in _tool_input: | ||
tool_input = _tool_input["__arg1"] | ||
else: | ||
tool_input = _tool_input | ||
|
||
content_msg = f"responded: {message.content}\n" if message.content else "\n" | ||
log = f"\nInvoking: `{function_name}` with `{tool_input}`\n{content_msg}\n" | ||
return AgentActionMessageLog( | ||
tool=function_name, | ||
tool_input=tool_input, | ||
log=log, | ||
message_log=[message], | ||
) | ||
|
||
return AgentFinish( | ||
return_values={"output": message.content}, log=str(message.content) | ||
) | ||
|
||
def parse_result( | ||
self, result: List[Generation], *, partial: bool = False | ||
) -> Union[AgentAction, AgentFinish]: | ||
if not isinstance(result[0], ChatGeneration): | ||
raise ValueError("This output parser only works on ChatGeneration output") | ||
message = result[0].message | ||
return self._parse_ai_message(message) | ||
|
||
async def aparse_result( | ||
self, result: List[Generation], *, partial: bool = False | ||
) -> Union[AgentAction, AgentFinish]: | ||
return await asyncio.get_running_loop().run_in_executor( | ||
None, self.parse_result, result | ||
) | ||
|
||
def parse(self, text: str) -> Union[AgentAction, AgentFinish]: | ||
raise ValueError("Can only parse messages") | ||
|
||
|
||
def patch() -> None: | ||
"""Patch the parser.""" | ||
from langchain.agents import openai_functions_agent | ||
|
||
openai_functions_agent.OpenAIFunctionsAgentOutputParser = OpenAIFunctionsAgentOutputParser # type: ignore |