Insights Queries¶
Build typed analytics queries against Mixpanel's Insights engine — the same engine that powers the Mixpanel web UI.
New in v0.2
Workspace.query() is the primary way to run analytics queries programmatically. It supports capabilities not available through the legacy query methods, including DAU/WAU/MAU, multi-metric comparison, formulas, per-user aggregation, rolling windows, and percentiles.
When to Use query()¶
query() uses the Insights engine via inline bookmark params. The legacy methods (segmentation(), funnel(), retention()) use the older Query API endpoints. Use query() when you need any of the capabilities in the right column:
| Capability | Legacy methods | query() |
|---|---|---|
| Simple event count over time | segmentation() |
ws.query("Login") |
| Unique users | segmentation(type="unique") |
math="unique" |
| DAU / WAU / MAU | Not available | math="dau" |
| Multi-metric comparison | Not available | ["Signup", "Login", "Purchase"] |
| Formulas (conversion rates, ratios) | Not available | formula="(B/A)*100" |
| Per-user aggregation | Not available | per_user="average" |
| Rolling / cumulative analysis | Not available | rolling=7 |
| Percentiles (p25/p75/p90/p99) | Not available | math="p90" |
| Typed filters | Expression strings | Filter.equals("country", "US") |
| Numeric bucketed breakdowns | Not available | GroupBy("revenue", property_type="number") |
| Save query as a report | N/A | result.params → create_bookmark() |
Use the legacy methods when:
- You need to query a saved funnel by ID →
funnel() - You need cohort retention curves →
query_retention()(Retention Queries) - You need raw JQL execution →
jql() - You need to query a saved Flows report →
query_saved_flows()
For ad-hoc funnel conversion analysis with typed step definitions, see Funnel Queries. For ad-hoc flow path analysis with typed step definitions, see Flow Queries.
Getting Started¶
The simplest possible query — total event count per day for the last 30 days:
import mixpanel_data as mp
ws = mp.Workspace()
result = ws.query("Login")
print(result.df.head())
# date event count
# 0 2025-03-01 Login [Total Events] 142
# 1 2025-03-02 Login [Total Events] 158
Add a time range and aggregation:
# Unique users per week for the last 7 days
result = ws.query("Login", math="unique", last=7, unit="week")
# Last 90 days of DAU
result = ws.query("Login", math="dau", last=90)
# Specific date range
result = ws.query(
"Purchase",
from_date="2025-01-01",
to_date="2025-03-31",
unit="month",
)
Aggregation¶
Counting¶
| Math type | What it counts |
|---|---|
"total" (default) |
Total event occurrences |
"unique" |
Unique users per period |
"dau" |
Daily active users |
"wau" |
Weekly active users |
"mau" |
Monthly active users |
# DAU over the last 90 days
result = ws.query("Login", math="dau", last=90)
# Monthly active users
result = ws.query("Login", math="mau", last=6, unit="month")
Property Aggregation¶
Aggregate a numeric property across events. Requires math_property:
| Math type | Aggregation |
|---|---|
"total" + math_property |
Sum of a numeric property |
"average" |
Mean value |
"median" |
Median value |
"min" / "max" |
Extremes |
"p25" / "p75" / "p90" / "p99" |
Percentiles |
"percentile" + percentile_value |
Custom percentile (e.g. p95) |
"histogram" |
Distribution of property values |
# Average purchase amount per day
result = ws.query(
"Purchase",
math="average",
math_property="amount",
from_date="2025-01-01",
to_date="2025-01-31",
)
# P90 response time
result = ws.query("API Call", math="p90", math_property="duration_ms")
# Custom percentile (p95) — use math="percentile" with percentile_value
result = ws.query(
"API Call",
math="percentile",
math_property="duration_ms",
percentile_value=95,
)
# Histogram — distribution of purchase amounts
result = ws.query("Purchase", math="histogram", math_property="amount")
Per-User Aggregation¶
Aggregate per user first, then across all users — like a SQL subquery. For example, "what's the average number of purchases per user per week?"
# Average purchases per user per week
result = ws.query(
"Purchase",
math="total",
per_user="average",
unit="week",
)
Valid per_user values: "unique_values", "total", "average", "min", "max".
Note
per_user is incompatible with dau, wau, mau, and unique math types.
The Metric Class¶
When different events need different aggregation settings, use Metric objects instead of plain strings:
from mixpanel_data import Metric
# Different math per event
result = ws.query([
Metric("Signup", math="unique"),
Metric("Purchase", math="total", property="revenue"),
])
Metric also supports percentile_value for custom percentiles:
# Per-metric custom percentile
result = ws.query(
Metric("API Call", math="percentile", property="duration_ms", percentile_value=95),
)
Plain strings inherit the top-level math, math_property, and per_user defaults. Metric objects override them per-event:
# These are equivalent:
ws.query("Login", math="unique")
ws.query(Metric("Login", math="unique"))
# Top-level defaults apply to all string events:
ws.query(["Signup", "Login"], math="unique")
# Both events use math="unique"
# Metric overrides per event:
ws.query([
Metric("Signup", math="unique"), # unique users
Metric("Purchase", math="total"), # total events
])
Filters¶
Global Filters¶
Apply filters across all metrics with where=. Construct filters using Filter class methods:
from mixpanel_data import Filter
# Single filter
result = ws.query(
"Purchase",
where=Filter.equals("country", "US"),
)
# Multiple filters (combined with AND)
result = ws.query(
"Purchase",
where=[
Filter.equals("country", "US"),
Filter.greater_than("amount", 50),
],
)
Available Filter Methods¶
String filters:
Filter.equals("browser", "Chrome") # equals value
Filter.equals("browser", ["Chrome", "Firefox"]) # equals any in list
Filter.not_equals("browser", "Safari") # does not equal
Filter.contains("email", "@company.com") # contains substring
Filter.not_contains("url", "staging") # does not contain
Numeric filters:
Filter.greater_than("amount", 100) # > 100
Filter.less_than("age", 65) # < 65
Filter.between("amount", 10, 100) # 10 <= x <= 100
Existence filters:
Boolean filters:
Per-Metric Filters¶
Apply filters to individual metrics using Metric.filters:
from mixpanel_data import Metric, Filter
# Different filters on each event
result = ws.query([
Metric("Purchase", math="unique"),
Metric(
"Purchase",
math="unique",
filters=[Filter.equals("plan", "premium")],
),
])
By default, multiple per-metric filters combine with AND logic. Use filters_combinator="any" for OR logic:
result = ws.query(Metric(
"Purchase",
math="unique",
filters=[
Filter.equals("country", "US"),
Filter.equals("country", "CA"),
],
filters_combinator="any", # match US OR CA
))
Date Filters¶
Filter by datetime properties using purpose-built factory methods:
from mixpanel_data import Filter
# Absolute date filters
Filter.on("created", "2025-01-15") # exact date match
Filter.not_on("created", "2025-01-15") # not on date
Filter.before("created", "2025-01-01") # before a date
Filter.since("created", "2025-01-01") # on or after a date
Filter.date_between("created", "2025-01-01", "2025-06-30") # date range
# Relative date filters — "in the last N units"
Filter.in_the_last("created", 30, "day") # last 30 days
Filter.in_the_last("last_seen", 2, "week") # last 2 weeks
Filter.not_in_the_last("created", 90, "day") # NOT in last 90 days
The relative date methods accept a FilterDateUnit: "hour", "day", "week", or "month".
from mixpanel_data import FilterDateUnit # Literal["hour", "day", "week", "month"]
# Example: recent signups with purchases
result = ws.query(
"Purchase",
where=Filter.in_the_last("signup_date", 7, "day"),
last=30,
)
Breakdowns¶
String Breakdowns¶
Break down results by property values with group_by:
# Simple string breakdown
result = ws.query("Login", group_by="platform", last=14)
# Multiple breakdowns
result = ws.query("Purchase", group_by=["country", "platform"])
The GroupBy Class¶
For numeric bucketing, boolean breakdowns, or explicit type annotations, use GroupBy:
from mixpanel_data import GroupBy
# Numeric breakdown with buckets
result = ws.query(
"Purchase",
group_by=GroupBy(
"revenue",
property_type="number",
bucket_size=50,
bucket_min=0,
bucket_max=500,
),
)
# Boolean breakdown
result = ws.query(
"Login",
group_by=GroupBy("is_premium", property_type="boolean"),
)
# Mixed: string shorthand + GroupBy
result = ws.query(
"Purchase",
group_by=[
"country",
GroupBy("amount", property_type="number", bucket_size=25),
],
)
Formulas¶
Compute derived metrics from multiple events. Letters A-Z reference events by their position in the list.
Top-Level formula Parameter¶
from mixpanel_data import Metric
# Conversion rate: purchases / signups * 100
result = ws.query(
[Metric("Signup", math="unique"), Metric("Purchase", math="unique")],
formula="(B / A) * 100",
formula_label="Conversion Rate",
unit="week",
)
When formula is set, the underlying metrics are automatically hidden — only the formula result appears in the output.
Formula Class in Events List¶
For inline formula definitions, pass Formula objects alongside events:
from mixpanel_data import Metric, Formula
result = ws.query([
Metric("Signup", math="unique"),
Metric("Purchase", math="unique"),
Formula("(B / A) * 100", label="Conversion Rate"),
])
Both approaches produce identical results. Use whichever reads more naturally.
Multi-Metric Comparison (No Formula)¶
Compare multiple events side by side without a formula:
# Three events on the same chart
result = ws.query(
["Signup", "Login", "Purchase"],
math="unique",
last=30,
)
Time Ranges¶
Relative (Default)¶
By default, query() returns the last 30 days. Customize with last and unit:
# Last 7 days (daily granularity)
result = ws.query("Login", last=7)
# Last 4 weeks (weekly granularity)
result = ws.query("Login", last=4, unit="week")
# Last 6 months
result = ws.query("Login", last=6, unit="month")
The unit controls both what "last N" means and how data is bucketed on the time axis.
Absolute¶
Specify explicit start and end dates:
# Q1 2025
result = ws.query(
"Purchase",
from_date="2025-01-01",
to_date="2025-03-31",
unit="week",
)
# From a date to today
result = ws.query("Login", from_date="2025-01-01")
Dates must be in YYYY-MM-DD format.
Hourly Granularity¶
Use unit="hour" for intraday analysis:
Analysis Modes¶
Rolling Windows¶
Smooth noisy data with a rolling average:
# 7-day rolling average of signups by country
result = ws.query(
"Signup",
math="unique",
group_by="country",
rolling=7,
last=60,
)
Cumulative¶
Show running totals over time:
Note
rolling and cumulative are mutually exclusive.
Result Modes¶
The mode parameter controls result aggregation semantics:
| Mode | Semantics | Use case |
|---|---|---|
"timeseries" (default) |
Per-period values | Trends over time |
"total" |
Single aggregate across the date range | KPI numbers |
"table" |
Tabular detail | Detailed breakdowns |
# Single KPI number: total unique purchasers this month
result = ws.query(
"Purchase",
math="unique",
from_date="2025-03-01",
to_date="2025-03-31",
mode="total",
)
total = result.df["count"].iloc[0]
Mode affects aggregation
mode="total" with math="unique" deduplicates users across the entire date range. mode="timeseries" with math="unique" counts unique users per period (not additive across periods). This is not just a display difference — it changes the numbers.
Working with Results¶
QueryResult¶
query() returns a QueryResult with:
result = ws.query("Login", math="unique", last=7)
# DataFrame (lazy, cached)
result.df # pandas DataFrame
# Raw series data
result.series # {"Login [Unique Users]": {"2025-03-01...": 142, ...}}
# Time range
result.from_date # "2025-03-25T00:00:00-07:00"
result.to_date # "2025-03-31T23:59:59.999000-07:00"
# Metadata
result.computed_at # "2025-03-31T12:00:00.000000+00:00"
result.headers # ["$metric"]
result.meta # {"min_sampling_factor": 1.0, ...}
# Generated bookmark params (for debugging or persistence)
result.params # dict — the full bookmark JSON sent to API
DataFrame Structure¶
For timeseries mode, the DataFrame has columns date, event, count:
result = ws.query(["Signup", "Login"], math="unique")
print(result.df.head())
# date event count
# 0 2025-03-01 Signup [Unique Users] 85
# 1 2025-03-01 Login [Unique Users] 312
# 2 2025-03-02 Signup [Unique Users] 92
For total mode, the DataFrame has columns event, count (no date).
Persisting as a Saved Report¶
The generated bookmark params can be saved as a Mixpanel report:
from mixpanel_data import CreateBookmarkParams
# Run query
result = ws.query("Login", math="dau", group_by="platform", last=90)
# Save as a report using the generated params
ws.create_bookmark(CreateBookmarkParams(
name="DAU by Platform (90d)",
bookmark_type="insights",
params=result.params,
))
Debugging¶
Inspect result.params to see the exact bookmark JSON sent to the API. This is useful for:
- Understanding what was actually queried
- Comparing with Mixpanel web UI bookmark params
- Diagnosing unexpected results
import json
result = ws.query("Login", math="unique", group_by="platform")
print(json.dumps(result.params, indent=2))
Validation¶
query() validates all parameter combinations before making an API call and raises ValueError with descriptive messages:
| Rule | Error message |
|---|---|
| Property math without property | math='average' requires math_property to be set |
| Property set with counting math | math_property is only valid with property-based math types |
| Per-user with DAU/WAU/MAU | per_user is incompatible with math='dau' |
| Formula with < 2 events | formula requires at least 2 events |
| Rolling + cumulative | rolling and cumulative are mutually exclusive |
to_date without from_date |
to_date requires from_date |
| Invalid date format | from_date must be YYYY-MM-DD format |
| Invalid bucket config | bucket_min/bucket_max require bucket_size |
Complete Examples¶
Revenue Dashboard Metrics¶
import mixpanel_data as mp
from mixpanel_data import Metric, Filter, GroupBy
ws = mp.Workspace()
# Total revenue by country this quarter
revenue = ws.query(
"Purchase",
math="total",
math_property="amount",
group_by="country",
from_date="2025-01-01",
to_date="2025-03-31",
unit="month",
)
# Revenue distribution by bucket
distribution = ws.query(
"Purchase",
group_by=GroupBy(
"amount",
property_type="number",
bucket_size=25,
bucket_min=0,
bucket_max=500,
),
last=30,
)
# Conversion rate with per-metric filters
conversion = ws.query(
[
Metric("Purchase", math="unique"),
Metric(
"Purchase",
math="unique",
filters=[Filter.equals("plan", "premium")],
),
],
formula="(B / A) * 100",
formula_label="Premium %",
group_by="platform",
unit="week",
)
User Engagement Analysis¶
# 7-day rolling average of DAU by platform
engagement = ws.query(
"Login",
math="dau",
group_by="platform",
rolling=7,
last=90,
)
# Average sessions per user per week
sessions = ws.query(
"Session Start",
math="total",
per_user="average",
unit="week",
last=12,
)
# WAU trend for premium users
wau = ws.query(
"Login",
math="wau",
where=Filter.is_true("is_premium"),
last=6,
unit="month",
)
Generating Params Without Querying¶
Use build_params() to generate bookmark params without making an API call — useful for debugging, inspecting the generated JSON, or saving queries as reports:
# Same arguments as query(), returns dict instead of QueryResult
params = ws.build_params(
"Login",
math="dau",
group_by="platform",
where=Filter.in_the_last("created", 30, "day"),
last=90,
)
import json
print(json.dumps(params, indent=2)) # inspect the generated bookmark JSON
# Save as a report directly from params
ws.create_bookmark(CreateBookmarkParams(
name="DAU by Platform (90d)",
bookmark_type="insights",
params=params,
))
What's Next¶
query() is the foundation for a family of typed query methods. Each follows the same pattern — typed Python arguments generating the correct bookmark params:
query_funnel()— Ad-hoc funnel conversion analysis with typed step definitions, exclusions, and conversion windowsquery_retention()— Ad-hoc retention curves with event pairs, custom buckets, and alignment modesquery_flow()— Ad-hoc flow path analysis with step definitions, direction controls, and visualization modes- Cohort-scoped queries — Filter, break down, or track cohort membership across all engines (see below)
Cohort-Scoped Queries¶
Scope any query to a user segment — filter by cohort membership, break down by cohort, or track cohort size as a metric. Use saved cohort IDs or define cohorts inline with CohortDefinition.
Cohort Filters¶
Restrict queries to users in (or not in) a cohort using Filter.in_cohort() and Filter.not_in_cohort():
from mixpanel_data import Filter, CohortCriteria, CohortDefinition
# Saved cohort
result = ws.query("Purchase", where=Filter.in_cohort(123, "Power Users"))
# Inline cohort — define the segment right where you use it
power_users = CohortDefinition(
CohortCriteria.did_event("Purchase", at_least=3, within_days=30)
)
result = ws.query("Login", where=Filter.in_cohort(power_users, name="Power Users"))
# Exclude a cohort
result = ws.query("Purchase", where=Filter.not_in_cohort(789, "Bots"))
# Combine with property filters
result = ws.query(
"Purchase",
where=[Filter.in_cohort(power_users, name="PU"), Filter.equals("platform", "iOS")],
)
Cohort filters work with all four query methods: query(), query_funnel(), query_retention(), and query_flow().
Cohort Breakdowns¶
Segment results by cohort membership using CohortBreakdown in the group_by= parameter:
from mixpanel_data import CohortBreakdown
# Compare cohort vs. everyone else
result = ws.query(
"Purchase",
group_by=CohortBreakdown(123, "Power Users"),
)
# Result segments: "Power Users" and "Not In Power Users"
# Inline cohort breakdown
result = ws.query(
"Purchase",
group_by=CohortBreakdown(power_users, name="Power Users"),
)
# Only the cohort segment (no "Not In" group)
result = ws.query(
"Purchase",
group_by=CohortBreakdown(123, "PU", include_negated=False),
)
# Mix with property breakdowns
result = ws.query(
"Purchase",
group_by=[CohortBreakdown(123, "Power Users"), "platform"],
)
Cohort breakdowns work with query(), query_funnel(), and query_retention() (not flows).
Cohort Metrics¶
Track cohort size over time as a metric — insights only:
from mixpanel_data import CohortMetric, Metric
# Track cohort growth
result = ws.query(CohortMetric(123, "Power Users"), last=90, unit="week")
# What % of active users are power users?
result = ws.query(
[Metric("Login", math="unique"), CohortMetric(123, "Power Users")],
formula="(B / A) * 100",
formula_label="Power User %",
)
CohortMetric is insights-only — it cannot be used with query_funnel(), query_retention(), or query_flow().
Engine Compatibility¶
| Capability | query() |
query_funnel() |
query_retention() |
query_flow() |
|---|---|---|---|---|
Cohort Filters (where=) |
✓ | ✓ | ✓ | ✓ |
Cohort Breakdowns (group_by=) |
✓ | ✓ | ✓ | — |
Cohort Metrics (events=) |
✓ | — | — | — |
Custom Properties in Queries¶
Use saved custom properties or define computed properties inline — in breakdowns, filters, and metric measurement. Custom properties work everywhere a plain string property name does.
To create and manage custom properties in Mixpanel, see Data Governance — Custom Properties.
Referencing a Saved Custom Property¶
Use CustomPropertyRef to reference a custom property that already exists in your Mixpanel project by its numeric ID:
from mixpanel_data import CustomPropertyRef, GroupBy, Filter, Metric
ref = CustomPropertyRef(42)
# Breakdown by saved custom property
result = ws.query("Purchase", group_by=GroupBy(property=ref, property_type="number"))
# Filter by saved custom property
result = ws.query("Purchase", where=Filter.greater_than(property=ref, value=100))
# Aggregate a saved custom property
result = ws.query(Metric("Purchase", math="average", property=ref))
Find custom property IDs with ws.list_custom_properties() or mp custom-properties list.
Inline Custom Properties¶
Use InlineCustomProperty to define a computed property at query time — no need to save it to your project first. Formulas reference raw properties through single-letter variables (A–Z), each mapped to a PropertyInput:
from mixpanel_data import InlineCustomProperty, PropertyInput
# Full constructor — explicit control over types
revenue = InlineCustomProperty(
formula="A * B",
inputs={
"A": PropertyInput("price", type="number"),
"B": PropertyInput("quantity", type="number"),
},
property_type="number",
)
For the common case of numeric formulas over event properties, use the numeric() convenience constructor:
# Shorthand — auto-creates numeric PropertyInput objects
revenue = InlineCustomProperty.numeric("A * B", A="price", B="quantity")
Both forms produce identical results. Use the full constructor when you need non-numeric types or user-profile properties (resource_type="user").
Custom Property Breakdowns¶
Pass a custom property to GroupBy.property for breakdowns. Numeric bucketing works the same as with regular properties:
from mixpanel_data import GroupBy, CustomPropertyRef, InlineCustomProperty
# Saved custom property with numeric buckets
result = ws.query(
"Purchase",
group_by=GroupBy(
property=CustomPropertyRef(42),
property_type="number",
bucket_size=50,
),
)
# Inline computed property
result = ws.query(
"Purchase",
group_by=GroupBy(
property=InlineCustomProperty.numeric("A * B", A="price", B="quantity"),
property_type="number",
bucket_size=100,
bucket_min=0,
bucket_max=1000,
),
)
# Mix with regular property breakdowns
result = ws.query(
"Purchase",
group_by=["country", GroupBy(property=CustomPropertyRef(42), property_type="number")],
)
Custom Property Filters¶
All 18 Filter factory methods accept custom properties in the property parameter:
from mixpanel_data import Filter, CustomPropertyRef, InlineCustomProperty
# Saved custom property
result = ws.query(
"Purchase",
where=Filter.greater_than(property=CustomPropertyRef(42), value=100),
)
# Inline computed property
result = ws.query(
"Purchase",
where=Filter.between(
property=InlineCustomProperty.numeric("A * B", A="price", B="quantity"),
value=[100, 1000],
),
)
# Combine with regular filters
result = ws.query(
"Purchase",
where=[
Filter.equals("country", "US"),
Filter.greater_than(property=CustomPropertyRef(42), value=50),
],
)
Custom Property Measurement¶
Aggregate a custom property as the metric value using Metric(property=...):
from mixpanel_data import Metric, CustomPropertyRef, InlineCustomProperty
# Average of a saved custom property
result = ws.query(
Metric("Purchase", math="average", property=CustomPropertyRef(42)),
)
# Sum of an inline computed property
result = ws.query(
Metric("Purchase", math="total", property=InlineCustomProperty.numeric("A * B", A="price", B="quantity")),
)
# Per-metric custom properties in multi-metric queries
result = ws.query([
Metric("Purchase", math="total", property=InlineCustomProperty.numeric("A * B", A="price", B="quantity")),
Metric("Purchase", math="unique"),
])
Use Metric(property=...), not math_property=
The top-level math_property parameter only accepts plain string property names. To use a custom property for measurement, wrap the event in a Metric object and set property= on it.
Engine Compatibility¶
| Capability | query() |
query_funnel() |
query_retention() |
query_flow() |
|---|---|---|---|---|
CP Breakdowns (group_by=) |
✓ | ✓ | ✓ | — |
CP Filters (where=) |
✓ | ⚠ | ⚠ | — |
CP Measurement (Metric.property=) |
✓ | — | — | — |
⚠ = Supported, but a known Mixpanel server bug may cause errors when custom property filters are used in funnel and retention global where=. Custom property breakdowns and measurement work reliably in those engines.
query_flow() does not support custom properties in any position (Mixpanel limitation).
Custom Property Validation¶
Custom properties are validated before any API call. Invalid configurations raise BookmarkValidationError:
| Rule | Error code | Error message |
|---|---|---|
| ID must be positive integer | CP1_INVALID_ID |
custom property ID must be a positive integer (got {id}) |
| Formula must be non-empty | CP2_EMPTY_FORMULA |
inline custom property formula must be non-empty |
| At least one input required | CP3_EMPTY_INPUTS |
inline custom property must have at least one input |
| Input keys must be single A–Z | CP4_INVALID_INPUT_KEY |
input keys must be single uppercase letters (A-Z), got {key!r} |
| Formula max 20,000 chars | CP5_FORMULA_TOO_LONG |
formula exceeds maximum length of 20,000 characters (got {len}) |
| Input property name non-empty | CP6_EMPTY_INPUT_NAME |
input {key!r} has an empty property name |
from mixpanel_data import BookmarkValidationError, CustomPropertyRef
try:
ws.query("Purchase", group_by=GroupBy(property=CustomPropertyRef(0), property_type="number"))
except BookmarkValidationError as e:
for error in e.errors:
print(f"[{error.code}] {error.path}: {error.message}")
# [CP1_INVALID_ID] group_by[0].property: custom property ID must be a positive integer (got 0)
Next Steps¶
- Funnel Queries — Typed funnel conversion analysis
- Retention Queries — Typed retention analysis with event pairs and custom buckets
- Flow Queries — Typed flow path analysis with steps, directions, and graph output
- Live Analytics — Legacy query methods (segmentation, funnels, retention)
- Data Discovery — Explore events and properties before querying
- Data Governance — Custom Properties — Create and manage custom properties
- API Reference — Workspace — Full method signatures
- API Reference — Types — Metric, Filter, GroupBy, CustomPropertyRef, InlineCustomProperty, QueryResult details