Metrics & the /metrics endpoint
Metrics are the third observability pillar alongside the structured logger and the OpenTelemetry-compatible tracer. As of 0.37.0 DaloyJS ships a dependency-free Prometheus / OpenMetrics stack: a metrics registry (counters, gauges, histograms), RED (Rate / Errors / Duration) instrumentation for every route, and an opt-in, auth-guarded /metrics scrape route that inherits the same hardened posture as app.healthcheck().
Everything is built on Web-standard primitives (plus optional process.* gauges guarded for non-Node runtimes), so it runs unchanged on Node, Bun, Deno, Cloudflare Workers, and Vercel Edge.
Quick start
Call app.metrics() before registering the routes you want measured. It installs RED instrumentation and registers the scrape route in one step.
Because the instrumentation is installed as a group hook, it only wraps routes registered after the app.metrics() call — the same ordering rule as any app.use(...) middleware.
What gets exported
Out of the box, the scrape route exposes:
daloy_http_requests_total{method,route,status}— a request counter (rate; the error rate is the subset with a4xx/5xxstatus).daloy_http_request_duration_seconds{method,route}— a latency histogram with conventional Prometheus buckets.daloy_http_requests_in_flight— a gauge of concurrently-handled requests.- process gauges (
daloy_process_resident_memory_bytes,daloy_process_heap_used_bytes,daloy_process_uptime_seconds) collected at scrape time on Node-like runtimes.
The route label
High-cardinality labels are the classic way to melt a Prometheus server. By default the route label uses the request pathname, capped at maxRouteCardinality (100) distinct values before further paths collapse to <other>. For templated routes, supply a resolver that returns the route template:
Custom application metrics
Pass your own MetricsRegistry to register business metrics that render alongside the built-in HTTP series.
Use registry.collect(fn) to refresh point-in-time gauges (queue depth, connection-pool size) only when the endpoint is actually scraped, instead of on a timer.
Manual instrumentation
Prefer to wire the pieces yourself? httpMetrics() returns a Hooks bundle you can app.use(...) without the built-in scrape route — render the registry from your own handler.
Security posture
A /metrics endpoint leaks internal route names, latency distributions, request volume, and process memory — so it ships with the same hardened defaults as app.healthcheck():
- Bearer token (
opts.token) compared withtimingSafeEqual. Missing token is a401withWWW-Authenticate; wrong token is a403. - Per-IP rate limit (default
{ limit: 60, windowMs: 60_000 }) returning429withRetry-Afteron overflow. PassrateLimit: falseto disable. - Refuse-to-boot: an unauthenticated scrape endpoint in production throws at registration unless you set a token or explicitly pass
acknowledgeUnauthenticated: true. - Cardinality cap: every metric is bounded by
maxSeries(default 5000); overflowing label combinations are dropped and counted indaloy_metrics_series_dropped_total, a memory-exhaustion defense. - Exposition-injection defense: metric and label names are validated against the Prometheus grammar at definition time, and label values escape
\\,", and newlines so a hostile value cannot forge extra samples.
In most deployments you should also scope the scrape endpoint to your monitoring network at the ingress/firewall layer in addition to the bearer token.