Day 2: Tracking Users with track_context()
📋 Overview
Welcome back to our learning path! In the previous section, you learned how to use payi_instrument() for basic auto-instrumentation of your GenAI calls. Now, we'll build on that foundation by adding user-specific context to your tracked calls.
Goal: Learn how to associate individual users with your GenAI calls using the track_context() function.
Why Track Users?
Tracking which users are making GenAI calls provides several important benefits:
- Cost allocation: Understand which users or customers are driving your AI costs
- Usage patterns: Analyze how different users interact with your AI features
- Per-user limits: Apply budget constraints or rate limits for specific users
- Troubleshooting: Quickly identify issues related to specific users
- Compliance: Meet regulatory requirements for user-level tracking and accountability
Recap: Auto-Instrumentation
Previously, we set up basic instrumentation with a single function call:
from payi.lib.instrument import payi_instrument
# Initialize Pay-i auto-instrumentation
payi_instrument()
# All supported GenAI calls are now automatically tracked!This approach is great for getting started quickly, but it doesn't tell you who is making each call. That's where custom instrumentation with the track_context() function comes in.
🔑 Core Concept: The track_context() Function
track_context() FunctionWhile payi_instrument() handles automatic instrumentation of all supported GenAI calls, the track_context() function allows you to add custom context to specific code blocks in your application, particularly when dealing with dynamic information that changes with each request.
Important distinctions:
- payi_instrument()is called once at the start of your application
- track_context()creates a context manager for specific code blocks where you want to add runtime information
- Both work together: payi_instrument()captures the calls, whiletrack_context()enriches them with context
Note: The
track_context()function creates a scope that applies to all GenAI calls made within its block. This behavior gives you precise control over context propagation. Later in the learning path, we'll cover the@trackdecorator for function-level static annotations.
The track_context() function works alongside auto-instrumentation, adding valuable runtime context like user IDs to your tracked events.
Tracking Users
When your user ID is available at runtime (during function execution):
from payi.lib.instrument import track_context
def my_function(prompt):
    # Your code here to get the current user ID at runtime
    user_id = get_current_user_id()
    
    # Use track_context for runtime values
    with track_context(user_id=user_id):
        response = client.chat.completions.create(
            model="gpt-3.5-turbo",
            messages=[{"role": "user", "content": prompt}]
        )
    return response.choices[0].message.contentThis approach is ideal when the user ID is determined at runtime and changes with each request.
Note: You can also use custom headers with the
extra_headersparameter, buttrack_context()provides a cleaner, more readable solution for Python applications.
Best Practices for User IDs
Your user_id should be a string that uniquely identifies the end-user responsible for each call:
- Always usetrack_context()for user IDs: Since user IDs typically change with each request
- Use consistent identifiers that match your application's user system
- Consider privacy implications when selecting identifiers
- Avoid using personally identifiable information (PII) when possible
- Use anonymous but consistent IDs for unauthenticated users
For more details on user-level attribution in Pay-i, see the User-level Attribution.
✅ Verification: Checking User Tracking
To verify that your user tracking is working correctly:
- Run your application and make a few AI calls through your decorated functions using different user IDs
- Log in to developer.pay-i.com
- Navigate to your application dashboard
- Click on Cost Drivers in the left sidebar
- Select the Users tab at the top of the page
- You'll see a list of all user IDs that have been tracked, along with their request counts and spend
You should see different user IDs listed with their corresponding request counts and costs, allowing you to analyze usage patterns on a per-user basis.
➡️ Next Steps
Congratulations! You've learned how to add user context to your GenAI calls using track_context(). In the next section, we'll build on this knowledge by introducing the @track decorator for assigning specific Use Cases with @track(use_case_name="...").
The @track decorator is well-suited for use case names since they are typically static values known when your functions are defined, unlike user IDs which change with each request.
💡 Additional Resources
- Custom Instrumentation guide - Comprehensive coverage of instrumentation options
- track_context() reference - Details on the track_context()function and its parameters
- User-level Attribution - Advanced strategies for user attribution
Updated 2 days ago
