from__future__importannotationsfromtypingimportAnnotated,Any,TypeVar,Unionfromlangchain_core.exceptionsimportOutputParserExceptionfromlangchain_core.language_modelsimportBaseLanguageModelfromlangchain_core.output_parsersimportBaseOutputParser,StrOutputParserfromlangchain_core.prompt_valuesimportPromptValuefromlangchain_core.promptsimportBasePromptTemplate,PromptTemplatefromlangchain_core.runnablesimportRunnableSerializablefrompydanticimportSkipValidationfromtyping_extensionsimportTypedDictNAIVE_COMPLETION_RETRY="""Prompt:{prompt}Completion:{completion}Above, the Completion did not satisfy the constraints given in the Prompt.Please try again:"""NAIVE_COMPLETION_RETRY_WITH_ERROR="""Prompt:{prompt}Completion:{completion}Above, the Completion did not satisfy the constraints given in the Prompt.Details: {error}Please try again:"""NAIVE_RETRY_PROMPT=PromptTemplate.from_template(NAIVE_COMPLETION_RETRY)NAIVE_RETRY_WITH_ERROR_PROMPT=PromptTemplate.from_template(NAIVE_COMPLETION_RETRY_WITH_ERROR)T=TypeVar("T")
[docs]classRetryOutputParser(BaseOutputParser[T]):"""Wrap a parser and try to fix parsing errors. Does this by passing the original prompt and the completion to another LLM, and telling it the completion did not satisfy criteria in the prompt. """parser:Annotated[BaseOutputParser[T],SkipValidation()]"""The parser to use to parse the output."""# Should be an LLMChain but we want to avoid top-level imports from langchain.chainsretry_chain:Annotated[Union[RunnableSerializable[RetryOutputParserRetryChainInput,str],Any],SkipValidation(),]"""The RunnableSerializable to use to retry the completion (Legacy: LLMChain)."""max_retries:int=1"""The maximum number of times to retry the parse."""legacy:bool=True"""Whether to use the run or arun method of the retry_chain."""
[docs]@classmethoddeffrom_llm(cls,llm:BaseLanguageModel,parser:BaseOutputParser[T],prompt:BasePromptTemplate=NAIVE_RETRY_PROMPT,max_retries:int=1,)->RetryOutputParser[T]:"""Create an RetryOutputParser from a language model and a parser. Args: llm: llm to use for fixing parser: parser to use for parsing prompt: prompt to use for fixing max_retries: Maximum number of retries to parse. Returns: RetryOutputParser """chain=prompt|llm|StrOutputParser()returncls(parser=parser,retry_chain=chain,max_retries=max_retries)
[docs]defparse_with_prompt(self,completion:str,prompt_value:PromptValue)->T:"""Parse the output of an LLM call using a wrapped parser. Args: completion: The chain completion to parse. prompt_value: The prompt to use to parse the completion. Returns: The parsed completion. """retries=0whileretries<=self.max_retries:try:returnself.parser.parse(completion)exceptOutputParserExceptionase:ifretries==self.max_retries:raiseeelse:retries+=1ifself.legacyandhasattr(self.retry_chain,"run"):completion=self.retry_chain.run(prompt=prompt_value.to_string(),completion=completion,)else:completion=self.retry_chain.invoke(dict(prompt=prompt_value.to_string(),completion=completion,))raiseOutputParserException("Failed to parse")
[docs]asyncdefaparse_with_prompt(self,completion:str,prompt_value:PromptValue)->T:"""Parse the output of an LLM call using a wrapped parser. Args: completion: The chain completion to parse. prompt_value: The prompt to use to parse the completion. Returns: The parsed completion. """retries=0whileretries<=self.max_retries:try:returnawaitself.parser.aparse(completion)exceptOutputParserExceptionase:ifretries==self.max_retries:raiseeelse:retries+=1ifself.legacyandhasattr(self.retry_chain,"arun"):completion=awaitself.retry_chain.arun(prompt=prompt_value.to_string(),completion=completion,error=repr(e),)else:completion=awaitself.retry_chain.ainvoke(dict(prompt=prompt_value.to_string(),completion=completion,))raiseOutputParserException("Failed to parse")
[docs]defparse(self,completion:str)->T:raiseNotImplementedError("This OutputParser can only be called by the `parse_with_prompt` method.")
[docs]classRetryWithErrorOutputParser(BaseOutputParser[T]):"""Wrap a parser and try to fix parsing errors. Does this by passing the original prompt, the completion, AND the error that was raised to another language model and telling it that the completion did not work, and raised the given error. Differs from RetryOutputParser in that this implementation provides the error that was raised back to the LLM, which in theory should give it more information on how to fix it. """parser:Annotated[BaseOutputParser[T],SkipValidation()]"""The parser to use to parse the output."""# Should be an LLMChain but we want to avoid top-level imports from langchain.chainsretry_chain:Annotated[Union[RunnableSerializable[RetryWithErrorOutputParserRetryChainInput,str],Any],SkipValidation(),]"""The RunnableSerializable to use to retry the completion (Legacy: LLMChain)."""max_retries:int=1"""The maximum number of times to retry the parse."""legacy:bool=True"""Whether to use the run or arun method of the retry_chain."""
[docs]@classmethoddeffrom_llm(cls,llm:BaseLanguageModel,parser:BaseOutputParser[T],prompt:BasePromptTemplate=NAIVE_RETRY_WITH_ERROR_PROMPT,max_retries:int=1,)->RetryWithErrorOutputParser[T]:"""Create a RetryWithErrorOutputParser from an LLM. Args: llm: The LLM to use to retry the completion. parser: The parser to use to parse the output. prompt: The prompt to use to retry the completion. max_retries: The maximum number of times to retry the completion. Returns: A RetryWithErrorOutputParser. """chain=prompt|llm|StrOutputParser()returncls(parser=parser,retry_chain=chain,max_retries=max_retries)
[docs]defparse_with_prompt(self,completion:str,prompt_value:PromptValue)->T:retries=0whileretries<=self.max_retries:try:returnself.parser.parse(completion)exceptOutputParserExceptionase:ifretries==self.max_retries:raiseeelse:retries+=1ifself.legacyandhasattr(self.retry_chain,"run"):completion=self.retry_chain.run(prompt=prompt_value.to_string(),completion=completion,error=repr(e),)else:completion=self.retry_chain.invoke(dict(completion=completion,prompt=prompt_value.to_string(),error=repr(e),))raiseOutputParserException("Failed to parse")
[docs]asyncdefaparse_with_prompt(self,completion:str,prompt_value:PromptValue)->T:retries=0whileretries<=self.max_retries:try:returnawaitself.parser.aparse(completion)exceptOutputParserExceptionase:ifretries==self.max_retries:raiseeelse:retries+=1ifself.legacyandhasattr(self.retry_chain,"arun"):completion=awaitself.retry_chain.arun(prompt=prompt_value.to_string(),completion=completion,error=repr(e),)else:completion=awaitself.retry_chain.ainvoke(dict(prompt=prompt_value.to_string(),completion=completion,error=repr(e),))raiseOutputParserException("Failed to parse")
[docs]defparse(self,completion:str)->T:raiseNotImplementedError("This OutputParser can only be called by the `parse_with_prompt` method.")