Log intermediate steps
To help you debug and deep dive into your LLM apps logs, you can set up tracing using the phospho
library.
This traces every intermediate steps of your LLM app pipeline, from the input text to the output text.
Setup
Install phospho
Info
This feature is currently only available for Python. NodeJS version coming soon!
Make sure you have the phospho
module installed:
Install OpenTelemetry instrumentations
phospho leverages OpenTelemetry instrumentations to trace your LLM app pipeline. To trace a library, you need to install the corresponding instrumentation.
For example, here is how to trace OpenAI and Mistral API calls:
#Â This will trace OpenAI API calls
pip install opentelemetry-instrumentation-openai
#Â This will trace MistralAI API calls
pip install opentelemetry-instrumentation-mistralai
Refer to this list of available instrumentations to find the one that fits your needs.
Initialize phospho
Initialize phospho with phospho.init()
and enable tracing with tracing=True
:
Automatic tracing
All calls to the installed instrumentations are traced.
For example, when you do phospho.log
, the OpenAI API calls will be linked to this log.
import phospho
phospho.init(tracing=True)
#Â This is your LLM app code
openai_client = OpenAI()
color = openai_client.chat.completions.create(
messages=[{"role": "user", "content": "Say a color"}],
model="gpt-4o-mini"
)
animal = openai_client.chat.completions.create(
messages=[{"role": "user", "content": "Say an animal"}],
model="gpt-4o-mini",
)
#Â This is how you log to phospho
#Â All the API calls made by the OpenAI client will me linked to this log
phospho.log(
input="Give me a color and an animal",
output=f"Color: {color}, Animal: {animal}",
)
You can view intermediate steps in the Phospho dashboard when reading a message transcript.
In the automatic tracing mode, the link between API calls and logs is done using the timestamps. If you want more control, you can use the context tracing or manual tracing.
Context tracing
To have more control over which instrumentations calls are linked to which logs, define a context using the phospho.tracer()
context block or @phospho.trace()
decorator syntax.
Context block
This links all calls to the instrumentations made inside the context block to the phospho log. For example, this will link the OpenAI API call to the log:
with phospho.tracer():
messages = [{"role": "user", "content": "Say good bye"}]
openai_client.chat.completions.create(
messages=messages,
model="gpt-4o-mini",
max_tokens=1,
)
phospho.log(input="Say good bye", output=response)
To add session_id
, task_id
and metadata
, pass them as arguments to the context block:
with phospho.tracer(
task_id="some_id",
session_id="my_session_id",
metadata={"user_id": "bob"}
):
messages = [{"role": "user", "content": "Say good bye"}]
openai_client.chat.completions.create(
messages=messages,
model="gpt-4o-mini",
max_tokens=1,
)
phospho.log(input="Say good bye", output=response)
Decorator syntax
This works the same way as the context block.
@phospho.trace()
def my_function():
messages = [{"role": "user", "content": "Say good bye"}]
openai_client.chat.completions.create(
messages=messages,
model="gpt-4o-mini",
max_tokens=1,
)
phospho.log(input="Say good bye", output=response)
my_function()
Note
The context is phospho.tracer
, while the decorator is phospho.trace
, without the r
.
To add session_id
, task_id
and metadata
, pass them as arguments to the decorator:
@phospho.trace(
task_id="some_id",
session_id="my_session_id",
metadata={"user_id": "bob"}
)
def my_function():
messages = [{"role": "user", "content": "Say good bye"}]
openai_client.chat.completions.create(
messages=messages,
model="gpt-4o-mini",
max_tokens=1,
)
phospho.log(input="Say good bye", output=response)
Manual tracing
Pass intermediate steps as a steps
parameter to phospho.log
to trace your pipeline:
phospho.log(
input="Give me a color and an animal",
output=f"Color: {color}, Animal: {animal}",
steps=[
{"name": "OpenAI API call", "input": "Say a color", "output": color},
{"name": "OpenAI API call", "input": "Say an animal", "output": animal},
]
)
This is useful to trace custom modules, which don't have an Opentelemetry instrumentation available. For example, document retrieval, data augmentation, etc.