mirror of
https://github.com/langchain-ai/langgraph-example-pyproject.git
synced 2026-07-01 17:04:09 -04:00
reformat
This commit is contained in:
@@ -1,102 +0,0 @@
|
||||
from typing import TypedDict, Annotated, Sequence, Literal
|
||||
|
||||
from functools import lru_cache
|
||||
from langchain_core.messages import BaseMessage
|
||||
from langchain_anthropic import ChatAnthropic
|
||||
from langchain_openai import ChatOpenAI
|
||||
from langchain_community.tools.tavily_search import TavilySearchResults
|
||||
from langgraph.prebuilt import ToolNode
|
||||
from langgraph.graph import StateGraph, END, add_messages
|
||||
|
||||
tools = [TavilySearchResults(max_results=1)]
|
||||
|
||||
@lru_cache(maxsize=4)
|
||||
def _get_model(model_name: str):
|
||||
if model_name == "openai":
|
||||
model = ChatOpenAI(temperature=0, model_name="gpt-4o")
|
||||
elif model_name == "anthropic":
|
||||
model = ChatAnthropic(temperature=0, model_name="claude-3-sonnet-20240229")
|
||||
else:
|
||||
raise ValueError(f"Unsupported model type: {model_name}")
|
||||
|
||||
model = model.bind_tools(tools)
|
||||
return model
|
||||
|
||||
|
||||
class AgentState(TypedDict):
|
||||
messages: Annotated[Sequence[BaseMessage], add_messages]
|
||||
|
||||
|
||||
# Define the function that determines whether to continue or not
|
||||
def should_continue(state):
|
||||
messages = state["messages"]
|
||||
last_message = messages[-1]
|
||||
# If there are no tool calls, then we finish
|
||||
if not last_message.tool_calls:
|
||||
return "end"
|
||||
# Otherwise if there is, we continue
|
||||
else:
|
||||
return "continue"
|
||||
|
||||
|
||||
system_prompt = """Be a helpful assistant"""
|
||||
|
||||
# Define the function that calls the model
|
||||
def call_model(state, config):
|
||||
messages = state["messages"]
|
||||
messages = [{"role": "system", "content": system_prompt}] + messages
|
||||
model_name = config.get('configurable', {}).get("model_name", "anthropic")
|
||||
model = _get_model(model_name)
|
||||
response = model.invoke(messages)
|
||||
# We return a list, because this will get added to the existing list
|
||||
return {"messages": [response]}
|
||||
|
||||
|
||||
# Define the function to execute tools
|
||||
tool_node = ToolNode(tools)
|
||||
|
||||
# Define the config
|
||||
class GraphConfig(TypedDict):
|
||||
model_name: Literal["anthropic", "openai"]
|
||||
|
||||
|
||||
# Define a new graph
|
||||
workflow = StateGraph(AgentState, config_schema=GraphConfig)
|
||||
|
||||
# Define the two nodes we will cycle between
|
||||
workflow.add_node("agent", call_model)
|
||||
workflow.add_node("action", tool_node)
|
||||
|
||||
# Set the entrypoint as `agent`
|
||||
# This means that this node is the first one called
|
||||
workflow.set_entry_point("agent")
|
||||
|
||||
# We now add a conditional edge
|
||||
workflow.add_conditional_edges(
|
||||
# First, we define the start node. We use `agent`.
|
||||
# This means these are the edges taken after the `agent` node is called.
|
||||
"agent",
|
||||
# Next, we pass in the function that will determine which node is called next.
|
||||
should_continue,
|
||||
# Finally we pass in a mapping.
|
||||
# The keys are strings, and the values are other nodes.
|
||||
# END is a special node marking that the graph should finish.
|
||||
# What will happen is we will call `should_continue`, and then the output of that
|
||||
# will be matched against the keys in this mapping.
|
||||
# Based on which one it matches, that node will then be called.
|
||||
{
|
||||
# If `tools`, then we call the tool node.
|
||||
"continue": "action",
|
||||
# Otherwise we finish.
|
||||
"end": END,
|
||||
},
|
||||
)
|
||||
|
||||
# We now add a normal edge from `tools` to `agent`.
|
||||
# This means that after `tools` is called, `agent` node is called next.
|
||||
workflow.add_edge("action", "agent")
|
||||
|
||||
# Finally, we compile it!
|
||||
# This compiles it into a LangChain Runnable,
|
||||
# meaning you can use it as you would any other runnable
|
||||
graph = workflow.compile()
|
||||
+1
-1
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"dependencies": ["."],
|
||||
"graphs": {
|
||||
"agent": "./langchain-example-pyproject/agent.py:graph"
|
||||
"agent": "./my_agent/agent.py:graph"
|
||||
},
|
||||
"env": ".env"
|
||||
}
|
||||
|
||||
@@ -0,0 +1,52 @@
|
||||
from typing import TypedDict, Literal
|
||||
|
||||
from langgraph.graph import StateGraph, END
|
||||
from my_agent.utils.nodes import call_model, should_continue, tool_node
|
||||
from my_agent.utils.state import AgentState
|
||||
|
||||
|
||||
# Define the config
|
||||
class GraphConfig(TypedDict):
|
||||
model_name: Literal["anthropic", "openai"]
|
||||
|
||||
|
||||
# Define a new graph
|
||||
workflow = StateGraph(AgentState, config_schema=GraphConfig)
|
||||
|
||||
# Define the two nodes we will cycle between
|
||||
workflow.add_node("agent", call_model)
|
||||
workflow.add_node("action", tool_node)
|
||||
|
||||
# Set the entrypoint as `agent`
|
||||
# This means that this node is the first one called
|
||||
workflow.set_entry_point("agent")
|
||||
|
||||
# We now add a conditional edge
|
||||
workflow.add_conditional_edges(
|
||||
# First, we define the start node. We use `agent`.
|
||||
# This means these are the edges taken after the `agent` node is called.
|
||||
"agent",
|
||||
# Next, we pass in the function that will determine which node is called next.
|
||||
should_continue,
|
||||
# Finally we pass in a mapping.
|
||||
# The keys are strings, and the values are other nodes.
|
||||
# END is a special node marking that the graph should finish.
|
||||
# What will happen is we will call `should_continue`, and then the output of that
|
||||
# will be matched against the keys in this mapping.
|
||||
# Based on which one it matches, that node will then be called.
|
||||
{
|
||||
# If `tools`, then we call the tool node.
|
||||
"continue": "action",
|
||||
# Otherwise we finish.
|
||||
"end": END,
|
||||
},
|
||||
)
|
||||
|
||||
# We now add a normal edge from `tools` to `agent`.
|
||||
# This means that after `tools` is called, `agent` node is called next.
|
||||
workflow.add_edge("action", "agent")
|
||||
|
||||
# Finally, we compile it!
|
||||
# This compiles it into a LangChain Runnable,
|
||||
# meaning you can use it as you would any other runnable
|
||||
graph = workflow.compile()
|
||||
@@ -0,0 +1,45 @@
|
||||
from functools import lru_cache
|
||||
from langchain_anthropic import ChatAnthropic
|
||||
from langchain_openai import ChatOpenAI
|
||||
from my_agent.utils.tools import tools
|
||||
from langgraph.prebuilt import ToolNode
|
||||
|
||||
|
||||
@lru_cache(maxsize=4)
|
||||
def _get_model(model_name: str):
|
||||
if model_name == "openai":
|
||||
model = ChatOpenAI(temperature=0, model_name="gpt-4o")
|
||||
elif model_name == "anthropic":
|
||||
model = ChatAnthropic(temperature=0, model_name="claude-3-sonnet-20240229")
|
||||
else:
|
||||
raise ValueError(f"Unsupported model type: {model_name}")
|
||||
|
||||
model = model.bind_tools(tools)
|
||||
return model
|
||||
|
||||
# Define the function that determines whether to continue or not
|
||||
def should_continue(state):
|
||||
messages = state["messages"]
|
||||
last_message = messages[-1]
|
||||
# If there are no tool calls, then we finish
|
||||
if not last_message.tool_calls:
|
||||
return "end"
|
||||
# Otherwise if there is, we continue
|
||||
else:
|
||||
return "continue"
|
||||
|
||||
|
||||
system_prompt = """Be a helpful assistant"""
|
||||
|
||||
# Define the function that calls the model
|
||||
def call_model(state, config):
|
||||
messages = state["messages"]
|
||||
messages = [{"role": "system", "content": system_prompt}] + messages
|
||||
model_name = config.get('configurable', {}).get("model_name", "anthropic")
|
||||
model = _get_model(model_name)
|
||||
response = model.invoke(messages)
|
||||
# We return a list, because this will get added to the existing list
|
||||
return {"messages": [response]}
|
||||
|
||||
# Define the function to execute tools
|
||||
tool_node = ToolNode(tools)
|
||||
@@ -0,0 +1,6 @@
|
||||
from langgraph.graph import add_messages
|
||||
from langchain_core.messages import BaseMessage
|
||||
from typing import TypedDict, Annotated, Sequence
|
||||
|
||||
class AgentState(TypedDict):
|
||||
messages: Annotated[Sequence[BaseMessage], add_messages]
|
||||
@@ -0,0 +1,3 @@
|
||||
from langchain_community.tools.tavily_search import TavilySearchResults
|
||||
|
||||
tools = [TavilySearchResults(max_results=1)]
|
||||
+3
-5
@@ -1,19 +1,17 @@
|
||||
[tool.poetry]
|
||||
name = "langchain-example-pyproject"
|
||||
name = "my_agent"
|
||||
version = "0.1.0"
|
||||
description = "Example LangGraph project for deployment to LangGraph Cloud"
|
||||
authors = [
|
||||
"langchain-ai"
|
||||
]
|
||||
packages = [
|
||||
{ include = "langchain-example-pyproject" },
|
||||
{ include = "my_agent" },
|
||||
]
|
||||
|
||||
|
||||
|
||||
[tool.poetry.dependencies]
|
||||
python = ">=3.9.0,<3.13"
|
||||
langgraph = "^0.1.7"
|
||||
langgraph = "^0.2.0"
|
||||
langchain_anthropic = "^0.1.0"
|
||||
langchain_core = "^0.2.0"
|
||||
langchain_openai = "^0.1.0"
|
||||
|
||||
Reference in New Issue
Block a user