npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

coolhand-node

v0.7.0

Published

Monitor and LLM API calls for analysis

Readme

Coolhand Node.js Monitor

npm version

Monitor and log LLM API calls from multiple providers (OpenAI, Anthropic, Google AI, GitHub Models, Vertex AI, Cloudflare AI Gateway, and more) to the Coolhand analytics platform.

Related Packages

| Package | Environment | Purpose | |---------|-------------|---------| | coolhand-node | Node.js | Server-side monitoring and logging of LLM API calls | | coolhand | Browser | Feedback widget for collecting user sentiment on AI outputs |

This package (coolhand-node) is the server-side SDK for monitoring LLM calls. For browser-based feedback collection widgets, see coolhand.

Installation

npm install coolhand-node

Getting Started

  1. Get API Key: Visit coolhandlabs.com and get an API key
  2. Install: npm install coolhand-node
  3. Initialize: Add monitoring to your app's startup (see examples below)
  4. Configure: Set COOLHAND_API_KEY in your environment variables
  5. Deploy: Your AI calls are now automatically monitored!

Quick Start

Option 1: Universal Global Monitoring (Recommended)

🔥 RECOMMENDED - Zero Configuration AI Monitoring

Note: Global monitoring works in Node.js server environments. For React frontend apps, see our React Integration Guide.

Set it and forget it! Monitor ALL AI API calls across your entire application with just one line of code, so you'll never be surprised by new LLM calls added to your production codebase.

Note: coolhand-node ships both ESM and CommonJS builds. See Module System Compatibility for details.

ESM projects ("type": "module" in package.json):

// Add this ONE line at the top of your main application file
import 'coolhand-node/auto-monitor';

// That's it! ALL AI API calls are now automatically monitored:
// ✅ OpenAI SDK calls
// ✅ LangChain operations
// ✅ Anthropic API calls
// ✅ Custom AI libraries
// ✅ Direct fetch/axios requests to AI APIs
// ✅ ANY library making AI API calls

// NO code changes needed in your existing services!

ESM + LangChain / OpenAI static imports: @langchain/openai (and the openai SDK) cache a reference to https.request at module load time. In CJS projects (TypeScript compiling to CommonJS) import 'coolhand-node/auto-monitor' patches https.request synchronously and everything works automatically. In native ESM projects ("type": "module") the patch cannot be applied synchronously from a sibling import — use one of the approaches below.

Recommended — --import flag (Node ≥ 20.6, no code changes):

# CLI
node --import 'coolhand-node/auto-monitor' app.mjs

# Via environment variable (Docker, CI, process managers)
NODE_OPTIONS="--import 'coolhand-node/auto-monitor'" node app.mjs
{ "scripts": { "start": "node --import 'coolhand-node/auto-monitor' dist/app.mjs" } }

--import loads auto-monitor before the application's module graph is evaluated, so https.request is patched before any library can cache it — no refactoring needed.

Fallback — dynamic import (Node < 20.6):

import { initializeGlobalMonitoring } from 'coolhand-node';

await initializeGlobalMonitoring({ apiKey: process.env.COOLHAND_API_KEY });

// Dynamic import runs after https.request is patched
const { ChatOpenAI } = await import('@langchain/openai');
const model = new ChatOpenAI({ model: 'gpt-4o' });
await model.invoke('hello'); // ✅ intercepted

Environment Variables:

# .env
COOLHAND_API_KEY=your_api_key_here
COOLHAND_DEBUG=false     # Set to true for verbose logging
COOLHAND_DRY_RUN=false   # Set to true to skip API submissions (dry-run mode)

Or manual initialization:

import { initializeGlobalMonitoring } from 'coolhand-node';

// Initialize once at application startup
await initializeGlobalMonitoring({
  apiKey: 'your-api-key',
  dryRun: false  // Set to true to skip all data submission (dry-run mode)
});

// Now ALL outbound AI API calls are automatically monitored

✨ Why Global Monitoring is Recommended:

  • 🚫 Zero refactoring - No code changes to existing services
  • 📊 Complete coverage - Monitors ALL AI libraries automatically
  • 🔒 Security built-in - Automatic credential sanitization
  • Performance optimized - Negligible overhead
  • 🛡️ Future-proof - Automatically captures new AI calls added by your team

Option 2: Instance-Based Monitoring (Explicit Control)

For cases where you need explicit control over which AI calls are monitored:

import Coolhand from 'coolhand-node';

// Initialize the monitor
const monitor = new Coolhand({
    apiKey: 'your-api-key',
    dryRun: false  // Set to true to skip data submission (dry-run mode)
});

Module System Compatibility

coolhand-node ships both ESM and CommonJS builds, so it works in any Node.js project regardless of module system.

ESM Projects

If your package.json has "type": "module", or you use .mjs files:

import { initializeGlobalMonitoring } from 'coolhand-node';
// or
import 'coolhand-node/auto-monitor';

LangChain / OpenAI users: See the ESM static import caveat above — use await initializeGlobalMonitoring(...) followed by a dynamic await import('@langchain/openai') to ensure the patch is applied before the library loads.

CommonJS Projects

