"""Base callback handler that can be used to handle callbacks from langchain.""" from abc import ABC, abstractmethod from typing import Any, Dict, List, Union from langchain.schema import AgentAction, AgentFinish, LLMResult class BaseCallbackHandler(ABC): """Base callback handler that can be used to handle callbacks from langchain.""" @property def always_verbose(self) -> bool: """Whether to call verbose callbacks even if verbose is False.""" return False @property def ignore_llm(self) -> bool: """Whether to ignore LLM callbacks.""" return False @property def ignore_chain(self) -> bool: """Whether to ignore chain callbacks.""" return False @property def ignore_agent(self) -> bool: """Whether to ignore agent callbacks.""" return False @abstractmethod def on_llm_start( self, serialized: Dict[str, Any], prompts: List[str], **kwargs: Any ) -> None: """Run when LLM starts running.""" @abstractmethod def on_llm_end(self, response: LLMResult, **kwargs: Any) -> None: """Run when LLM ends running.""" @abstractmethod def on_llm_error( self, error: Union[Exception, KeyboardInterrupt], **kwargs: Any ) -> None: """Run when LLM errors.""" @abstractmethod def on_chain_start( self, serialized: Dict[str, Any], inputs: Dict[str, Any], **kwargs: Any ) -> None: """Run when chain starts running.""" @abstractmethod def on_chain_end(self, outputs: Dict[str, Any], **kwargs: Any) -> None: """Run when chain ends running.""" @abstractmethod def on_chain_error( self, error: Union[Exception, KeyboardInterrupt], **kwargs: Any ) -> None: """Run when chain errors.""" @abstractmethod def on_tool_start( self, serialized: Dict[str, Any], action: AgentAction, **kwargs: Any ) -> None: """Run when tool starts running.""" @abstractmethod def on_tool_end(self, output: str, **kwargs: Any) -> None: """Run when tool ends running.""" @abstractmethod def on_tool_error( self, error: Union[Exception, KeyboardInterrupt], **kwargs: Any ) -> None: """Run when tool errors.""" @abstractmethod def on_text(self, text: str, **kwargs: Any) -> None: """Run on arbitrary text.""" @abstractmethod def on_agent_finish(self, finish: AgentFinish, **kwargs: Any) -> None: """Run on agent end.""" class BaseCallbackManager(BaseCallbackHandler, ABC): """Base callback manager that can be used to handle callbacks from LangChain.""" @abstractmethod def add_handler(self, callback: BaseCallbackHandler) -> None: """Add a handler to the callback manager.""" @abstractmethod def remove_handler(self, handler: BaseCallbackHandler) -> None: """Remove a handler from the callback manager.""" def set_handler(self, handler: BaseCallbackHandler) -> None: """Set handler as the only handler on the callback manager.""" self.set_handlers([handler]) @abstractmethod def set_handlers(self, handlers: List[BaseCallbackHandler]) -> None: """Set handlers as the only handlers on the callback manager.""" class CallbackManager(BaseCallbackManager): """Callback manager that can be used to handle callbacks from langchain.""" def __init__(self, handlers: List[BaseCallbackHandler]) -> None: """Initialize callback manager.""" self.handlers: List[BaseCallbackHandler] = handlers def on_llm_start( self, serialized: Dict[str, Any], prompts: List[str], verbose: bool = False, **kwargs: Any ) -> None: """Run when LLM starts running.""" for handler in self.handlers: if not handler.ignore_llm: if verbose or handler.always_verbose: handler.on_llm_start(serialized, prompts, **kwargs) def on_llm_end( self, response: LLMResult, verbose: bool = False, **kwargs: Any ) -> None: """Run when LLM ends running.""" for handler in self.handlers: if not handler.ignore_llm: if verbose or handler.always_verbose: handler.on_llm_end(response) def on_llm_error( self, error: Union[Exception, KeyboardInterrupt], verbose: bool = False, **kwargs: Any ) -> None: """Run when LLM errors.""" for handler in self.handlers: if not handler.ignore_llm: if verbose or handler.always_verbose: handler.on_llm_error(error) def on_chain_start( self, serialized: Dict[str, Any], inputs: Dict[str, Any], verbose: bool = False, **kwargs: Any ) -> None: """Run when chain starts running.""" for handler in self.handlers: if not handler.ignore_chain: if verbose or handler.always_verbose: handler.on_chain_start(serialized, inputs, **kwargs) def on_chain_end( self, outputs: Dict[str, Any], verbose: bool = False, **kwargs: Any ) -> None: """Run when chain ends running.""" for handler in self.handlers: if not handler.ignore_chain: if verbose or handler.always_verbose: handler.on_chain_end(outputs) def on_chain_error( self, error: Union[Exception, KeyboardInterrupt], verbose: bool = False, **kwargs: Any ) -> None: """Run when chain errors.""" for handler in self.handlers: if not handler.ignore_chain: if verbose or handler.always_verbose: handler.on_chain_error(error) def on_tool_start( self, serialized: Dict[str, Any], action: AgentAction, verbose: bool = False, **kwargs: Any ) -> None: """Run when tool starts running.""" for handler in self.handlers: if not handler.ignore_agent: if verbose or handler.always_verbose: handler.on_tool_start(serialized, action, **kwargs) def on_tool_end(self, output: str, verbose: bool = False, **kwargs: Any) -> None: """Run when tool ends running.""" for handler in self.handlers: if not handler.ignore_agent: if verbose or handler.always_verbose: handler.on_tool_end(output, **kwargs) def on_tool_error( self, error: Union[Exception, KeyboardInterrupt], verbose: bool = False, **kwargs: Any ) -> None: """Run when tool errors.""" for handler in self.handlers: if not handler.ignore_agent: if verbose or handler.always_verbose: handler.on_tool_error(error) def on_text(self, text: str, verbose: bool = False, **kwargs: Any) -> None: """Run on additional input from chains and agents.""" for handler in self.handlers: if verbose or handler.always_verbose: handler.on_text(text, **kwargs) def on_agent_finish( self, finish: AgentFinish, verbose: bool = False, **kwargs: Any ) -> None: """Run on agent end.""" for handler in self.handlers: if not handler.ignore_agent: if verbose or handler.always_verbose: handler.on_agent_finish(finish, **kwargs) def add_handler(self, handler: BaseCallbackHandler) -> None: """Add a handler to the callback manager.""" self.handlers.append(handler) def remove_handler(self, handler: BaseCallbackHandler) -> None: """Remove a handler from the callback manager.""" self.handlers.remove(handler) def set_handlers(self, handlers: List[BaseCallbackHandler]) -> None: """Set handlers as the only handlers on the callback manager.""" self.handlers = handlers