Passing an Authorization Token to the tool & avoiding the LLM. #534
Replies: 7 comments 1 reply
-
The general strategy would be to sub-class form StructuredTools, implement some configurable tools for tokens, and use configurable runnables. However, because agent executor does not propagate configuration information currently, this strategy will not work. (Issue needs to be fixed in langchain) The way to do this right now is to instantiate an appropriate agent at run time with auth bound to the tools. Instantiation of most of these objects at run time is very fast (should be far less than ~1ms), so performance is not a concern here. Here are two reference examples that show how an agent can be instantiated at run time:
If the token is supplied directly by the user, the examples are sufficient as they are. If the token comes from database look up that relies on the user identity, take a look at the auth examples here: https://python.langchain.com/docs/langserve#examples If you're comfortable with FastAPI / python, you can use Here's documentation for configurable runnables: https://python.langchain.com/docs/expression_language/how_to/configue Configurable runnables examples: https://github.com/langchain-ai/langserve/blob/main/examples/configurable_chain/server.py |
Beta Was this translation helpful? Give feedback.
-
Lets say the auth is passed to the endpoint like this: class Input(BaseModel):
question: str
token: str I want to pass the question to the LLM but once the corresponding function is mapped, I want to intercept the function call and pass the token before it is executed. It is unclear how I can pass the token directly to the mapped function from the examples shared. I attempted to write a custom parser and pass the token to it but I am not able to extract the token from the input request as it gets overwritten by the previous step ( I am not an expert with LCEL do point out if there is a better way to do this, I tried using RunnablePassthrough with RunnableParallel still didn't get the result I wanted) agent = (
{
"input": lambda x: x["question"],
"agent_scratchpad": lambda x: format_to_openai_functions(x["intermediate_steps"]),
}
| prompt
| llm_with_tools
| CustomOpenAIFunctionsAgentOutputParser(config_param={"token" :lambda x: x["token"]}) # Pass the "token" key
)
) Would appreciate a nudge to the right direction @eyurtsev |
Beta Was this translation helpful? Give feedback.
-
Any updates on this? Have the same issue, @eyurtsev would appreciate any help! |
Beta Was this translation helpful? Give feedback.
-
I'm currently struggling with this too, I'll provide my solution if I find one. I also think that this is a very common pattern, when you, in your tools, want to access some data coming from the request without passing them through the chain. So, I believe it should have an elegant solution within the framework, along with a clear, end-to-end example explaining the principles. |
Beta Was this translation helpful? Give feedback.
-
I am also facing a similar usecase and would appreciate help. the bot suggested multiple approaches but did not work langchain-ai/langchain#22529 |
Beta Was this translation helpful? Give feedback.
-
I was able to solve this problem in the following way for a Service API that requires a bearer token Create the required tool by calling the wrapper func api_service = wrap_api_service({'secret' : '358123'})
tools = [api_service]
#Pass the `tools` to an agent executor or an LLM directly Definition of the wrapper func and the tool from typing import Callable, Any, Dict
from functools import wraps
def wrap_api_service(wrapper_config : Dict[str, Any]):
@tool
def api_service(request_body) -> str:
#Wrapper config will contain the secret
response = http_utils.post(url, request_body, wrapper.config['secret'])
return str(response)
return api_service
FYI - @fatimatayeb , @casperakos @bukacdan @eyurtsev Credits - |
Beta Was this translation helpful? Give feedback.
-
If anyone is looking for a way to dynamically inject a JWT token into the tools during runtime, Python's Example:Create a Context Manager for Token Managementfrom contextvars import ContextVar
from typing import Optional
jwt_token: ContextVar[Optional[str]] = ContextVar("jwt_token", default=None)
class JWTTokenManager:
def __init__(self, token: str):
self.token = token
self.token_context = None
def __enter__(self):
# Set the token in the context variable
self.token_context = jwt_token.set(self.token)
def __exit__(self, exc_type, exc_val, exc_tb):
# Reset the context variable after use
jwt_token.reset(self.token_context) Create a Decorator to Inject the Token Into LangChain Toolsfrom functools import wraps
from context import jwt_token
def inject_jwt_token(func):
@wraps(func)
async def wrapper(*args, **kwargs):
# Retrieve the token from the context variable
token = jwt_token.get()
if token:
# Inject the token into the headers
kwargs["headers"] = kwargs.get("headers", {})
kwargs["headers"]["Authorization"] = f"Bearer {token}"
return await func(*args, **kwargs)
return wrapper Use the Decorator in a LangChain Toolsfrom langchain_core.tools import tool
import httpx
from decorators import inject_jwt_token
@tool()
@inject_jwt_token
async def make_request(endpoint: str, headers: dict = None) -> dict:
async with httpx.AsyncClient() as client:
response = await client.get(endpoint, headers=headers)
response.raise_for_status()
return response.json() UsageThe from token_manager import JWTTokenManager
from tools import make_request
async def main():
# Set the token using the context manager
with JWTTokenManager("your_jwt_token_here"):
# Use the tool to make an API request
result = await make_request("https://api.example.com/data")
print(result) |
Beta Was this translation helpful? Give feedback.
-
I am using Langserve Conversation Agent with GPT models to map user questions to private API's and do function calling.
I want to be able to pass the authorization token directly to the tool instead of it going through the LLM.
Could you point me in to the right direction to do this and what is the best practice?
Beta Was this translation helpful? Give feedback.
All reactions