If your project uses CommonJS (no "type": "module", or .cjs files), require() works directly.

Option A — zero-config auto-monitor (reads COOLHAND_API_KEY from the environment):

require('coolhand-node/auto-monitor');
// That's it — all AI API calls are now monitored

Option B — manual initialization (explicit control over options):

const { initializeGlobalMonitoring } = require('coolhand-node');

async function startServer() {
  await initializeGlobalMonitoring({
    apiKey: process.env.COOLHAND_API_KEY,
    silent: true
  });

  // ... start your server
}

startServer();

TypeScript Compiling to CommonJS

If your tsconfig.json has "module": "commonjs", require() works directly — no workarounds needed.

Option A — zero-config auto-monitor:

require('coolhand-node/auto-monitor');

Option B — manual initialization:

const { initializeGlobalMonitoring } = require('coolhand-node');

async function startServer() {
  await initializeGlobalMonitoring({
    apiKey: process.env.COOLHAND_API_KEY,
    silent: true
  });

  // ... start your server
}

See the framework guides for complete examples.

Feedback API

Collect feedback on LLM responses to improve model performance.

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.

import { Coolhand } from 'coolhand-node';

const coolhand = new Coolhand({
  apiKey: 'your-api-key'
});

// Create feedback for an LLM response
const feedback = await coolhand.createFeedback({
  llm_request_log_id: 123,
  llm_provider_unique_id: 'req_xxxxxxx',
  client_unique_id: 'workorder-chat-456',
  creator_unique_id: 'user-789'
  original_output: 'Here is the original LLM response!',
  revised_output: 'Here is the human edit of the original LLM response.',
  explanation: 'Tone of the original response read like AI-generated open source README docs',
  sentiment: 'like', // preferred; replaces `like: true`
});

Field Guide: All fields are optional, but here's how to get the best results:

Matching Fields

  • llm_request_log_id 🎯 Exact Match - ID from the Coolhand API response when the original LLM request was logged. Provides exact matching.
  • llm_provider_unique_id 🎯 Exact Match - The x-request-id from the LLM API response (e.g., "req_xxxxxxx")
  • original_output 🔍 Fuzzy Match - The original LLM response text. Provides fuzzy matching but isn't 100% reliable.
  • client_unique_id 🔗 Your Internal Matcher - Connect to an identifier from your system for internal matching

Quality Data

  • revised_outputBest Signal - End user revision of the LLM response. The highest value data for improving quality scores.

  • explanation 💬 Medium Signal - End user explanation of why the response was good or bad. Valuable qualitative data.

  • sentiment 🎭 Preferred - String: 'like', 'dislike', or 'neutral'. Takes precedence if both sentiment and like are provided.

  • like 👍 Low Signal (Deprecated) - Boolean: true = like, false = dislike. Use sentiment instead. Automatically converted to sentiment before sending.

    | like (boolean) | sentiment (string) | |------------------|----------------------| | true | "like" | | false | "dislike" | | omitted | (no conversion) |

  • creator_unique_id 👤 User Tracking - Unique ID to match feedback to the end user who created it

  • workload_hashid 🔗 Workload Association - Associate feedback with a specific workload

Framework Integration

📚 Framework Integration Guide - Complete documentation for all supported frameworks

Supported Frameworks: Works with any Node.js framework (Express.js, NestJS, Fastify, Koa, AWS Lambda, Vercel Functions), extensively tested with Next.js/T3 Stack

Quick Links by Framework:

Configuration Options

Global Monitoring Options

| Option | Type | Default | Description | |--------|------|---------|-------------| | apiKey | string | required | Your Coolhand API key for authentication | | silent | boolean | true | Whether to suppress console output | | debug | boolean | false | Enable verbose logging (endpoint URL and payload size logged before each call) | | dryRun | boolean | false | Skip all API submissions — no data is sent to Coolhand | | patternsFile | string | undefined | Path to custom API patterns file | | excludeApiPatterns | string[] | [] | Glob patterns for endpoints to exclude from monitoring (e.g. health checks). | | baseUrl | string | 'https://coolhandlabs.com' | Override the API host for self-hosted deployments. Must be https:// (or http://localhost for local dev). |

Environment Variables

