Skip to content

Setting up Custom Authentication (Part ⅓)

This is part 1 of our authentication series:

  1. Basic Authentication (you are here) - Control who can access your bot
  2. Resource Authorization - Let users have private conversations
  3. Production Auth - Add real user accounts and validate using OAuth2

In this tutorial, we will build a chatbot that only lets specific users access it. We'll start with the LangGraph template and add token-based security step by step. By the end, you'll have a working chatbot that checks for valid tokens before allowing access.

Setting up our project

First, let's create a new chatbot using the LangGraph starter template:

pip install -U "langgraph-cli[inmem]"
langgraph new --template=new-langgraph-project-python custom-auth
cd custom-auth

The template gives us a placeholder LangGraph app. Let's try it out by installing the local dependencies and running the development server.

pip install -e .
langgraph dev
If everything works, the server should start and open the studio in your browser.

This in-memory server is designed for development and testing. For production use, please use LangGraph Cloud.

Now that we've seen the base LangGraph app, let's add authentication to it! In part 1, we will start with a hard-coded token for illustration purposes. We will get to a "production-ready" authentication scheme in part 3, after mastering the basics.

Adding Authentication

The Auth object lets you register an authentication function that the LangGraph platform will run on every request. This function receives each request and decides whether to accept or reject.

Create a new file src/security/auth.py. This is where our code will live to check if users are allowed to access our bot:

src/security/auth.py
from langgraph_sdk import Auth

# This is our toy user database. Do not do this in production
VALID_TOKENS = {
    "user1-token": {"id": "user1", "name": "Alice"},
    "user2-token": {"id": "user2", "name": "Bob"},
}

# The "Auth" object is a container that LangGraph will use to mark our authentication function
auth = Auth()


# The `authenticate` decorator tells LangGraph to call this function as middleware
# for every request. This will determine whether the request is allowed or not
@auth.authenticate
async def get_current_user(authorization: str | None) -> Auth.types.MinimalUserDict:
    """Check if the user's token is valid."""
    assert authorization
    scheme, token = authorization.split()
    assert scheme.lower() == "bearer"
    # Check if token is valid
    if token not in VALID_TOKENS:
        raise Auth.exceptions.HTTPException(status_code=401, detail="Invalid token")

    # Return user info if valid
    user_data = VALID_TOKENS[token]
    return {
        "identity": user_data["id"],
    }

Notice that our authentication handler does two important things:

  1. Checks if a valid token is provided in the request's Authorization header
  2. Returns the user's identity

Now tell LangGraph to use our authentication by adding the following to the langgraph.json configuration:

langgraph.json
{
  "dependencies": ["."],
  "graphs": {
    "agent": "./src/agent/graph.py:graph"
  },
  "env": ".env",
  "auth": {
    "path": "src/security/auth.py:auth"
  }
}

Testing Our "Secure" Bot

Let's start the server again to test everything out!

langgraph dev --no-browser
Custom auth in the studio

If you didn't add the --no-browser, the studio UI will open in the browser. You may wonder, how is the studio able to still connect to our server? By default, we also permit access from the LangGraph studio, even when using custom auth. This makes it easier to develop and test your bot in the studio. You can remove this alternative authentication option by setting disable_studio_auth: "true" in your auth configuration:

{
    "auth": {
        "path": "src/security/auth.py:auth",
        "disable_studio_auth": "true"
    }
}

Now let's try to chat with our bot. Run the following code in a file or notebook:

from langgraph_sdk import get_client

# Try without a token (should fail)
client = get_client(url="http://localhost:2024")
try:
    thread = await client.threads.create()
    print("❌ Should have failed without token!")
except Exception as e:
    print("✅ Correctly blocked access:", e)

# Try with a valid token
client = get_client(
    url="http://localhost:2024", headers={"Authorization": "Bearer user1-token"}
)

# Create a thread and chat
thread = await client.threads.create()
print(f"✅ Created thread as Alice: {thread['thread_id']}")

response = await client.runs.create(
    thread_id=thread["thread_id"],
    assistant_id="agent",
    input={"messages": [{"role": "user", "content": "Hello!"}]},
)
print("✅ Bot responded:")
print(response)

You should see that:

  1. Without a valid token, we can't access the bot
  2. With a valid token, we can create threads and chat

Congratulations! You've built a chatbot that only lets "authenticated" users access it. While this system doesn't (yet) implement a production-ready security scheme, we've learned the basic mechanics of how to control access to our bot. In the next tutorial, we'll learn how to give each user their own private conversations.

What's Next?

Now that you can control who accesses your bot, you might want to:

  1. Continue the tutorial by going to Making Conversations Private (Part ⅔) to learn about resource authorization.
  2. Read more about authentication concepts.
  3. Check out the API reference for more authentication details.

Comments