Do11y is a documentation observability solution from Axiom. It turns documentation usage into machine data. It streams behavioral events like the ones below from your docs site to Axiom in real time:
- Page views
- Scroll depth
- Link clicks
- Search queries
- Code-block copies
- Section reading time
- Tab switches
- Table of contents (TOC) usage
- Feedback widget usage
- Expand/collapse interactions
Do11y is built for humans and machines alike. It emits observability data designed to be easy to use for human users, while also being easy to query and analyze for machines.
Do11y is agent-native: in an era where AI assistants and autonomous agents increasingly read and cite documentation alongside human users, Do11y detects AI platform referrers (ChatGPT, Perplexity, Claude, Gemini, and others) so you can understand how agents and humans engage with your content differently.
The runtime artifact is a single dependency-free JavaScript file. The source is TypeScript (src/do11y.ts). rolldown produces the built output.
Do11y collects anonymous usage data:
- No cookies. Do11y uses
sessionStorage, which the browser clears when it closes. - No personal identifiable information (PII).
- No device fingerprinting.
- No cross-site tracking.
You don't need a GDPR consent banner for using Do11y.
Do11y supports the latest versions of the following frameworks:
- Mintlify
- Docusaurus
- Nextra
- MkDocs Material
- VitePress
For other frameworks, use manual setup.
- Create an Axiom account.
- Create a dataset in Axiom to store observability data for your documentation site.
- Create an API token in Axiom with ingest-only permissions scoped to the dataset.
-
Download the latest release from GitHub and extract the
do11y-<version>.zipfile. -
Copy
dist/do11y.min.jsandexamples/do11y-config.example.jsto the same folder in your documentation project (for example,scripts/). Alphabetical ordering ensures the config loads first. -
Rename
do11y-config.example.jstodo11y-config.js. -
In
do11y-config.js, replace the placeholder values with your Axiom credentials.window.Do11yConfig = { axiomHost: 'AXIOM_DOMAIN', axiomToken: 'API_TOKEN', axiomDataset: 'DATASET_NAME', framework: 'mintlify', };
-
Optional: Set up the automatic sync to your docs repo to keep your copy of
do11y.min.jsup to date.
Add the following to the headTags and scripts fields in docusaurus.config.js if you use JavaScript, or docusaurus.config.ts if you use TypeScript:
headTags: [
{ tagName: 'meta', attributes: { name: 'axiom-do11y-domain', content: 'AXIOM_DOMAIN' } },
{ tagName: 'meta', attributes: { name: 'axiom-do11y-token', content: 'API_TOKEN' } },
{ tagName: 'meta', attributes: { name: 'axiom-do11y-dataset', content: 'DATASET_NAME' } },
{ tagName: 'meta', attributes: { name: 'axiom-do11y-framework', content: 'docusaurus' } },
],
scripts: [{ src: 'https://cdn.jsdelivr.net/npm/@axiomhq/do11y@latest/dist/do11y.min.js', defer: true }],Add the following to the <Head> component in pages/_app.jsx (or _app.tsx) if you use the Pages Router, or app/layout.jsx (or app/layout.tsx) if you use the App Router:
<Head>
<meta name="axiom-do11y-domain" content="AXIOM_DOMAIN" />
<meta name="axiom-do11y-token" content="API_TOKEN" />
<meta name="axiom-do11y-dataset" content="DATASET_NAME" />
<meta name="axiom-do11y-framework" content="nextra" />
<script src="https://cdn.jsdelivr.net/npm/@axiomhq/do11y@latest/dist/do11y.min.js" defer />
</Head>Add the following to the head field in .vitepress/config.js (or .vitepress/config.ts):
head: [
['meta', { name: 'axiom-do11y-domain', content: 'AXIOM_DOMAIN' }],
['meta', { name: 'axiom-do11y-token', content: 'API_TOKEN' }],
['meta', { name: 'axiom-do11y-dataset', content: 'DATASET_NAME' }],
['meta', { name: 'axiom-do11y-framework', content: 'vitepress' }],
['script', { src: 'https://cdn.jsdelivr.net/npm/@axiomhq/do11y@latest/dist/do11y.min.js' }],
],Add the following to mkdocs.yml:
theme:
name: material
custom_dir: overrides
extra_javascript:
- https://cdn.jsdelivr.net/npm/@axiomhq/do11y@latest/dist/do11y.min.jsCreate overrides/main.html to inject the meta tags:
{% extends "base.html" %}
{% block extrahead %}
<meta name="axiom-do11y-domain" content="AXIOM_DOMAIN">
<meta name="axiom-do11y-token" content="API_TOKEN">
<meta name="axiom-do11y-dataset" content="DATASET_NAME">
<meta name="axiom-do11y-framework" content="mkdocs-material">
{% endblock %}See the MkDocs Material docs for details on custom theme overrides.
Once you've installed Do11y and information about your documentation usage is flowing into Axiom, you can query the data in Axiom. See QUERIES.md for exampleAPL queries to analyze your documentation, including:
- AI traffic detection and trends
- Traffic sources and entry points
- Page engagement and scroll completion
- Where users get stuck (exit pages, low engagement)
- Navigation patterns and user journeys
- Link and CTA performance
- Code block engagement
For more information, see Query data with Axiom.
An integration dashboard provides a visual overview of your documentation usage. It shows important metrics like the number of page views, scroll depth, link clicks, code-block copies, section reading time, tab switches, TOC usage, feedback widget usage, and expand/collapse interactions. It's automatically created when you add Do11y to your docs site.
To access the integration dashboard:
- In Axiom, click Dashboards.
- In the Integrations section, click the integration dashboard Documentation observability (Do11y) (DATASET_NAME).
Alternatively, access the integration dashboard with the URL https://app.axiom.co/ORG_ID/dashboards/do11y.DATASET_NAME.
Do11y classifies referrer domains to detect traffic from AI platforms such as ChatGPT, Perplexity, Claude, Gemini, Copilot, DeepSeek, and others. Each page_view event includes:
| Field | Values | Description |
|---|---|---|
referrerCategory |
ai, search-engine, social, community, code-host, direct, internal, other, unknown |
High-level traffic source category. |
aiPlatform |
ChatGPT, Perplexity, Claude, Gemini, Copilot, DeepSeek, Meta AI, Grok, Mistral, You.com, Phind, or null |
Specific AI platform when referrerCategory is ai. |
This detection is referrer-based: it checks whether the document.referrer hostname matches a known AI platform. Do11y uses no fingerprinting, user-agent parsing, or additional data collection.
Limitation: Most AI platforms (especially ChatGPT mobile and API-sourced visits) don't pass referrer headers. These visits appear as direct traffic. Referrer-based detection typically captures 20-40% of AI traffic. Detecting the remaining "dark AI" traffic would require fingerprinting techniques that conflict with Do11y's privacy-first design.
See QUERIES.md for APL queries to analyze AI traffic, including per-platform breakdowns, trends, and engagement comparisons.
The selectors work on sites using the standard themes of each supported framework. Sites with heavily customized themes may render page elements differently. If you use a custom theme, check whether you need to set the selectors manually.
CSS selectors reflect each framework's current DOM output and may break when frameworks release major updates that change class names or HTML structure. The test suites (test-live-sites.ts and test-queries.ts) exist specifically to catch this. Run them periodically to verify selectors still match.
Add the script to every page of your docs site. The simplest setup uses meta tags for the required settings:
<meta name="axiom-do11y-domain" content="AXIOM_DOMAIN">
<meta name="axiom-do11y-token" content="API_TOKEN">
<meta name="axiom-do11y-dataset" content="DATASET_NAME">
<meta name="axiom-do11y-framework" content="FRAMEWORK">
<script src="https://cdn.jsdelivr.net/npm/@axiomhq/do11y@latest/dist/do11y.min.js"></script>Replace the meta tag values with your Axiom credentials and docs framework. To pin a specific version, replace latest with a version tag like 1.0.0.
Meta tags only cover the essential settings. To configure any of the advanced options such as scroll thresholds, tracking toggles, or custom selectors, set window.Do11yConfig in an inline script placed before the CDN script tag:
<script>
window.Do11yConfig = {
axiomHost: 'us-east-1.aws.edge.axiom.co',
axiomToken: 'xaat-your-ingest-token',
axiomDataset: 'do11y',
framework: 'vitepress',
scrollThresholds: [25, 50, 75, 100],
trackFeedback: false,
sectionVisibleThreshold: 5,
// Any option from the Configuration table below can be set here
};
</script>
<script src="https://cdn.jsdelivr.net/npm/@axiomhq/do11y@1.0.0/dist/do11y.min.js"></script>When both are present, meta tags take precedence over window.Do11yConfig, which takes precedence over the defaults.
If you can't use a CDN, self-host the script.
-
Download the latest release from GitHub and extract the
do11y-<version>.zipfile. -
Copy
dist/do11y.min.jsandexamples/do11y-config.example.jsto your documentation project (for example,scripts/). -
Rename
do11y-config.example.jstodo11y-config.js. -
In
do11y-config.js, replace the placeholder values with your Axiom credentials.window.Do11yConfig = { axiomHost: 'AXIOM_DOMAIN', axiomToken: 'API_TOKEN', axiomDataset: 'DATASET_NAME', framework: 'FRAMEWORK', };
-
Add both scripts to every page, with the config file loading first:
<script src="/path/to/do11y-config.js"></script> <script src="/path/to/do11y.min.js"></script>
-
Optional: Set up the automatic sync to your docs repo to keep your copy of
do11y.min.jsup to date.
Don't edit do11y.min.js directly. It's a build artifact and updating to a new release overwrites it.
If you self-host do11y.min.js in GitHub repo, the included GitHub Action (examples/sync-do11y-to-docs.yml) keeps your copy up to date automatically.
-
Copy
examples/sync-do11y-to-docs.ymlto.github/workflows/in your docs repo. It runs every Monday and opens a PR whenever a new do11y release is available. -
Create an empty file at
do11y.version. This file is used to track the version ofdo11y.min.js. -
Add the following repository variables in your docs repo under Settings > Secrets and variables > Actions > Variables > New repository variable:
Variable Example Description DO11Y_JS_PATHscripts/do11y.min.jsPath to do11y.min.jsin your docs repo.DO11Y_VER_PATHscripts/do11y.versionPath to a version tracking file in your docs repo. -
Ensure the GitHub Action has permission to push to your docs repo. Go to Settings > Actions > General > Workflow permissions, and turn on Allow GitHub Actions to create and approve pull requests.
You don't need to add any secrets.
All options can be set via window.Do11yConfig (inline script or a separate config file) or via meta tags.
| Option | Default | Description |
|---|---|---|
axiomHost |
'AXIOM_DOMAIN' |
Base domain of the edge deployment where you want to store your data. For more information, see Edge deployments. |
axiomDataset |
'DATASET_NAME' |
Name of the Axiom dataset where you want to store your data. |
axiomToken |
'API_TOKEN' |
Ingest-only API token scoped to the dataset. |
| Option | Default | Description |
|---|---|---|
debug |
false |
Log events to the browser console. |
flushInterval |
5000 |
Milliseconds between batch flushes. |
maxBatchSize |
10 |
Events queued before forcing a flush. |
trackOutboundLinks |
true |
Track clicks on external links. |
trackInternalLinks |
true |
Track clicks on internal links. |
trackScrollDepth |
true |
Track scroll depth thresholds. |
scrollThresholds |
[25, 50, 75, 90] |
Scroll percentages to record. |
trackSectionVisibility |
true |
Track which headings users actually read (via IntersectionObserver). |
sectionVisibleThreshold |
3 |
Minimum seconds a section must be visible before recording. |
trackTabSwitches |
true |
Track code language/framework tab switches. |
trackTocClicks |
true |
Track on-page table of contents clicks. |
trackExpandCollapse |
true |
Track expand/collapse interactions (details, accordions). |
trackFeedback |
true |
Track "Was this helpful?" feedback widget clicks. |
allowedDomains |
['ALLOWED_DOMAINS'] |
Restrict which domains may send data. Set to null to allow any. |
respectDNT |
true |
Honor the browser's Do Not Track setting. |
maxRetries |
2 |
Retry count for failed requests. |
retryDelay |
1000 |
Base delay between retries (doubles each attempt). |
rateLimitMs |
100 |
Minimum gap between events of the same type. |
Set framework to auto-configure CSS selectors for your docs platform:
| Value | Framework |
|---|---|
'mintlify' |
Mintlify (default) |
'docusaurus' |
Docusaurus |
'nextra' |
Nextra |
'mkdocs-material' |
MkDocs Material |
'vitepress' |
VitePress |
'custom' |
Provide your own selectors (see below) |
When framework is set to a supported value, the script automatically uses the correct CSS selectors for search bars, copy buttons, code blocks, navigation, footers, and content areas. Optional: Set the framework via a meta tag:
<meta name="axiom-do11y-framework" content="docusaurus">Set framework: 'custom' and provide any combination of these selectors. Any selector left null falls back to the Mintlify default.
| Selector | What it targets |
|---|---|
searchSelector |
Search trigger elements (input, button). |
copyButtonSelector |
"Copy code" buttons inside code blocks. |
codeBlockSelector |
Code block containers (<pre>, wrappers). |
navigationSelector |
Navigation and sidebar regions. |
footerSelector |
Page footer. |
contentSelector |
Main content area. |
tabContainerSelector |
Tab groups for code language/framework switching. |
tocSelector |
On-page table of contents container. |
feedbackSelector |
"Was this helpful?" feedback widget container. |
| Event | Description | Key fields |
|---|---|---|
page_view |
Fires on every page load or SPA navigation. | referrerDomain, referrerCategory, aiPlatform, isFirstPage, previousPath |
link_click |
Internal, external, anchor, or email link click. | linkType, targetUrl, linkText, linkContext, linkSection, linkIndex |
scroll_depth |
User scrolls past a configured threshold. | threshold, scrollPercent |
page_exit |
Fires on beforeunload. |
totalTimeSeconds, activeTimeSeconds, engagementRatio, maxScrollDepth, referrerCategory, aiPlatform |
search_opened |
User opens the search dialog (click or Cmd/Ctrl+K). | trigger |
code_copied |
User clicks a code block's copy button. | language, codeSection, codeBlockIndex |
section_visible |
A heading stayed visible in the viewport long enough for the user to read it. | heading, headingLevel, visibleSeconds |
tab_switch |
User switches a code language/framework tab. | tabLabel, tabGroup, isDefault |
toc_click |
User clicks an entry in the on-page table of contents. | heading, headingLevel, tocPosition |
feedback |
User clicks a "Was this helpful?" button. | rating |
expand_collapse |
User toggles a <details> element or accordion. |
summary, action, section |
Every event also includes: sessionId, sessionPageCount, path, hash, title, viewportCategory, browserFamily, deviceType, language, and timezoneOffset.
Do11y exposes window.AxiomDo11y for debugging and integration:
AxiomDo11y.getConfig() // Current config (token redacted)
AxiomDo11y.isEnabled() // Whether tracking is active
AxiomDo11y.flush() // Force-send queued events
AxiomDo11y.getQueueSize() // Number of queued events
AxiomDo11y.version // Script versionDo11y doesn't expose cleanup() and debug() on the global object. Exposing cleanup() would allow any third-party script on the page to silently stop tracking. Exposing debug() would allow any script to enable verbose console output that reveals the configured ingest endpoint and queued event data.
The tests folder contains multiple layers of testing.
Runs headless Chromium via Puppeteer against real documentation sites to validate that selectors match elements in production.
cd tests
npm i
npx puppeteer browsers install chrome
npm run test-live-sitesThe test covers the following sites:
| Framework | URL |
|---|---|
| Mintlify | https://axiom.co/docs/query-data/explore |
| Docusaurus | https://docusaurus.io/docs/configuration |
| Nextra | https://nextra.site/docs/getting-started |
| MkDocs Material | https://squidfunk.github.io/mkdocs-material/getting-started/ |
| VitePress | https://vitepress.dev/guide/getting-started |
Validates that all APL queries in QUERIES.md are syntactically correct by executing them against the Axiom API.
cd tests
npm run test-queriesEnd-to-end tests that install each supported framework, inject do11y.js, start a local dev server, drive user interactions via Puppeteer, and then query the Axiom API to verify that events arrived correctly.
cd tests
npm i
npx puppeteer browsers install chromeCopy tests/.env.example to tests/.env and add your credentials:
AXIOM_DOMAIN=us-east-1.aws.edge.axiom.co
AXIOM_TOKEN=xaat-your-ingest-token
AXIOM_DATASET=do11y
The token requires both ingest and query permissions on the target dataset.
Run the full suite:
npm run test-integrationsRun a subset of frameworks:
FRAMEWORKS=mintlify,vitepress npm run test-integrationsSkip dependency installation on repeat runs:
SKIP_INSTALL=1 npm run test-integrationsThe test covers the following frameworks:
| Name | Type | Port | Notes |
|---|---|---|---|
mintlify |
npm (Mintlify CLI) | 4005 | Full framework install |
docusaurus |
npm (Docusaurus 3) | 4001 | Full framework install |
nextra |
npm (Next.js + Nextra 3) | 4002 | Full framework install |
vitepress |
npm (VitePress 1.x) | 4003 | Full framework install |
mkdocs-material |
pip (MkDocs Material) | 4004 | Requires Python. Skips if unavailable. |
The test validates the following events per framework:
| Event | Minimum expected | Notes |
|---|---|---|
page_view |
2 | Start page + guide page |
scroll_depth |
1 | |
link_click |
1 | |
page_exit |
1 | |
expand_collapse |
1 | |
toc_click |
1 | |
search_opened |
0 | |
code_copied |
1 | |
feedback |
0 | |
section_visible |
1 | sectionVisibleThreshold: 1 + 2 s dwell on page load |
-
Run all tests.
-
Bump the version in
package.jsonandsrc/do11y.ts. -
Run the following commands to build the package and run the tests:
npm run build npm run check npm run lint
-
Commit the changes and push to the
mainbranch. -
Tag and release via the GitHub CLI:
git tag v1.1.0 git push origin v1.1.0 gh release create v1.1.0
Alternatively, use the GitHub UI to create a release at https://github.com/axiomhq/do11y/releases/new
-
Publish the package to npm as
@axiomhq/do11y. This requires access to the@axiomhqnpm organization.npm login npm publish --access public npm logout