Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

core: fix for BaseTool when args_schema has pydantic validators that change the input keys #30004

Open
wants to merge 3 commits into
base: master
Choose a base branch
from

Conversation

VMinB12
Copy link
Contributor

@VMinB12 VMinB12 commented Feb 26, 2025

Description:

We encountered a bug where we used a pydantic validator to patch broken llm inputs to our tools. The validator would fix the input dict, but the BaseTool would return an empty dict to the tool because our validator changed the dictionary keys. This PR changes one line of code to fix the issue and adds a test to improve the test coverage.

  • Issue:

Concretely, the following code would fail before the PR:

from pydantic import BaseModel, model_validator
from langchain_core.tools import tool


class B(BaseModel):
    b: int


class A(BaseModel):
    a: B

    @model_validator(mode="before")
    def a_wrapper(cls, data):
        if "a" not in data:
            return {"a": data}
        return data


if __name__ == "__main__":
    # Valid input passes
    print(A.model_validate({"a": {"b": 1}}))
    # Invalid input also passes because validator patches it
    print(A.model_validate({"b": 1}))


@tool(args_schema=A, infer_schema=False)
def answer(
    **kwargs,
):
    """Use this function to provide your final answer to the request."""
    return kwargs


if __name__ == "__main__":
    # Valid input passes
    assert answer.invoke({"a": {"b": 1}}) == {"a": B(b=1)}
    # Invalid input should pass, but throws assertion error
    res = answer.invoke({"b": 1})
    assert res == {"a": B(b=1)}, f"Failure: {res} of type {type(res)}"
    # AssertionError: Failure: {} of type <class 'dict'>

Add tests and docs:
I added a test for the above use case.

Comments:

Please note that I had to remove *args: Any from one of the tests. The previous code would filter out the args keyword. I don't see any realistic use case where one would have *args in their tool as this gives no meaningful instruction to the LLM on how to use the *args. Please let me know if you see any issues here. I don't see a clean and robust method that would have the new test passing while still filtering out the *args argument. Let me know if you see a better solution.

@dosubot dosubot bot added the size:M This PR changes 30-99 lines, ignoring generated files. label Feb 26, 2025
Copy link

vercel bot commented Feb 26, 2025

The latest updates on your projects. Learn more about Vercel for Git ↗︎

1 Skipped Deployment
Name Status Preview Comments Updated (UTC)
langchain ⬜️ Ignored (Inspect) Visit Preview Mar 28, 2025 7:28pm

@dosubot dosubot bot added langchain Related to the langchain package 🤖:bug Related to a bug, vulnerability, unexpected error with an existing feature labels Feb 26, 2025
@hugodendievelb12
Copy link

@ccurme

@eyurtsev eyurtsev self-assigned this Mar 28, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
🤖:bug Related to a bug, vulnerability, unexpected error with an existing feature langchain Related to the langchain package size:M This PR changes 30-99 lines, ignoring generated files.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants