This guide provides SQL query examples for common analytics scenarios covering all the use cases from the legacy API and more. All examples use ClickHouse SQL syntax and work with the /v2/analytics.getVerifications endpoint.
Using Queries in API Requests
When making API requests, you need to format the SQL query as a JSON string on a single line. Here’s how:
SELECT COUNT(*) as total
FROM key_verifications_v1
WHERE time >= now() - INTERVAL 7 DAY
Each example below shows both the readable multi-line SQL and the single-line
JSON format you can copy directly into your API requests.
Usage Analytics
Use this for: High-level usage metrics, health monitoring, and trend analysis.
Key patterns: Total counts, outcome breakdowns, time series analysis.
Total verifications in the last 7 days
Count total verifications across all APIs in the last 7 days.
SELECT SUM(count) as total_verifications
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 7 DAY
Verifications by outcome
Break down verifications by outcome to understand success vs failure rates.
SELECT
outcome,
SUM(count) as count
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 30 DAY
GROUP BY outcome
ORDER BY count DESC
Daily verification trend
Track daily verification patterns over the last 30 days.
SELECT
time as date,
SUM(count) as verifications
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 30 DAY
GROUP BY date
ORDER BY date
Hourly breakdown for today
Analyze hourly verification patterns for today with outcome breakdown.
SELECT
time as hour,
outcome,
SUM(count) as verifications
FROM key_verifications_per_hour_v1
WHERE time >= toStartOfDay(now())
GROUP BY time, outcome
ORDER BY time, outcome
Usage by User
Use this for: Understanding user behavior, identifying power users, tracking user activity over time.
Key patterns: User ranking, activity trends, specific user analysis.
All users ranked by usage
Rank all users by their total verification usage over the last 30 days.
SELECT
external_id,
SUM(count) as total_verifications,
SUM(CASE WHEN outcome = 'VALID' THEN count ELSE 0 END) as successful,
SUM(CASE WHEN outcome = 'RATE_LIMITED' THEN count ELSE 0 END) as rate_limited
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 30 DAY
AND external_id != ''
GROUP BY external_id
ORDER BY total_verifications DESC
LIMIT 100
Usage for a specific user
Analyze usage patterns for a specific user over the last 30 days.
SELECT
SUM(count) as total_verifications,
SUM(CASE WHEN outcome = 'VALID' THEN count ELSE 0 END) as successful,
SUM(CASE WHEN outcome = 'RATE_LIMITED' THEN count ELSE 0 END) as rate_limited
FROM key_verifications_per_day_v1
WHERE external_id = 'user_123'
AND time >= now() - INTERVAL 30 DAY
Top 10 users by API usage
Identify your most active users by verification count.
SELECT
external_id,
SUM(count) as total_verifications
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 30 DAY
AND external_id != ''
GROUP BY external_id
ORDER BY total_verifications DESC
LIMIT 10
Daily usage per user
Track daily verification patterns for each user over 30 days.
SELECT
external_id,
time as date,
SUM(count) as verifications
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 30 DAY
GROUP BY external_id, date
ORDER BY external_id, date
API Analytics
Use this for: Comparing API performance, usage across different APIs, API-specific analysis.
Key patterns: API comparison, success rates, per-API breakdowns.
Usage per API
Compare usage across all APIs to identify most active endpoints.
SELECT
key_space_id,
SUM(count) as total_verifications,
SUM(CASE WHEN outcome = 'VALID' THEN count ELSE 0 END) as successful
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 30 DAY
GROUP BY key_space_id
ORDER BY total_verifications DESC
Usage for a specific API
Analyze detailed usage patterns for a specific API over 30 days.
SELECT
SUM(count) as total_verifications,
SUM(CASE WHEN outcome = 'VALID' THEN count ELSE 0 END) as successful,
SUM(CASE WHEN outcome = 'RATE_LIMITED' THEN count ELSE 0 END) as rate_limited,
SUM(CASE WHEN outcome = 'INVALID' THEN count ELSE 0 END) as invalid
FROM key_verifications_per_day_v1
WHERE key_space_id = 'ks_1234'
AND time >= now() - INTERVAL 30 DAY
Compare multiple APIs
Calculate success rates for multiple APIs to compare performance.
SELECT
key_space_id,
SUM(count) as verifications,
round(SUM(CASE WHEN outcome = 'VALID' THEN count ELSE 0 END) / SUM(count) * 100, 2) as success_rate
FROM key_verifications_per_day_v1
WHERE key_space_id IN ('ks_1234', 'ks_5678')
AND time >= now() - INTERVAL 7 DAY
GROUP BY key_space_id
Key Analytics
Use this for: Individual API key analysis, identifying problematic keys, key-specific usage patterns.
Key patterns: Key ranking, error analysis, specific key monitoring.
Usage per key
Identify your most frequently used API keys over the last 30 days.
SELECT
key_id,
SUM(count) as total_verifications,
SUM(CASE WHEN outcome = 'VALID' THEN count ELSE 0 END) as successful
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 30 DAY
GROUP BY key_id
ORDER BY total_verifications DESC
LIMIT 100
Usage for a specific key
Analyze detailed usage patterns for a specific API key.
SELECT
SUM(count) as total_verifications,
SUM(CASE WHEN outcome = 'VALID' THEN count ELSE 0 END) as successful,
SUM(CASE WHEN outcome = 'RATE_LIMITED' THEN count ELSE 0 END) as rate_limited
FROM key_verifications_per_day_v1
WHERE key_id = 'key_1234'
AND time >= now() - INTERVAL 30 DAY
Keys with most errors
Find API keys that are generating the most errors.
SELECT
key_id,
SUM(count) as total_errors,
groupArray(DISTINCT outcome) as error_types
FROM key_verifications_per_day_v1
WHERE outcome != 'VALID'
AND time >= now() - INTERVAL 7 DAY
GROUP BY key_id
ORDER BY total_errors DESC
LIMIT 20
Tag-Based Analytics
Use this for: Custom metadata filtering, endpoint analysis, user segmentation using tags.
Key patterns: Tag filtering, endpoint breakdowns, custom attribute analysis.
Tags allow you to add custom metadata to verification requests for filtering and aggregation.
Filter by single tag
Count verifications for requests with a specific tag.
SELECT SUM(count) as total
FROM key_verifications_per_day_v1
WHERE has(tags, 'path=/api/v1/users')
AND time >= now() - INTERVAL 7 DAY
Count verifications matching any of multiple tags.
SELECT SUM(count) as total
FROM key_verifications_per_day_v1
WHERE hasAny(tags, ['path=/api/v1/users', 'path=/api/v1/posts'])
AND time >= now() - INTERVAL 7 DAY
Count verifications matching all specified tags.
SELECT SUM(count) as total
FROM key_verifications_per_day_v1
WHERE hasAll(tags, ['environment=production', 'team=backend'])
AND time >= now() - INTERVAL 7 DAY
Group by tag
Aggregate verifications by individual tags to see usage patterns.
SELECT
arrayJoin(tags) as tag,
SUM(count) as verifications
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 7 DAY
GROUP BY tag
ORDER BY verifications DESC
LIMIT 20
Breakdown by endpoint (using path tag)
Analyze request volume by API endpoint over the last 24 hours.
This query uses the raw table for detailed tag analysis. For longer time
ranges, consider using aggregated tables and pre-filtered tags.
SELECT
arrayJoin(arrayFilter(x -> startsWith(x, 'path='), tags)) as endpoint,
COUNT(*) as requests
FROM key_verifications_v1
WHERE time >= now() - INTERVAL 24 HOUR
GROUP BY endpoint
ORDER BY requests DESC
Billing & Usage-Based Pricing
Use this for: Usage-based billing implementation, credit tracking, user tier calculation.
Key patterns: Credit aggregation, billing cycles, tier determination, cost analysis.
Monthly credits per user
Calculate monthly credit consumption per user for billing.
SELECT
external_id,
toStartOfMonth(time) as month,
SUM(spent_credits) as total_credits
FROM key_verifications_per_day_v1
WHERE external_id != ''
AND time >= toStartOfMonth(now())
GROUP BY external_id, month
ORDER BY total_credits DESC
Current billing period credits
Calculate credit usage for a specific billing period.
SELECT
external_id,
SUM(spent_credits) as credits_this_period
FROM key_verifications_per_day_v1
WHERE external_id = 'user_123'
AND time >= 1704067200000 -- Start of billing period (Unix millis)
AND time < 1706745600000 -- End of billing period (Unix millis)
GROUP BY external_id
Credit-based tier calculation
Determine user tiers based on monthly credit consumption.
SELECT
external_id,
SUM(spent_credits) as total_credits,
CASE
WHEN total_credits <= 1000 THEN 'free'
WHEN total_credits <= 10000 THEN 'starter'
WHEN total_credits <= 100000 THEN 'pro'
ELSE 'enterprise'
END as tier
FROM key_verifications_per_day_v1
WHERE time >= toStartOfMonth(now())
AND external_id = 'user_123'
GROUP BY external_id
Daily credit usage and cost
Track daily credit consumption and calculate estimated costs.
SELECT
time as date,
SUM(spent_credits) as credits_used,
credits_used * 0.001 as estimated_cost -- $0.001 per credit
FROM key_verifications_per_day_v1
WHERE external_id = 'user_123'
AND time >= now() - INTERVAL 30 DAY
GROUP BY date
ORDER BY date
Advanced Queries
Use this for: Complex analytical patterns, cohort analysis, moving averages, advanced insights.
Key patterns: User retention, trend smoothing, complex joins, window functions.
Cohort analysis: New vs returning users
Perform cohort analysis to understand user retention patterns.
WITH first_seen AS (
SELECT
external_id,
min(time) as first_verification
FROM key_verifications_per_day_v1
WHERE external_id != ''
GROUP BY external_id
)
SELECT
toDate(kv.time) as date,
SUM(CASE WHEN kv.time = fs.first_verification THEN kv.count ELSE 0 END) as new_users,
SUM(CASE WHEN kv.time > fs.first_verification THEN kv.count ELSE 0 END) as returning_users
FROM key_verifications_per_day_v1 kv
JOIN first_seen fs ON kv.external_id = fs.external_id
WHERE kv.time >= now() - INTERVAL 30 DAY
GROUP BY date
ORDER BY date
Moving average (7-day)
Calculate 7-day moving average to smooth out daily fluctuations.
SELECT
date,
verifications,
avg(verifications) OVER (
ORDER BY date
ROWS BETWEEN 6 PRECEDING AND CURRENT ROW
) as moving_avg_7d
FROM (
SELECT
time as date,
SUM(count) as verifications
FROM key_verifications_per_day_v1
WHERE time >= now() - INTERVAL 60 DAY
GROUP BY date
)
ORDER BY date
Using Aggregated Tables
For better performance on large time ranges, use pre-aggregated tables:
Hourly aggregates
Query hourly verification counts for the last 7 days.
SELECT
time,
SUM(count) as total
FROM key_verifications_per_hour_v1
WHERE time >= toStartOfHour(now() - INTERVAL 7 DAY)
GROUP BY time
ORDER BY time
Daily aggregates
Query daily verification counts for the last 30 days.
SELECT
time,
SUM(count) as total
FROM key_verifications_per_day_v1
WHERE time >= toStartOfDay(now() - INTERVAL 30 DAY)
GROUP BY time
ORDER BY time
Monthly aggregates
Query monthly verification counts for the last year.
SELECT
time,
SUM(count) as total
FROM key_verifications_per_month_v1
WHERE time >= toStartOfMonth(now() - INTERVAL 12 MONTH)
GROUP BY time
ORDER BY time
Tips for Efficient Queries
- Always filter by time - Use indexes by including time filters
- Use aggregated tables - Hourly/daily/monthly tables for longer ranges
- Add LIMIT clauses - Prevent returning too much data
- Filter before grouping - Use WHERE instead of HAVING when possible