Exot Trace
Exot Trace is a small, performant library simplifying application tracing. Utilizing AsyncLocalStorage, this library simplifies tracing function calls by integrating the trace() function directly into your codebase without the need to manage context manually.
Usage
import { Tracer } from '@exotjs/trace';
const tracer = new Tracer();
const { trace } = tracer;
tracer.on('endSpan', (span) => {
if (!span.parent) {
// Print only if it's a span without a parent
console.log(JSON.stringify(span, null, ' '));
}
});
async function getUser(id: string) {
// This trace will be automatically grouped with router:getUser
return trace('prisma:getUser', () => prisma.users.findUnique({
where: {
id,
},
}));
}
app.get('/user/:id', (req) => {
// Use the `trace()` function in your code:
trace('router:getUser', async () => {
return {
user: await getUser(req.params.id),
};
});
});After execution, the console will print the following traces:
{
"attributes": {
"http:method": "GET",
"http:path": "/hello-world",
},
"children": [
{
"attributes": {},
"children": [],
"duration": 6.110107421875,
"events": [],
"name": "prisma:getUser",
"start": 1707548525703.3647
}
],
"duration": 6.533447265625,
"events": [{
"attributes": {
"level": "debug"
},
"text": "log text...",
"time": 1707548525703.3647
}],
"name": "router:getUser",
"start": 1707548525703.1785,
"uuid": "12652d90-c01e-48c7-ae2a-7984033b515e"
}How It Works
The trace(name, fn) function executes the fn function and returns its return value. The start time of the execution and the duration are automatically tracked and stored into a âspanâ.
Nested trace() calls work out-of-the-box without the need to pass around any context or parent span, thanks to Nodeâs AsyncLocalStorage from the node:async_hooks module. This feature allows you to track nested executions and construct a tree of trace spans.
To store traces in OpenTelemetry or a database, use the createSpan and endSpan events.
Performance
trace()
- baseline (no tracing)................... 6,365,908 ops/s ±1.20%
- tracing inactive........................ 6,426,187 ops/s ±0.72%
- tracing active.......................... 1,037,015 ops/s ±0.95%startSpan() + endSpan()
- baseline (no tracing)................... 6,400,684 ops/s ±1.07%
- tracing inactive........................ 6,620,705 ops/s ±1.54%
- tracing active.......................... 1,396,103 ops/s ±0.81%Using the trace() function in your code with tracing enabled incurs a significant performance penalty (approximately ~85% according to the benchmark).
Itâs important to note, that this 85% drop compares to the raw function call of an empty function (which doesnât do anything) and it doesnât mean that youâll encounter the same drop relative to the real-world code. The performance of 1M ops/s is considered very good and it means you can trace at least 1 million function calls a second.
When tracing is deactivated using the active property, the penalty is negligible. Thus, itâs acceptable to keep trace() functions in your production code and enable tracing only when needed.
See /benchmarks folder.
Compatibility
This library is meant to be used only server-side and is compatible with the latest versions of Node.js, Bun and Deno.
- Node.js 16+
- Bun 1+
- Deno 1+
Deno
When using with Deno, execute your application with --allow-hrtime to allow high-precision time tracking.
Using trace()
The trace() function executes the fn function and returns its return value. The fn function receives one argument, the context.
Parameters:
name: string(required) Descriptive trace name.fn: (ctx: Context) => any(required) The function to be traced. Can be synchronous or asynchronous.options?: SpanOptionsSee below.
Returns the return value of the fn function.
Using startSpan() and endSpan()
An alternative to the trace() function is to use functions startSpan() and endSpan():
const span = tracer.startSpan('myspan');
// Your code here...
tracer.endSpan(span);
console.log('Duration:', span.duration);These functions do not automatically carry context; to nest spans together, you have to provide the parent span using the options parameter.
startSpan(name, options?)
Parameters:
name: string(required) Descriptive trace name.options?: SpanOptionsSee below.
endSpan(span)
Parameters:
span: TraceSpan(required) Span to end.
Context
trace('mytrace', (ctx) => {
// add custom attributes to the current span
ctx.addAttribute('custom-attribute', 'some value');
// add custom events to the current span
ctx.addEvent('some-event', {
attr1: 'abc',
});
// terminate the current span
ctx.end();
})ctx.name: string
The name of the root span.
ctx.addAttribute(name, value)
Adds a custom attribute to the current span.
ctx.addEvent(text, attributes?)
Adds a custom event with optional attributes to the current span.
ctx.end()
Terminates the current span.
ctx.setStatus(status, attributes?)
Sets the status (ok or error) of the current span.
Span
The TraceSpan has the following structure:
interface TraceSpan {
attributes: Record<string, unknown>;
children: TraceSpan[];
duration: number;
events: {
attributes?: Record<string, unknown>;
time: number;
text: string;
}[];
name: string;
parent?: TraceSpan;
start: number;
status?: TraceSpanStatus;
uuid?: string;
}
enum TraceSpanStatus {
ERROR = 'error',
OK = 'ok',
}The uuid property is set only for the âroot spansâ (which donât a parent span) and you can use these UUIDs (v4) as âtrace IDâ.
SpanOptions
interface SpanOptions {
attributes?: Attributes;
parent?: TraceSpan;
}The trace() function also accepts onEnd function, which gets called once the snap has ended:
interface TraceOptions extends SpanOptions {
onEnd?: (ctx: TraceContext) => void;
}Events
import { Tracer } from '@exotjs/trace';
const tracer = new Tracer();
tracer.on('startSpan', (span) => {
// your implementation here...
});addAttribute
Triggered when a new attribute has been assigned to a span.
Arguments:
span: TraceSpanThe span instance.name: stringThe name of the attribute.value: unknownThe value of the attribute.
addEvent
Triggered when a new event has been added to a span.
Arguments:
span: TraceSpanThe span instance.text: stringThe text of the event.attributes: Record<string, unknown>Optional attributes of the event.
startSpan
Triggered when a new span has started.
Arguments:
span: TraceSpanThe span instance.
endSpan
Triggered when a span has ended.
Arguments:
span: TraceSpanThe span instance.
setStatus
Triggered when the spanâs status is set.
Arguments:
span: TraceSpanThe span instance.status: TraceSpanStatusThe status of the span.attributes: Record<string, unknown>Optional attributes of the event.
License
MIT