| Variable | Type | Default | Description | |----------|------|---------|-------------| | COOLHAND_API_KEY | string | required | Your Coolhand API key | | COOLHAND_SILENT | 'true' | 'false' | 'true' | Whether to suppress console output | | COOLHAND_DEBUG | 'true' | 'false' | 'false' | Enable verbose logging | | COOLHAND_DRY_RUN | 'true' | 'false' | 'false' | Skip all API submissions (dry-run mode) | | COOLHAND_PATTERNS_FILE | string | undefined | Path to custom API patterns file | | COOLHAND_BASE_URL | string | undefined | Override the API host (e.g. https://feedback.example.com). Same rules as baseUrl option. |

Instance-Based Monitoring Options

Same options as global monitoring, passed to the Coolhand constructor. Includes baseUrl.

TypeScript Support

Full TypeScript support with exported types:

import { Coolhand, CoolhandOptions, CoolhandCallData, CoolhandStats } from 'coolhand-node';

const monitor = new Coolhand({
  apiKey: 'your-api-key',
  silent: true,
  dryRun: false
});

What Gets Logged

The monitor captures:

  • Request Data: Method, URL, headers, request body
  • Response Data: Status code, headers, response body
  • Metadata: Timestamp, protocol used
  • LLM-Specific: Model used, token counts, temperature settings

Headers containing API keys are automatically sanitized for security.

Supported Libraries

The monitor works with any Node.js library that makes HTTP(S) requests to LLM APIs, including:

  • OpenAI official SDK
  • Anthropic SDK
  • Google AI SDK
  • GitHub Models (via OpenAI SDK pointed at models.github.ai or models.inference.ai.azure.com)
  • Vertex AI (native Gemini and OpenAI-compatible endpoints on aiplatform.googleapis.com)
  • Cloudflare AI Gateway (gateway.ai.cloudflare.com, proxying any upstream provider)
  • LangChain
  • Direct fetch() calls
  • https/http module usage
  • Any other HTTP client

Custom AI Providers

Add support for custom AI providers by creating a patterns file:

import Coolhand from 'coolhand-node';

const monitor = new Coolhand({
    apiKey: 'your-api-key',
    patternsFile: './my-patterns.json'
});

Example patterns file (my-patterns.json):

{
  "patterns": [
    {
      "name": "My Custom AI",
      "domains": ["api.mycustomai.com"],
      "paths": ["/v1/generate", "/v1/chat"],
      "headers": {
        "authorization": "[REDACTED]",
        "api-key": "[REDACTED]"
      }
    }
  ]
}

Monitoring Statistics

Track monitoring statistics in your application:

import { getGlobalStats } from 'coolhand-node';

setInterval(() => {
  const stats = getGlobalStats();
  console.log(`AI Calls: ${stats.interceptedCalls}, Total Requests: ${stats.totalRequests}`);
}, 60000);

Dry-Run Mode

Use dryRun: true (or COOLHAND_DRY_RUN=true) to prevent any data from being sent to Coolhand — useful for CI environments or initial integration testing:

// Via environment variable
// Set COOLHAND_DRY_RUN=true in .env, then:
import 'coolhand-node/auto-monitor';

// Or via manual initialization
import { initializeGlobalMonitoring } from 'coolhand-node';

await initializeGlobalMonitoring({
  apiKey: 'your-api-key',
  dryRun: true
});

When dry-run mode is enabled:

  • All API calls to Coolhand are skipped
  • Log messages indicate what would have been sent
  • No data reaches Coolhand servers

Debug Mode (Verbose Logging)

Use debug: true (or COOLHAND_DEBUG=true) to enable extra logging — the endpoint URL and payload size are printed before each outbound call. Data is still submitted normally.

await initializeGlobalMonitoring({
  apiKey: 'your-api-key',
  debug: true  // extra logs, data still sent
});

Migrating from v0.3.x to v0.4.0

The environment option has been removed. Use baseUrl instead:

// before (≤0.3.x)
new Coolhand({ apiKey, environment: 'local' });
initializeGlobalMonitoring({ apiKey, environment: 'local' });

// after (≥0.4.0)
new Coolhand({ apiKey, baseUrl: 'http://localhost:3000' });
initializeGlobalMonitoring({ apiKey, baseUrl: 'http://localhost:3000' });

environment: 'production' can simply be removed — the default endpoint (https://coolhandlabs.com) is unchanged.

Migrating from v0.4.x

Prior to v0.5.0, debug: true suppressed all API submissions. This behavior has been renamed to dryRun: true. If you previously used debug: true to prevent data from being sent, replace it with dryRun: true. Passing debug: true without dryRun: true will emit a console.warn deprecation notice.

Advanced Usage

Modular Architecture

Access individual services for advanced use cases:

import { PatternMatchingService, LoggingService } from 'coolhand-node';

// Use pattern matching independently
const patternService = new PatternMatchingService('./custom-patterns.json');
const match = patternService.matchesAPIPattern(requestOptions);

// Use logging service independently
const loggingService = new LoggingService({
  apiKey: 'your-key',
  silent: false,
  dryRun: 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
  • Feedback collection and quality scoring
  • No credit card required to start

Error Handling

The monitor handles errors gracefully:

  • Failed API logging attempts are logged to console but don't interrupt your application
  • Invalid API keys will be reported but won't crash your app
  • Network issues are handled with appropriate error messages

Security

  • API keys in request headers are automatically redacted
  • No sensitive data is exposed in logs
  • Dry-run mode (dryRun: true) prevents any data from being sent to external servers

Documentation

  • Framework Integration Guide - Complete setup for all frameworks. (Well, some are more complete than others.)
  • Global Monitoring Guide - Advanced global monitoring features. Even easier than asking your favorite LLM coding tool to do it for you.
  • React Integration Guide - Frontend integration patterns. We won't ask about how you are planning to keep your API keys secret.
  • Manual Submission API - Submit captured LLM requests outside of automatic monitoring (e.g. from a CLI tool).

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
  • Python: coolhand package - Coolhand monitoring for Python applications

Community