skills/consuming-endpoints-from-client-code/SKILL.md
Wire a PostHog endpoint into a client app or SDK. Covers fetching the OpenAPI spec, generating a typed client with openapi-generator or @hey-api/openapi-ts, sending the right auth header, shaping the variables payload (HogQL code_name vs insight breakdown property), handling rate-limit and materialised-endpoint error responses. Use when the user says "how do I call my endpoint", "generate a client for this", or "what auth header do I use".
npx skillsauth add posthog/ai-plugin consuming-endpoints-from-client-codeInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
This skill is the caller-side counterpart to creating-an-endpoint. It helps integrate an
existing endpoint into a separate codebase — a mobile app, server backend, customer dashboard,
or downstream pipeline. No PostHog code is modified here.
user_id" → materialised-endpoint variable
questionsIf the user is creating the endpoint, use creating-an-endpoint first.
| Tool | Purpose |
| ----------------------- | ---------------------------------------------------------------------------------------------------------- |
| endpoint-get | Full config for a named endpoint, including the query shape and required variables |
| endpoint-openapi-spec | OpenAPI 3.0 spec for one endpoint, ready to feed to a code generator |
| endpoint-run | A live call against the endpoint — useful to confirm a payload works before sharing it with the user's app |
/api/projects/{team_id}/endpoints/{name}/run
team_id is the project ID (numeric). Available in PostHog under project settings, or via
posthog-get-projects if the user doesn't know it.name is the endpoint name — see endpoints-get-all if the user isn't sure./run is required.POST is the canonical method. GET also works for simple cases without a request body but
POST is preferred — variables go in the body.
Endpoints are authenticated with a personal API key. The header is:
Authorization: Bearer <key>
Keys are scoped — for endpoints, the key needs at least endpoint:read. If the user gets a 403,
they're usually missing the scope; if they get a 401, the key is missing or malformed.
Never put a personal API key in client-side code that's shipped to end users (mobile apps, browser JS). Personal API keys grant scoped account access. For customer-facing apps, route through the user's own backend, which holds the key.
{
"variables": { "code_name_1": value, "code_name_2": value },
"limit": 100,
"offset": 0,
"refresh": "cache"
}
| Field | Notes |
| ----------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| variables | Keyed by code_name for HogQL endpoints; for insight endpoints with breakdowns, key is the breakdown property name |
| limit | Max rows returned. |
| offset | Skip rows. Only HogQL endpoints |
| refresh | "cache" (return cached results if fresh enough), "force" (always recalculate), "direct" (bypass materialisation, materialised endpoints only). Default is "cache" |
Call endpoint-get to see the exact variable shape. The response includes the query definition
with declared variables — each variable's code_name is what the client should send.
If endpoint-get shows is_materialized: true on the current version, the endpoint requires
every declared variable to be passed on each call. This is a security boundary — without
filters, a single call would return the entire pre-aggregated dataset.
Common symptom: the user's app worked when the endpoint was unmaterialised, then started returning 400 errors after materialisation was enabled. The error message lists which variables are missing.
Optional/partial variables on materialised endpoints are a known limitation the PostHog team plans
to lift. If requiring every variable is blocking the user's use case, send a note via the
agent-feedback tool — that demand signal is how the team prioritises it.
The endpoint exposes its own OpenAPI 3.0 spec via endpoint-openapi-spec. Feed that into a code
generator:
| Language | Tool | Command shape |
| ---------- | ----------------------- | -------------------------------------------------------------------------------- |
| TypeScript | @hey-api/openapi-ts | openapi-ts -i spec.json -o ./generated |
| TypeScript | openapi-generator-cli | openapi-generator-cli generate -i spec.json -g typescript-fetch -o ./generated |
| Python | openapi-generator-cli | openapi-generator-cli generate -i spec.json -g python -o ./generated |
| Go | oapi-codegen | oapi-codegen -package=client spec.json > client.go |
The generated client gives the user types for the variables payload and the response shape. Re- generate when the endpoint's query changes (each new version may have different variables).
If the user has multiple endpoints, generate a spec per endpoint and either combine them, or generate one client per endpoint and use them side-by-side.
A typical successful response:
{
"results": [[...], [...]],
"columns": ["col_a", "col_b"],
"types": ["Int64", "String"],
"hasMore": false,
"name": "endpoint_name",
"endpoint_version": 4,
"endpoint_version_created_at": "2026-01-15T..."
}
results is an array of rows; each row is an array of cell values in the order of columns.endpoint_version tells the client which version actually ran — useful for logging and for
pinning to a known version with ?version=N.For insight endpoints, the response shape depends on the query kind (TrendsQuery,
LifecycleQuery, RetentionQuery) — the OpenAPI spec captures the right shape for the current
version. Insight kinds that can't be materialised (e.g. FunnelsQuery) still return their inline
result shape.
For local testing, scripts, or CI, the repo's posthog-cli calls endpoints without hand-rolling
HTTP:
posthog-cli exp endpoints run — execute an endpoint (from a local YAML definition)posthog-cli exp endpoints {list,get,pull,push,diff} — inspect endpoints, or manage them as YAML
files in version control (GitOps-style)Auth uses the same personal API key, via posthog-cli login or the POSTHOG_CLI_API_KEY /
POSTHOG_CLI_PROJECT_ID / POSTHOG_CLI_HOST env vars. (These live under exp — experimental, may
change.)
| Status | When | Handling |
| ------ | ------------------------------------------------------------------------------ | -------------------------------------------------------------------------------- |
| 400 | Missing required variable on a materialised endpoint, or invalid variable type | Surface the error message; fix the call |
| 401 | Missing / wrong personal API key | Check the Authorization header |
| 403 | Key lacks endpoint:read scope, or endpoint is in another project | Adjust key scopes |
| 404 | Endpoint name typo, or endpoint not active | Confirm name; check is_active |
| 429 | Rate limited — limits are per team, not per endpoint (see note below) | Exponential backoff; cache responses client-side if possible |
| 5xx | Query execution failure (ClickHouse error, timeout, etc.) | Retry with backoff. If persistent, hand off to diagnosing-endpoint-performance |
execute-sql on
system.data_modeling_endpoints (or endpoints-get-all).endpoint-get to see the full shape: variables, materialisation status, query kind.endpoint-openapi-spec, hand them the spec, and
recommend a generator for their language.endpoint-run from this skill to confirm the payload works.?version=N.User: "I want to call monthly_active_users from my Node.js backend.
Can you help me wire it up with types?"
Agent steps:
- endpoint-get monthly_active_users → HogQL, variables: lookback_days
(int, default 30), org_id (string, no default), is_materialized: true
- "Two things to know upfront:
1. This endpoint is materialised, so both lookback_days and org_id
must be passed on every call. Calls without them will be rejected.
2. You'll need a personal API key with endpoint:read scope. Set it
as PH_API_KEY in your server env — never ship it in client code."
- endpoint-openapi-spec monthly_active_users → returns spec
- "Here's the spec. Drop it in your repo and run:
pnpm dlx @hey-api/openapi-ts -i spec.json -o ./generated/posthog
That gives you typed client functions.
A minimum call looks like:
POST /api/projects/12345/endpoints/monthly_active_users/run
Authorization: Bearer <PH_API_KEY>
Content-Type: application/json
{ 'variables': { 'org_id': 'acme', 'lookback_days': 30 } }
The response includes columns and rows — your client will pick that
up from the generated types.
Want me to do a sample call to verify the payload works?"
?version=N. Without it the
latest active version runs, so a future query edit (which cuts a new version) can silently change
a caller's results. Bump the pinned version deliberately once you've validated the new one.data_freshness_seconds, but the client can layer another cache on top for hot paths. Be
mindful of total staleness (endpoint cache + client cache).agent-feedback tool — it's the main signal the team uses to improve endpoints and these
tools.tools
Focused Signals scout for PostHog projects with web traffic. Watches the acquisition and site-health layer the web analytics product reports on: per-channel session volume diverging from the site's own rhythm (an acquisition source silently collapsing or surging), attribution breakage (paid/campaign traffic reclassifying into Direct or Unknown when tagging breaks), landing pages that break (bounce-rate steps, 404 spikes, entry-path cliffs), and page-performance regressions (web vitals p75 steps). Emits findings only when they clear the confidence bar; otherwise writes durable memory and closes out empty. Self-contained peer in the signals-scout-* fleet.
tools
Focused Signals scout for PostHog projects using session replay. Watches two promises the replay product makes: that sessions are actually being recorded (capture integrity — recording volume vanishing while site traffic doesn't), and that the friction evidence inside recordings gets seen (rage-click / dead-click clusters concentrating on a page or element, error-after-interaction cohorts, recurring replay vision themes nobody aggregates). Emits findings only when they clear the confidence bar; otherwise writes durable memory and closes out empty. Self-contained peer in the signals-scout-* fleet.
tools
Focused Signals scout for PostHog setup health. Reads the project's active health issues — the deterministic findings of PostHog's own health checks (no live events, outdated SDKs, missing reverse proxy, absent web vitals, ingestion warnings, failing data-warehouse models, and more) — and decides which are genuinely worth surfacing. Unlike a one-signal-per-issue push, it bundles kind-clusters into a single finding, weights by real blast radius (cross-referencing actual event volume and reach), and prioritizes issues an agent can resolve via the MCP. Emits only above the confidence bar; otherwise writes durable memory and closes out empty. Self-contained peer in the signals-scout-* fleet — no dependencies on other skills.
tools
Focused Signals scout for PostHog projects using feature flags. Watches the flag roster and the `$feature_flag_called` evaluation stream for contradictions between a flag's configured state and its real traffic: evaluation cliffs on healthy flags, ghost flags (code calling keys that no longer exist), response-distribution shifts with no corresponding flag edit, and flag debt (stale, fully-rolled-out, or dead flags still burning evaluations). Emits findings only when they clear the confidence bar; otherwise writes durable memory and closes out empty. Self-contained peer in the signals-scout-* fleet — no dependencies on other skills.