Instrument: TS
While Phoenix is heavily a Python-based Observability and Evaluation framework, it supports other languages like TypeScript / JavaScript
Phoenix is written and maintained in Python to make it natively runnable in Python notebooks. However, it can be stood up as a trace collector so that your LLM traces from your NodeJS application (e.g., LlamaIndex.TS, Langchain.js) can be collected. The traces collected by Phoenix can then be downloaded to a Jupyter notebook and used to run evaluations (e.g., LLM Evals, Ragas).
Getting Started
Instrumentation is the act of adding observability code to an app yourself.
If you’re instrumenting an app, you need to use the OpenTelemetry SDK for your language. You’ll then use the SDK to initialize OpenTelemetry and the API to instrument your code. This will emit telemetry from your app, and any library you installed that also comes with instrumentation.
Phoenix natively supports automatic instrumentation provided by OpenInference. For more details on OpenInfernce, checkout the project
instrumentation setup
Dependencies
Install OpenTelemetry API packages:
Install OpenInference instrumentation packages. Below is an example of adding instrumentation for OpenAI as well as the semantic conventions for OpenInference.
Initialize the SDK
If you instrument a Node.js application install the OpenTelemetry SDK for Node.js:
Before any other module in your application is loaded, you must initialize the SDK. If you fail to initialize the SDK or initialize it too late, no-op implementations will be provided to any library that acquires a tracer or meter from the API.
For debugging and local development purposes, the following example exports telemetry to the console. After you have finished setting up manual instrumentation, you need to configure an appropriate exporter to export the app’s telemetry data to to Phoenix (e.g. .
The example also sets up the mandatory SDK default attribute service.name
, which holds the logical name of the service, and the optional (but highly encouraged!) attribute service.version
, which holds the version of the service API or implementation.
Alternative methods exist for setting up resource attributes. For more information, see Resources.
To verify your code, run the app by requiring the library:
This basic setup has no effect on your app yet. You need to add code for traces, metrics, and/or logs.
You can register instrumentation libraries with the OpenTelemetry SDK for Node.js in order to generate telemetry data for your dependencies. For more information, see Libraries.
Traces
Initialize Tracing
To enable tracing in your app, you’ll need to have an initialized TracerProvider
that will let you create a Tracer
.
If a TracerProvider
is not created, the OpenTelemetry APIs for tracing will use a no-op implementation and fail to generate data. As explained next, modify the instrumentation.ts
(or instrumentation.js
) file to include all the SDK initialization code in Node and the browser.
Node.js
If you followed the instructions to initialize the SDK above, you have a TracerProvider
setup for you already. You can continue with acquiring a tracer.
First, ensure you’ve got the right packages:
Next, update instrumentation.ts
(or instrumentation.js
) to contain all the SDK initialization code in it:
You’ll need to bundle this file with your web application to be able to use tracing throughout the rest of your web application.
This will have no effect on your app yet: you need to create spans to have telemetry emitted by your app.
Picking the right span processor
By default, the Node SDK uses the BatchSpanProcessor
, and this span processor is also chosen in the Web SDK example. The BatchSpanProcessor
processes spans in batches before they are exported. This is usually the right processor to use for an application.
In contrast, the SimpleSpanProcessor
processes spans as they are created. This means that if you create 5 spans, each will be processed and exported before the next span is created in code. This can be helpful in scenarios where you do not want to risk losing a batch, or if you’re experimenting with OpenTelemetry in development. However, it also comes with potentially significant overhead, especially if spans are being exported over a network - each time a call to create a span is made, it would be processed and sent over a network before your app’s execution could continue.
In most cases, stick with BatchSpanProcessor
over SimpleSpanProcessor
.
Acquiring a tracer
Anywhere in your application where you write manual tracing code should call getTracer
to acquire a tracer. For example:
The values of instrumentation-scope-name
and instrumentation-scope-version
should uniquely identify the Instrumentation Scope, such as the package, module or class name. While the name is required, the version is still recommended despite being optional.
It’s generally recommended to call getTracer
in your app when you need it rather than exporting the tracer
instance to the rest of your app. This helps avoid trickier application load issues when other required dependencies are involved.
In the case of the example app, there are two places where a tracer may be acquired with an appropriate Instrumentation Scope:
First, in the application file app.ts
(or app.js
):
Create spans
Now that you have tracers initialized, you can create spans.
The API of OpenTelemetry JavaScript exposes two methods that allow you to create spans:
tracer.startSpan
: Starts a new span without setting it on context.tracer.startActiveSpan
: Starts a new span and calls the given callback function passing it the created span as first argument. The new span gets set in context and this context is activated for the duration of the function call.
In most cases you want to use the latter (tracer.startActiveSpan
), as it takes care of setting the span and its context active.
The code below illustrates how to create an active span.
The above instrumented code can now be pasted in the /chat
handler. You should now be able to see spans emitted from your app.
Start your app as follows, and then send it requests by visiting http://localhost:8080/chat?message="how long is a pencil"
with your browser or curl
.
After a while, you should see the spans printed in the console by the ConsoleSpanExporter
, something like this:
Get the current span
Sometimes it’s helpful to do something with the current/active span at a particular point in program execution.
Get a span from context
It can also be helpful to get the span from a given context that isn’t necessarily the active span.
Attributes
Attributes let you attach key/value pairs to a Span
so it carries more information about the current operation that it’s tracking. For OpenInference related attributes, use the @arizeai/openinference-semantic-conventions
keys. However you are free to add any attributes you'd like!
You can also add attributes to a span as it’s created:
Semantic Attributes
There are semantic conventions for spans representing operations in well-known protocols like HTTP or database calls. OpenInference also publishes it's own set of semantic conventions related to LLM applications. Semantic conventions for these spans are defined in the specification under OpenInference. In the simple example of this guide the source code attributes can be used.
First add both semantic conventions as a dependency to your application:
Add the following to the top of your application file:
Finally, you can update your file to include semantic attributes:
Span events
A Span Event is a human-readable message on an Span
that represents a discrete event with no duration that can be tracked by a single timestamp. You can think of it like a primitive log.
You can also create Span Events with additional Attributes
While Phoenix captures these, they are currently not displayed in the UI. Contact us if you would like to support!
Span Status
A Status can be set on a Span, typically used to specify that a Span has not completed successfully - Error
. By default, all spans are Unset
, which means a span completed without error. The Ok
status is reserved for when you need to explicitly mark a span as successful rather than stick with the default of Unset
(i.e., “without error”).
The status can be set at any time before the span is finished.
Recording exceptions
It can be a good idea to record exceptions when they happen. It’s recommended to do this in conjunction with setting span status.
Using sdk-trace-base
and manually propagating span context
sdk-trace-base
and manually propagating span contextIn some cases, you may not be able to use either the Node.js SDK nor the Web SDK. The biggest difference, aside from initialization code, is that you’ll have to manually set spans as active in the current context to be able to create nested spans.
Initializing tracing with sdk-trace-base
Initializing tracing is similar to how you’d do it with Node.js or the Web SDK.
Like the other examples in this document, this exports a tracer you can use throughout the app.
Creating nested spans with sdk-trace-base
To create nested spans, you need to set whatever the currently-created span is as the active span in the current context. Don’t bother using startActiveSpan
because it won’t do this for you.
All other APIs behave the same when you use sdk-trace-base
compared with the Node.js SDKs.
Last updated