From a31106565aaeba6d2861d072aa42517bbbb2240c Mon Sep 17 00:00:00 2001 From: Bagatur Date: Thu, 8 Feb 2024 23:00:56 -0800 Subject: [PATCH 1/6] langchain[minor]: openai tools structured_output_chain --- .../chains/structured_output/base.py | 179 ++++++++++++++++-- .../langchain/output_parsers/openai_tools.py | 38 +++- 2 files changed, 190 insertions(+), 27 deletions(-) diff --git a/libs/langchain/langchain/chains/structured_output/base.py b/libs/langchain/langchain/chains/structured_output/base.py index d825c1fe7c896..5f20af8c4bc54 100644 --- a/libs/langchain/langchain/chains/structured_output/base.py +++ b/libs/langchain/langchain/chains/structured_output/base.py @@ -9,9 +9,16 @@ from langchain_core.prompts import BasePromptTemplate from langchain_core.pydantic_v1 import BaseModel from langchain_core.runnables import Runnable -from langchain_core.utils.function_calling import convert_to_openai_function +from langchain_core.utils.function_calling import ( + convert_to_openai_function, + convert_to_openai_tool, +) -from langchain.output_parsers import PydanticOutputParser +from langchain.output_parsers import ( + JsonOutputKeyToolsParser, + PydanticOutputParser, + PydanticToolsParser, +) from langchain.output_parsers.openai_functions import ( JsonOutputFunctionsParser, PydanticAttrOutputFunctionsParser, @@ -106,15 +113,17 @@ class RecordDog(BaseModel): return prompt | llm.bind(**llm_kwargs) | output_parser -# TODO: implement mode='openai-tools'. def create_structured_output_runnable( output_schema: Union[Dict[str, Any], Type[BaseModel]], llm: Runnable, prompt: BasePromptTemplate, *, output_parser: Optional[Union[BaseOutputParser, BaseGenerationOutputParser]] = None, - mode: Literal["openai-functions", "openai-json"] = "openai-functions", - enforce_single_function_usage: bool = True, + enforce_function_usage: bool = True, + return_single: bool = True, + mode: Literal[ + "openai-functions", "openai-tools", "openai-json" + ] = "openai-functions", **kwargs: Any, ) -> Runnable: """Create a runnable for extracting structured outputs. @@ -135,19 +144,101 @@ def create_structured_output_runnable( in, then the OutputParser will try to parse outputs using the pydantic class. Otherwise model outputs will be parsed as JSON. mode: How structured outputs are extracted from the model. If 'openai-functions' - then OpenAI function calling is used. If 'openai-json' then OpenAI model + then OpenAI function calling is used with the deprecated 'functions', + 'function_call' schema. If 'openai-tools' then OpenAI function + calling with the latest 'tools', 'tool_choice' schema is used. This is + recommended over 'openai-functions'. If 'openai-json' then OpenAI model with response_format set to JSON is used. - enforce_single_function_usage: Only used if mode is 'openai-functions'. Only - used if a single function is passed in. If - True, then the model will be forced to use the given function. If False, - then the model will be given the option to use the given function or not. + enforce_function_usage: Only applies when mode is 'openai-tools' or + 'openai-functions'. If True, then the model will be forced to use the given + output schema. If False, then the model can elect whether to use the output + schema. + return_single: Only applies when mode is 'openai-tools'. Whether to a list of + structured outputs or a single one. If True and model does not return any + structured outputs then chain output is None. If False and model does not + return any structured outputs then chain output is an empty list. **kwargs: Additional named arguments. Returns: - A runnable sequence that will return a structured output matching the given + A runnable sequence that will return a structured output(s) matching the given output_schema. + + OpenAI tools example with Pydantic schema (mode='openai-tools'): + .. code-block:: python + + from typing import Optional + + from langchain.chains import create_structured_output_runnable + from langchain_openai import ChatOpenAI + from langchain_core.pydantic_v1 import BaseModel, Field + + + class RecordDog(BaseModel): + '''Record some identifying information about a dog.''' - OpenAI functions example: + name: str = Field(..., description="The dog's name") + color: str = Field(..., description="The dog's color") + fav_food: Optional[str] = Field(None, description="The dog's favorite food") + + llm = ChatOpenAI(model="gpt-3.5-turbo-0125", temperature=0) + structured_llm = create_structured_output_runnable( + RecordDog, + llm, + mode="openai-tools", + enforce_function_usage=True, + return_single=True + ) + structured_llm.invoke("Harry was a chubby brown beagle who loved chicken") + # -> RecordDog(name="Harry", color="brown", fav_food="chicken") + + OpenAI tools example with dict schema (mode="openai-tools"): + .. code-block:: python + + from typing import Optional + + from langchain.chains import create_structured_output_runnable + from langchain_openai import ChatOpenAI + + + dog_schema = { + "type": "function", + "function": { + "name": "record_dog", + "description": "Record some identifying information about a dog.", + "parameters": { + "type": "object", + "properties": { + "name": { + "description": "The dog's name", + "type": "string" + }, + "color": { + "description": "The dog's color", + "type": "string" + }, + "fav_food": { + "description": "The dog's favorite food", + "type": "string" + } + }, + "required": ["name", "color"] + } + } + } + + + llm = ChatOpenAI(model="gpt-3.5-turbo-0125", temperature=0) + structured_llm = create_structured_output_runnable( + doc_schema, + llm, + mode="openai-tools", + enforce_function_usage=True, + return_single=True + ) + structured_llm.invoke("Harry was a chubby brown beagle who loved chicken") + # -> {'name': 'Harry', 'color': 'brown', 'fav_food': 'chicken'} + + OpenAI functions example (mode="openai-functions"): .. code-block:: python from typing import Optional @@ -176,7 +267,7 @@ class Dog(BaseModel): chain.invoke({"input": "Harry was a chubby brown beagle who loved chicken"}) # -> Dog(name="Harry", color="brown", fav_food="chicken") - OpenAI json response format example: + OpenAI json response format example (mode="openai-json"): .. code-block:: python from typing import Optional @@ -208,7 +299,22 @@ class Dog(BaseModel): chain = create_structured_output_runnable(Dog, llm, prompt, mode="openai-json") chain.invoke({"input": "Harry was a chubby brown beagle who loved chicken"}) """ # noqa: E501 - if mode == "openai-functions": + if mode == "openai-tools": + return _create_openai_tools_runnable( + output_schema, + llm, + prompt=prompt, + output_parser=output_parser, + enforce_tool_usage=enforce_function_usage, + return_single=return_single, + ) + + elif mode == "openai-functions": + # for backwards compatibility + enforce_single_function_usage = kwargs.get( + "enforce_single_function_usage", enforce_function_usage + ) + return _create_openai_functions_structured_output_runnable( output_schema, llm, @@ -223,11 +329,51 @@ class Dog(BaseModel): ) else: raise ValueError( - f"Invalid mode {mode}. Expected one of 'openai-functions', " + f"Invalid mode {mode}. Expected one of 'openai-tools', 'openai-functions', " f"'openai-json'." ) +def _create_openai_tools_runnable( + tool: Union[Dict[str, Any], Type[BaseModel], Callable], + llm: Runnable, + *, + prompt: Optional[BasePromptTemplate], + output_parser: Optional[Union[BaseOutputParser, BaseGenerationOutputParser]], + enforce_tool_usage: bool, + return_single: bool, +) -> Runnable: + oai_tool = convert_to_openai_tool(tool) + llm_kwargs: Dict[str, Any] = {"tools": [oai_tool]} + if enforce_tool_usage: + llm_kwargs["tool_choice"] = { + "type": "function", + "function": {"name": oai_tool["function"]["name"]}, + } + output_parser = output_parser or _get_openai_tool_output_parser( + tool, return_single=return_single + ) + if prompt: + return prompt | llm.bind(**llm_kwargs) | output_parser + else: + return llm.bind(**llm_kwargs) | output_parser + + +def _get_openai_tool_output_parser( + tool: Union[Dict[str, Any], Type[BaseModel], Callable], return_single: bool = False +) -> Union[BaseOutputParser, BaseGenerationOutputParser]: + if isinstance(tool, type) and issubclass(tool, BaseModel): + output_parser: Union[ + BaseOutputParser, BaseGenerationOutputParser + ] = PydanticToolsParser(tools=[tool], return_single=return_single) + else: + key_name = convert_to_openai_tool(tool)["function"]["name"] + output_parser = JsonOutputKeyToolsParser( + return_single=return_single, key_name=key_name + ) + return output_parser + + def get_openai_output_parser( functions: Sequence[Union[Dict[str, Any], Type[BaseModel], Callable]], ) -> Union[BaseOutputParser, BaseGenerationOutputParser]: @@ -244,11 +390,10 @@ def get_openai_output_parser( not a Pydantic class, then the output parser will automatically extract only the function arguments and not the function name. """ - function_names = [convert_to_openai_function(f)["name"] for f in functions] if isinstance(functions[0], type) and issubclass(functions[0], BaseModel): if len(functions) > 1: pydantic_schema: Union[Dict, Type[BaseModel]] = { - name: fn for name, fn in zip(function_names, functions) + convert_to_openai_function(fn)["name"]: fn for fn in functions } else: pydantic_schema = functions[0] diff --git a/libs/langchain/langchain/output_parsers/openai_tools.py b/libs/langchain/langchain/output_parsers/openai_tools.py index 045e32686efcf..900b302245488 100644 --- a/libs/langchain/langchain/output_parsers/openai_tools.py +++ b/libs/langchain/langchain/output_parsers/openai_tools.py @@ -22,6 +22,8 @@ class JsonOutputToolsParser(BaseGenerationOutputParser[Any]): """ return_id: bool = False """Whether to return the tool call id.""" + return_single: bool = False + """Whether to return only the first tool call.""" def parse_result(self, result: List[Generation], *, partial: bool = False) -> Any: generation = result[0] @@ -65,6 +67,8 @@ def parse_result(self, result: List[Generation], *, partial: bool = False) -> An final_tools.append(parsed) if exceptions: raise OutputParserException("\n\n".join(exceptions)) + if self.return_single: + return final_tools[0] if final_tools else None return final_tools @@ -73,21 +77,29 @@ class JsonOutputKeyToolsParser(JsonOutputToolsParser): key_name: str """The type of tools to return.""" - return_single: bool = False - """Whether to return only the first tool call.""" def __init__(self, key_name: str, **kwargs: Any) -> None: """Allow init with positional args.""" super().__init__(key_name=key_name, **kwargs) def parse_result(self, result: List[Generation], *, partial: bool = False) -> Any: - results = super().parse_result(result, partial=partial) - results = [res for res in results if res["type"] == self.key_name] - if not self.return_id: - results = [res["args"] for res in results] + parsed_result = super().parse_result(result, partial=partial) if self.return_single: - return results[0] if results else None - return results + single_result = ( + parsed_result + if parsed_result and parsed_result["type"] == self.key_name + else None + ) + if self.return_id: + return single_result + elif single_result: + return single_result["args"] + else: + return None + parsed_result = [res for res in parsed_result if res["type"] == self.key_name] + if not self.return_id: + parsed_result = [res["args"] for res in parsed_result] + return parsed_result class PydanticToolsParser(JsonOutputToolsParser): @@ -96,6 +108,12 @@ class PydanticToolsParser(JsonOutputToolsParser): tools: List[Type[BaseModel]] def parse_result(self, result: List[Generation], *, partial: bool = False) -> Any: - results = super().parse_result(result, partial=partial) + parsed_result = super().parse_result(result, partial=partial) name_dict = {tool.__name__: tool for tool in self.tools} - return [name_dict[res["type"]](**res["args"]) for res in results] + if self.return_single: + return ( + name_dict[parsed_result["type"]](**parsed_result["args"]) + if parsed_result + else None + ) + return [name_dict[res["type"]](**res["args"]) for res in parsed_result] From 7ea03dea5fff1458efec331dbeccb86aaae6ad19 Mon Sep 17 00:00:00 2001 From: Eugene Yurtsev Date: Thu, 22 Feb 2024 14:07:17 -0500 Subject: [PATCH 2/6] x --- .../langchain/chains/ernie_functions/base.py | 50 +++++++-------- .../chains/structured_output/base.py | 62 +++++++++++++------ .../langchain/output_parsers/openai_tools.py | 18 ++++-- 3 files changed, 80 insertions(+), 50 deletions(-) diff --git a/libs/langchain/langchain/chains/ernie_functions/base.py b/libs/langchain/langchain/chains/ernie_functions/base.py index 9a12b70c97a3c..2eb2a5a72f208 100644 --- a/libs/langchain/langchain/chains/ernie_functions/base.py +++ b/libs/langchain/langchain/chains/ernie_functions/base.py @@ -310,31 +310,31 @@ def create_structured_output_runnable( Example: .. code-block:: python - from typing import Optional - - from langchain.chains.ernie_functions import create_structured_output_chain - from langchain_community.chat_models import ErnieBotChat - from langchain.prompts import ChatPromptTemplate - from langchain.pydantic_v1 import BaseModel, Field - - class Dog(BaseModel): - \"\"\"Identifying information about a dog.\"\"\" - - name: str = Field(..., description="The dog's name") - color: str = Field(..., description="The dog's color") - fav_food: Optional[str] = Field(None, description="The dog's favorite food") - - llm = ErnieBotChat(model_name="ERNIE-Bot-4") - prompt = ChatPromptTemplate.from_messages( - [ - ("user", "Use the given format to extract information from the following input: {input}"), - ("assistant", "OK!"), - ("user", "Tip: Make sure to answer in the correct format"), - ] - ) - chain = create_structured_output_chain(Dog, llm, prompt) - chain.invoke({"input": "Harry was a chubby brown beagle who loved chicken"}) - # -> Dog(name="Harry", color="brown", fav_food="chicken") + from typing import Optional + + from langchain.chains.ernie_functions import create_structured_output_chain + from langchain_community.chat_models import ErnieBotChat + from langchain.prompts import ChatPromptTemplate + from langchain.pydantic_v1 import BaseModel, Field + + class Dog(BaseModel): + \"\"\"Identifying information about a dog.\"\"\" + + name: str = Field(..., description="The dog's name") + color: str = Field(..., description="The dog's color") + fav_food: Optional[str] = Field(None, description="The dog's favorite food") + + llm = ErnieBotChat(model_name="ERNIE-Bot-4") + prompt = ChatPromptTemplate.from_messages( + [ + ("user", "Use the given format to extract information from the following input: {input}"), + ("assistant", "OK!"), + ("user", "Tip: Make sure to answer in the correct format"), + ] + ) + chain = create_structured_output_chain(Dog, llm, prompt) + chain.invoke({"input": "Harry was a chubby brown beagle who loved chicken"}) + # -> Dog(name="Harry", color="brown", fav_food="chicken") """ # noqa: E501 if isinstance(output_schema, dict): function: Any = { diff --git a/libs/langchain/langchain/chains/structured_output/base.py b/libs/langchain/langchain/chains/structured_output/base.py index 5f20af8c4bc54..24342e00f20ae 100644 --- a/libs/langchain/langchain/chains/structured_output/base.py +++ b/libs/langchain/langchain/chains/structured_output/base.py @@ -33,7 +33,7 @@ def create_openai_fn_runnable( *, enforce_single_function_usage: bool = True, output_parser: Optional[Union[BaseOutputParser, BaseGenerationOutputParser]] = None, - **kwargs: Any, + **llm_kwargs: Any, ) -> Runnable: """Create a runnable sequence that uses OpenAI functions. @@ -60,6 +60,7 @@ def create_openai_fn_runnable( passed in and they are not pydantic.BaseModels, the chain output will include both the name of the function that was returned and the arguments to pass to the function. + **llm_kwargs: Additional named arguments to pass to the language model. Returns: A runnable sequence that will pass in the given functions to the model when run. @@ -106,7 +107,7 @@ class RecordDog(BaseModel): if not functions: raise ValueError("Need to pass in at least one function. Received zero.") openai_functions = [convert_to_openai_function(f) for f in functions] - llm_kwargs: Dict[str, Any] = {"functions": openai_functions, **kwargs} + llm_kwargs: Dict[str, Any] = {"functions": openai_functions, **llm_kwargs} if len(openai_functions) == 1 and enforce_single_function_usage: llm_kwargs["function_call"] = {"name": openai_functions[0]["name"]} output_parser = output_parser or get_openai_output_parser(functions) @@ -181,6 +182,12 @@ class RecordDog(BaseModel): fav_food: Optional[str] = Field(None, description="The dog's favorite food") llm = ChatOpenAI(model="gpt-3.5-turbo-0125", temperature=0) + prompt = ChatPromptTemplate.from_messages( + [ + ("system", "You are an extraction algorithm. Please extract every possible instance"), + ('human', '{input}') + ] + ) structured_llm = create_structured_output_runnable( RecordDog, llm, @@ -188,7 +195,7 @@ class RecordDog(BaseModel): enforce_function_usage=True, return_single=True ) - structured_llm.invoke("Harry was a chubby brown beagle who loved chicken") + structured_llm.invoke({"input": "Harry was a chubby brown beagle who loved chicken"}) # -> RecordDog(name="Harry", color="brown", fav_food="chicken") OpenAI tools example with dict schema (mode="openai-tools"): @@ -299,33 +306,46 @@ class Dog(BaseModel): chain = create_structured_output_runnable(Dog, llm, prompt, mode="openai-json") chain.invoke({"input": "Harry was a chubby brown beagle who loved chicken"}) """ # noqa: E501 + # for backwards compatibility + force_function_usage = kwargs.get( + "enforce_single_function_usage", enforce_function_usage + ) + if mode == "openai-tools": + # Protect against typos in kwargs + keys_in_kwargs = set(kwargs.keys()) + # Backwards compatibility keys + unrecognized_keys = keys_in_kwargs - {"enforce_single_function_usage"} + if unrecognized_keys: + raise TypeError( + f"Got an unexpected keyword argument(s): {unrecognized_keys}." + ) + return _create_openai_tools_runnable( output_schema, llm, prompt=prompt, output_parser=output_parser, - enforce_tool_usage=enforce_function_usage, - return_single=return_single, + enforce_tool_usage=force_function_usage, + first_tool_only=return_single, ) elif mode == "openai-functions": - # for backwards compatibility - enforce_single_function_usage = kwargs.get( - "enforce_single_function_usage", enforce_function_usage - ) - return _create_openai_functions_structured_output_runnable( output_schema, llm, prompt, output_parser=output_parser, - enforce_single_function_usage=enforce_single_function_usage, - **kwargs, + enforce_single_function_usage=force_function_usage, + **kwargs, # llm-specific kwargs ) elif mode == "openai-json": + if force_function_usage: + raise ValueError( + "enforce_single_function_usage is not supported for mode='openai-json'." + ) return _create_openai_json_runnable( - output_schema, llm, prompt, output_parser=output_parser, **kwargs + output_schema, llm, prompt, output_parser=output_parser ) else: raise ValueError( @@ -341,7 +361,7 @@ def _create_openai_tools_runnable( prompt: Optional[BasePromptTemplate], output_parser: Optional[Union[BaseOutputParser, BaseGenerationOutputParser]], enforce_tool_usage: bool, - return_single: bool, + first_tool_only: bool, ) -> Runnable: oai_tool = convert_to_openai_tool(tool) llm_kwargs: Dict[str, Any] = {"tools": [oai_tool]} @@ -351,7 +371,7 @@ def _create_openai_tools_runnable( "function": {"name": oai_tool["function"]["name"]}, } output_parser = output_parser or _get_openai_tool_output_parser( - tool, return_single=return_single + tool, first_tool_only=first_tool_only ) if prompt: return prompt | llm.bind(**llm_kwargs) | output_parser @@ -360,16 +380,18 @@ def _create_openai_tools_runnable( def _get_openai_tool_output_parser( - tool: Union[Dict[str, Any], Type[BaseModel], Callable], return_single: bool = False + tool: Union[Dict[str, Any], Type[BaseModel], Callable], + *, + first_tool_only: bool = False, ) -> Union[BaseOutputParser, BaseGenerationOutputParser]: if isinstance(tool, type) and issubclass(tool, BaseModel): output_parser: Union[ BaseOutputParser, BaseGenerationOutputParser - ] = PydanticToolsParser(tools=[tool], return_single=return_single) + ] = PydanticToolsParser(tools=[tool], first_tool_only=first_tool_only) else: key_name = convert_to_openai_tool(tool)["function"]["name"] output_parser = JsonOutputKeyToolsParser( - return_single=return_single, key_name=key_name + first_tool_only=first_tool_only, key_name=key_name ) return output_parser @@ -435,7 +457,7 @@ def _create_openai_functions_structured_output_runnable( prompt: BasePromptTemplate, *, output_parser: Optional[Union[BaseOutputParser, BaseGenerationOutputParser]] = None, - **kwargs: Any, + **llm_kwargs: Any, ) -> Runnable: if isinstance(output_schema, dict): function: Any = { @@ -462,5 +484,5 @@ class _OutputFormatter(BaseModel): llm, prompt, output_parser=output_parser, - **kwargs, + **llm_kwargs, ) diff --git a/libs/langchain/langchain/output_parsers/openai_tools.py b/libs/langchain/langchain/output_parsers/openai_tools.py index 900b302245488..9283499091a73 100644 --- a/libs/langchain/langchain/output_parsers/openai_tools.py +++ b/libs/langchain/langchain/output_parsers/openai_tools.py @@ -22,8 +22,16 @@ class JsonOutputToolsParser(BaseGenerationOutputParser[Any]): """ return_id: bool = False """Whether to return the tool call id.""" - return_single: bool = False - """Whether to return only the first tool call.""" + first_tool_only: bool = False + """Whether to return only the first tool call. + + If False, the result will be a list of tool calls, or an empty list + if no tool calls are found. + + If true, and multiple tool calls are found, only the first one will be returned, + and the other tool calls will be ignored. + If no tool calls are found, None will be returned. + """ def parse_result(self, result: List[Generation], *, partial: bool = False) -> Any: generation = result[0] @@ -67,7 +75,7 @@ def parse_result(self, result: List[Generation], *, partial: bool = False) -> An final_tools.append(parsed) if exceptions: raise OutputParserException("\n\n".join(exceptions)) - if self.return_single: + if self.first_tool_only: return final_tools[0] if final_tools else None return final_tools @@ -84,7 +92,7 @@ def __init__(self, key_name: str, **kwargs: Any) -> None: def parse_result(self, result: List[Generation], *, partial: bool = False) -> Any: parsed_result = super().parse_result(result, partial=partial) - if self.return_single: + if self.first_tool_only: single_result = ( parsed_result if parsed_result and parsed_result["type"] == self.key_name @@ -110,7 +118,7 @@ class PydanticToolsParser(JsonOutputToolsParser): def parse_result(self, result: List[Generation], *, partial: bool = False) -> Any: parsed_result = super().parse_result(result, partial=partial) name_dict = {tool.__name__: tool for tool in self.tools} - if self.return_single: + if self.first_tool_only: return ( name_dict[parsed_result["type"]](**parsed_result["args"]) if parsed_result From a0f17472ab12cbe039c0e55cf2d9dd430a49ddea Mon Sep 17 00:00:00 2001 From: Eugene Yurtsev Date: Thu, 22 Feb 2024 14:41:12 -0500 Subject: [PATCH 3/6] x --- libs/langchain/langchain/chains/structured_output/base.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/libs/langchain/langchain/chains/structured_output/base.py b/libs/langchain/langchain/chains/structured_output/base.py index a0109d06bb7b0..17de92a398d0e 100644 --- a/libs/langchain/langchain/chains/structured_output/base.py +++ b/libs/langchain/langchain/chains/structured_output/base.py @@ -99,14 +99,14 @@ class RecordDog(BaseModel): if not functions: raise ValueError("Need to pass in at least one function. Received zero.") openai_functions = [convert_to_openai_function(f) for f in functions] - llm_kwargs: Dict[str, Any] = {"functions": openai_functions, **llm_kwargs} + llm_kwargs_: Dict[str, Any] = {"functions": openai_functions, **llm_kwargs} if len(openai_functions) == 1 and enforce_single_function_usage: - llm_kwargs["function_call"] = {"name": openai_functions[0]["name"]} + llm_kwargs_["function_call"] = {"name": openai_functions[0]["name"]} output_parser = output_parser or get_openai_output_parser(functions) if prompt: - return prompt | llm.bind(**llm_kwargs) | output_parser + return prompt | llm.bind(**llm_kwargs_) | output_parser else: - return llm.bind(**llm_kwargs) | output_parser + return llm.bind(**llm_kwargs_) | output_parser def create_structured_output_runnable( From d8e89c808019f524595a610411fcef6e8f2aee9e Mon Sep 17 00:00:00 2001 From: Eugene Yurtsev Date: Thu, 22 Feb 2024 14:52:54 -0500 Subject: [PATCH 4/6] x --- .../langchain/output_parsers/openai_tools.py | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/libs/langchain/langchain/output_parsers/openai_tools.py b/libs/langchain/langchain/output_parsers/openai_tools.py index 9283499091a73..16a2a92c6f0f3 100644 --- a/libs/langchain/langchain/output_parsers/openai_tools.py +++ b/libs/langchain/langchain/output_parsers/openai_tools.py @@ -7,7 +7,7 @@ from langchain_core.output_parsers import BaseGenerationOutputParser from langchain_core.output_parsers.json import parse_partial_json from langchain_core.outputs import ChatGeneration, Generation -from langchain_core.pydantic_v1 import BaseModel +from langchain_core.pydantic_v1 import BaseModel, Field class JsonOutputToolsParser(BaseGenerationOutputParser[Any]): @@ -85,9 +85,16 @@ class JsonOutputKeyToolsParser(JsonOutputToolsParser): key_name: str """The type of tools to return.""" - def __init__(self, key_name: str, **kwargs: Any) -> None: """Allow init with positional args.""" + # Backwards compatibility for old argument name. + if "return_single" in kwargs: + if not kwargs.get("first_tool_only"): + kwargs["first_tool_only"] = kwargs.pop("return_single") + else: + raise ValueError( + "Cannot use both 'return_single' and 'first_tool_only' arguments." + ) super().__init__(key_name=key_name, **kwargs) def parse_result(self, result: List[Generation], *, partial: bool = False) -> Any: From 3fbaa3c6d1829c7288a20731c1f0223b84880361 Mon Sep 17 00:00:00 2001 From: Eugene Yurtsev Date: Thu, 22 Feb 2024 14:56:25 -0500 Subject: [PATCH 5/6] lint --- libs/langchain/langchain/output_parsers/openai_tools.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/libs/langchain/langchain/output_parsers/openai_tools.py b/libs/langchain/langchain/output_parsers/openai_tools.py index 16a2a92c6f0f3..7e426f369f746 100644 --- a/libs/langchain/langchain/output_parsers/openai_tools.py +++ b/libs/langchain/langchain/output_parsers/openai_tools.py @@ -7,7 +7,7 @@ from langchain_core.output_parsers import BaseGenerationOutputParser from langchain_core.output_parsers.json import parse_partial_json from langchain_core.outputs import ChatGeneration, Generation -from langchain_core.pydantic_v1 import BaseModel, Field +from langchain_core.pydantic_v1 import BaseModel class JsonOutputToolsParser(BaseGenerationOutputParser[Any]): From 2479f9949041d8b24cf278b25afea54081715fc7 Mon Sep 17 00:00:00 2001 From: Eugene Yurtsev Date: Thu, 22 Feb 2024 15:00:22 -0500 Subject: [PATCH 6/6] this code hates me --- libs/langchain/langchain/output_parsers/openai_tools.py | 1 + 1 file changed, 1 insertion(+) diff --git a/libs/langchain/langchain/output_parsers/openai_tools.py b/libs/langchain/langchain/output_parsers/openai_tools.py index 7e426f369f746..c03f4ad7afbc3 100644 --- a/libs/langchain/langchain/output_parsers/openai_tools.py +++ b/libs/langchain/langchain/output_parsers/openai_tools.py @@ -85,6 +85,7 @@ class JsonOutputKeyToolsParser(JsonOutputToolsParser): key_name: str """The type of tools to return.""" + def __init__(self, key_name: str, **kwargs: Any) -> None: """Allow init with positional args.""" # Backwards compatibility for old argument name.