A Python SDK for monitoring and analyzing AI API interactions with LLM services
Project description
Coolhand Python
Monitor and log LLM API calls from OpenAI, Anthropic, Google Gemini, GitHub Copilot, and more to the Coolhand analytics platform.
Python 3.8 and 3.9 support deprecated: As of v0.4.0, coolhand requires Python 3.10 or later. Python 3.8 reached end-of-life in October 2024 and 3.9 in October 2025. If you are on an older Python version, pin to
coolhand<0.4.0.
Installation
pip install coolhand
Getting Started
- Get API Key: Visit coolhandlabs.com to create a free account
- Install:
pip install coolhand - Configure: Set
COOLHAND_API_KEYandimport coolhandin your app - Deploy: Your AI calls are now automatically monitored!
Quick Start
Automatic Global Monitoring
Set it and forget it! Monitor ALL AI API calls across your entire application with minimal configuration.
import coolhand # Auto-initializes and starts monitoring
# That's it! ALL AI API calls are now automatically monitored:
# ✅ OpenAI SDK calls
# ✅ Anthropic API calls
# ✅ Google Gemini API calls
# ✅ GitHub Copilot SDK calls
# ✅ ANY library making AI API calls via httpx
# Your existing code works unchanged:
from openai import OpenAI
client = OpenAI()
response = client.chat.completions.create(
model="gpt-4",
messages=[{"role": "user", "content": "Hello!"}]
)
# The request and response have been automatically logged to Coolhand!
Why Automatic Monitoring:
- Zero refactoring - No code changes to existing services
- Complete coverage - Monitors all AI libraries using httpx automatically
- Security built-in - Automatic credential sanitization
- Performance optimized - Negligible overhead via async logging
- Future-proof - Automatically captures new AI calls added by your team
Configuration
Environment Variables
| Variable | Required | Default | Description |
|---|---|---|---|
COOLHAND_API_KEY |
Yes | - | Your Coolhand API key for authentication |
COOLHAND_BASE_URL |
No | https://coolhandlabs.com |
Override the API host (self-hosted deployments) |
COOLHAND_SILENT |
No | true |
Set to false for verbose logging output |
Manual Configuration
from coolhand import Coolhand
coolhand_client = Coolhand(
api_key='your-api-key',
silent=False, # Enable verbose logging
)
Excluding API Patterns
Some endpoints — batch jobs, health checks, internal metrics — generate high-volume traffic that isn't useful to log. Use exclude_api_patterns to skip them:
from coolhand import Coolhand
coolhand_client = Coolhand(
api_key='your-api-key',
exclude_api_patterns=[
'/health',
'/metrics',
'/batchPredictionJobs/',
],
)
Any request whose URL contains one of the listed substrings is passed through without logging. The default list (DEFAULT_EXCLUDE_API_PATTERNS) excludes /batchPredictionJobs/; setting exclude_api_patterns replaces the default entirely.
from coolhand import DEFAULT_EXCLUDE_API_PATTERNS
# Extend the defaults rather than replace them
coolhand_client = Coolhand(
api_key='your-api-key',
exclude_api_patterns=DEFAULT_EXCLUDE_API_PATTERNS + ['/health'],
)
Self-Hosted Deployments
If you run your own Coolhand-compatible backend (e.g. for compliance or data-residency requirements), point the SDK at your host with base_url:
from coolhand import Coolhand
coolhand_client = Coolhand(
api_key='your-api-key',
base_url='https://feedback.example.com', # must use https://
)
Or via environment variable (useful for 12-factor deployments):
export COOLHAND_API_KEY=your-api-key
export COOLHAND_BASE_URL=https://feedback.example.com
import coolhand # picks up COOLHAND_BASE_URL automatically
The SDK rejects non-https:// URLs by default. http://localhost and http://127.0.0.1 are allowed for local development only.
Usage Examples
With OpenAI
import coolhand
from openai import OpenAI
client = OpenAI()
response = client.chat.completions.create(
model="gpt-4",
messages=[{"role": "user", "content": "Hello!"}]
)
print(response.choices[0].message.content)
# Request automatically logged to Coolhand!
With Anthropic
import coolhand
from anthropic import Anthropic
client = Anthropic()
response = client.messages.create(
model="claude-3-opus-20240229",
max_tokens=1024,
messages=[{"role": "user", "content": "Hello!"}]
)
print(response.content[0].text)
# Request automatically logged to Coolhand!
With Google Gemini
import coolhand
import google.generativeai as genai
genai.configure(api_key="your-gemini-api-key")
model = genai.GenerativeModel("gemini-pro")
response = model.generate_content("Hello!")
print(response.text)
# Request automatically logged to Coolhand!
With GitHub Copilot SDK
import coolhand
# GitHub Copilot SDK calls are automatically intercepted
# via JsonRpcClient patching — no additional setup needed.
With Streaming
import coolhand
from openai import OpenAI
client = OpenAI()
stream = client.chat.completions.create(
model="gpt-4",
messages=[{"role": "user", "content": "Tell me a story"}],
stream=True
)
for chunk in stream:
if chunk.choices[0].delta.content:
print(chunk.choices[0].delta.content, end="")
# Complete streamed response automatically logged to Coolhand!
What Gets Logged
The monitor captures:
- Request Data: Method, URL, headers, request body
- Response Data: Status code, headers, response body
- Timing: Request timestamp, response timestamp, duration
- LLM-Specific: Model used, token counts, streaming status
Headers containing API keys are automatically sanitized for security.
Supported Libraries
Coolhand intercepts AI API calls through two mechanisms:
httpx patching (covers any library built on httpx):
- OpenAI Python SDK
- Anthropic Python SDK
- Google Gemini (
google-generativeai/google-genai) - GitHub Models via Azure (
models.inference.ai.azure.com) - Any other library using httpx for HTTP requests
requests patching (covers any library built on requests):
azure-ai-inference,azure-openai, and otherazure-core-based SDKs- Any other library using
requestsfor HTTP requests (requestsis optional — patch is skipped if not installed)
JSON-RPC patching (direct protocol interception):
- GitHub Copilot SDK (
JsonRpcClient)
How It Works
- When you import
coolhand, it automatically patches httpx,requests.Session.send(if installed), and the GitHub CopilotJsonRpcClient - Requests to AI APIs are intercepted (OpenAI, Anthropic, Gemini, GitHub Models, GitHub Copilot, and more)
- Request and response data are captured (credentials and sensitive URL parameters sanitized)
- Data is sent to Coolhand asynchronously
- Your application continues without interruption
For non-LLM endpoints, requests pass through unchanged with zero overhead.
Feedback Service
Collect user feedback on LLM responses to improve your AI outputs. The FeedbackService lets you capture sentiment ratings, explanations, and corrections.
Frontend Feedback Widget: For browser-based feedback collection, see coolhand-js - an accessible, lightweight JavaScript widget that leverages best UX practices to capture actionable user feedback on any AI output.
Basic Usage
from coolhand import Coolhand
# Initialize with your API key
ch = Coolhand(api_key='your-api-key')
# Submit positive feedback
ch.create_feedback({
'llm_request_log_id': 12345, # From Coolhand logs
'sentiment': 'like',
'explanation': 'Very helpful response!'
})
# Using original output for fuzzy matching
ch.create_feedback({
'original_output': 'The capital of France is London.',
'sentiment': 'dislike',
'revised_output': 'The capital of France is Paris.'
})
Feedback Fields
All fields are optional. At least one matching field (marked *) is recommended to link feedback to the original LLM request.
| Field | Type | Description |
|---|---|---|
sentiment |
str | Preferred. "like", "dislike", or "neutral" |
like |
bool | Deprecated — use sentiment instead. Auto-converted and stripped from the wire payload. Emits DeprecationWarning. |
llm_request_log_id |
int | * Coolhand log ID (exact match) |
llm_provider_unique_id |
str | * Provider's x-request-id (exact match) |
original_output |
str | * Original response text (fuzzy match) |
client_unique_id |
str | * Your internal identifier |
explanation |
str | Why the response was good/bad |
revised_output |
str | User's corrected version |
creator_unique_id |
str | ID of user providing feedback |
workload_hashid |
str | Associate feedback with a specific workload |
collector |
str | Override the SDK-generated collector string |
Troubleshooting
Enable Debug Output
from coolhand import Coolhand
Coolhand(
api_key='your-api-key',
silent=False # Enable verbose logging
)
Or via environment variable:
export COOLHAND_SILENT=false
API Key
Sign up for free at coolhandlabs.com to get your API key and start monitoring your LLM usage.
What you get:
- Complete LLM request and response logging
- Usage analytics and insights
- No credit card required to start
Security
- API keys in request headers are automatically redacted
- Sensitive URL query parameters (
key,api_key,token, etc.) are automatically redacted - No sensitive data is exposed in logs
- All data is sent via HTTPS to Coolhand servers
Related Packages
- Frontend (Feedback Collection Widget): coolhand-js - Frontend feedback widget for collecting user feedback on AI outputs
- Ruby: coolhand gem - Coolhand monitoring for Ruby applications
- Node.js: coolhand-node package - Coolhand monitoring for Node.js applications
Community
- Questions? Create an issue
- Contribute? Submit a pull request
- Support? Visit coolhandlabs.com
License
Apache-2.0
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file coolhand-0.4.0.tar.gz.
File metadata
- Download URL: coolhand-0.4.0.tar.gz
- Upload date:
- Size: 61.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.8
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ca8126a48df51da004eb08f72ffbaaee3715f78ecb277f2b476a6756c0b0b5e4
|
|
| MD5 |
1e15c4f783a28039bb17f358c4f38f0c
|
|
| BLAKE2b-256 |
e9936fa5665a5ea10f9711b8f48b26cc0f90561a4647cf3bdfc840c521cd38ea
|
File details
Details for the file coolhand-0.4.0-py3-none-any.whl.
File metadata
- Download URL: coolhand-0.4.0-py3-none-any.whl
- Upload date:
- Size: 25.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.8
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3f41f339bcb5927be44466cd5e8f6b1dcf0867c087677ab4b9dd491fb4912125
|
|
| MD5 |
d30f2282b0bd4b811aad85097e3cc874
|
|
| BLAKE2b-256 |
8f38b56880b8e64eacd374b65e33bce27a07260fbb51d71054211bf231aa9bc4
|