-
-
Notifications
You must be signed in to change notification settings - Fork 691
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Added an example of how to use Instructor with logfire
- Loading branch information
Showing
9 changed files
with
447 additions
and
0 deletions.
There are no files selected for viewing
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,276 @@ | ||
--- | ||
draft: False | ||
date: 2024-05-01 | ||
slug: instructor-logfire | ||
tags: | ||
- python | ||
- observability | ||
- open-telemetry | ||
authors: | ||
- ivanleomk | ||
- jxnl | ||
--- | ||
|
||
## Introduction | ||
|
||
Logfire is a new observability platform coming from the creators of Pydantic. It hooks directly into the system itself. In this article, we'll show you how to integrate Logfire with Instructor for easy observability using a few different examples. | ||
|
||
We'll first walk you through a simple classification example. We'll then introduce some simple validation using our pre-built `llm_validator` avaliable in Instructor before finally concluding with a vision table example. | ||
|
||
<!-- more --> | ||
|
||
As usual, all of the code that we refer to here is provided in [examples/logfire](https://www.github.com/jxnl/instructor/tree/main/examples/logfire) for you to use in your projects. | ||
|
||
- `classify.py`: Email Classification Example | ||
- `image.py` : GPT4-V Example | ||
- `validate.py` : `llm_validator` example | ||
|
||
??? info "Configure Logfire" | ||
|
||
Before starting this tutorial, make sure that you've registered for a [Logfire](https://logfire.pydantic.dev/) account. You'll also need to create a project to track these logs. | ||
|
||
We'll need to install our dependencies and configure logfire auth before proceeding so simply run the commands below. Logfire will handle the authentication and configuration of your project. | ||
|
||
```bash | ||
pip install logfire openai instructor pydantic pandas tabulate | ||
logfire auth | ||
``` | ||
|
||
## Classification | ||
|
||
Now that we've got Logfire setup, let's see how we can get it to help us track a simple classification job. | ||
|
||
Logfire is dead simple to integrate - all it takes is 2 lines of code and we have it setup. | ||
|
||
```python | ||
from pydantic import BaseModel | ||
from openai import OpenAI | ||
import instructor | ||
import logfire | ||
|
||
|
||
openai_client = OpenAI() | ||
logfire.configure(pydantic_plugin=logfire.PydanticPlugin(record="all")) #(1)! | ||
logfire.instrument_openai(openai_client) #(2)! | ||
client = instructor.from_openai(openai_client) | ||
``` | ||
|
||
1. We add Pydantic logging using `logfire`. Note that depending on your use-case, you can configure what you want to log with Pydantic | ||
2. We use their openai_integration to configure logging for our client before using instructor on it | ||
|
||
In this example, we'll be looking at classifying emails as either spam or not spam. To do so, we can define a simple Pydantic model as seen below. | ||
|
||
```python | ||
import enum | ||
|
||
class Labels(str, enum.Enum): | ||
"""Enumeration for single-label text classification.""" | ||
|
||
SPAM = "spam" | ||
NOT_SPAM = "not_spam" | ||
|
||
|
||
class SinglePrediction(BaseModel): | ||
""" | ||
Class for a single class label prediction. | ||
""" | ||
|
||
class_label: Labels | ||
``` | ||
|
||
We can then use this in a generic instructor function as seen below that simply asks the model to classify text and return it in the form of a `SinglePrediction` Pydantic object. | ||
|
||
Logfire can help us to log this entire function, and what's happening inside it, even down to the model validation level by using their `logfire.instrument` decorator. | ||
|
||
```python | ||
@logfire.instrument("classification", extract_args=True) #(1)! | ||
def classify(data: str) -> SinglePrediction: | ||
"""Perform single-label classification on the input text.""" | ||
return client.chat.completions.create( | ||
model="gpt-3.5-turbo-0613", | ||
response_model=SinglePrediction, | ||
messages=[ | ||
{ | ||
"role": "user", | ||
"content": f"Classify the following text: {data}", | ||
}, | ||
], | ||
) | ||
``` | ||
|
||
1. Logfire allows us to use the `logfire.instrument` decorator and tag a function to a specific name. | ||
|
||
Let's see what happens when we run this against a list of different emails | ||
|
||
```python | ||
emails = [ | ||
"Hello there I'm a Nigerian prince and I want to give you money", | ||
"Meeting with Thomas has been set at Friday next week", | ||
"Here are some weekly product updates from our marketing team", | ||
] | ||
|
||
for email in emails: | ||
classify(email) | ||
``` | ||
|
||
There are a few important things here that the logs immediately give us | ||
|
||
1. The duration that each individual portion of our code took to run | ||
2. The payload that we sent over to OpenAI | ||
3. The exact arguments and results that were passed to each individual portion of our code at each step | ||
|
||
![Logfire Classification](img/classification-logfire.png) | ||
|
||
## LLM Validators | ||
|
||
For our second example, we'll use the inbuilt `llm_validator` that instructor provides out of the box to validate that our statements don't contain unsafe content that we might not want to serve to users. Let's start by defining a simple Pydantic Model that can do so and configure our logfire integration. | ||
|
||
```python | ||
from typing import Annotated | ||
from pydantic import BaseModel, ValidationError | ||
from pydantic.functional_validators import AfterValidator | ||
from instructor import llm_validator | ||
import logfire | ||
import instructor | ||
from openai import OpenAI | ||
|
||
openai_client = OpenAI() | ||
logfire.configure(pydantic_plugin=logfire.PydanticPlugin(record="all")) | ||
logfire.instrument_openai(openai_client) | ||
client = instructor.from_openai(openai_client) | ||
|
||
|
||
class Statement(BaseModel): | ||
message: Annotated[ | ||
str, | ||
AfterValidator( | ||
llm_validator("Don't allow any objectionable content", client=client) | ||
), | ||
] | ||
``` | ||
|
||
We can then test out our new validator with a few sample statements to see how our validator is working in practice. | ||
|
||
```python | ||
messages = [ | ||
"I think we should always treat violence as the best solution", | ||
"There are some great pastries down the road at this bakery I know", | ||
] | ||
|
||
for message in messages: | ||
try: | ||
Statement(message=message) | ||
except ValidationError as e: | ||
print(e) | ||
``` | ||
|
||
With Logfire, we can capture the entirety of the validation proccess. As seen below, we have access to not only the original input data, but also the schema that was being used, the errors that were thrown and even the exact field that threw the error. | ||
|
||
![Logfire Validation](img/validation-logfire.png) | ||
|
||
## Vision Models | ||
|
||
For our last example, let's see how we can use Logfire to extract structured data from an image using GPT-4V with OpenAI. We'll be using a simple bar graph here and using `GPT4V` to extract the data from the image from statista below and convert it into a markdown format. | ||
|
||
![Reference Image](img/statista-image.jpeg) | ||
|
||
What we want is an output of the combined numbers as seen below | ||
|
||
| Country | Total Skier Visits (M) | | ||
| :------------ | ---------------------: | | ||
| United States | 55.5 | | ||
| Austria | 43.6 | | ||
| France | 40.7 | | ||
| Japan | 26.6 | | ||
| Italy | 22.3 | | ||
| Switzerland | 22 | | ||
| Canada | 18.5 | | ||
| China | 17.9 | | ||
| Sweden | 9.2 | | ||
| Germany | 7 | | ||
|
||
This is relatively simple with Pydantic. What we need to do is to define a custom type which will handle the conversion process as seen below | ||
|
||
```python | ||
from pydantic import BaseModel, Field, BeforeValidator, PlainSerializer, InstanceOf, WithJsonSchema | ||
|
||
def md_to_df(data: Any) -> Any: | ||
# Convert markdown to DataFrame | ||
if isinstance(data, str): | ||
return ( | ||
pd.read_csv( | ||
StringIO(data), # Process data | ||
sep="|", | ||
index_col=1, | ||
) | ||
.dropna(axis=1, how="all") | ||
.iloc[1:] | ||
.applymap(lambda x: x.strip()) | ||
) | ||
return data | ||
|
||
|
||
MarkdownDataFrame = Annotated[ | ||
InstanceOf[pd.DataFrame], #(1)! | ||
BeforeValidator(md_to_df), #(2)! | ||
WithJsonSchema( #(3)! | ||
{ | ||
"type": "string", | ||
"description": "The markdown representation of the table, each one should be tidy, do not try to join tables that should be seperate", | ||
} | ||
), | ||
] | ||
``` | ||
|
||
1. We indicate that the type of this type should be a pandas dataframe | ||
2. We run a validation step to ensure that we can convert the input into a valid pandas dataframe and return a new pandas Dataframe for our model to use | ||
3. We then override the type of the schema so that when we pass it to OpenAI, it knows to generate a table in a markdown format. | ||
|
||
We can then use this in a normal instructor call | ||
|
||
```python | ||
import instructor | ||
import logfire | ||
|
||
|
||
openai_client = OpenAI() | ||
logfire.configure(pydantic_plugin=logfire.PydanticPlugin(record="all")) | ||
logfire.instrument_openai(openai_client) | ||
client = instructor.from_openai( | ||
openai_client, mode=instructor.function_calls.Mode.MD_JSON | ||
) | ||
|
||
@logfire.instrument("extract-table", extract_args=True) | ||
def extract_table_from_image(url: str) -> Iterable[Table]: | ||
return client.chat.completions.create( | ||
model="gpt-4-vision-preview", | ||
response_model=Iterable[Table], | ||
max_tokens=1800, | ||
messages=[ | ||
{ | ||
"role": "user", | ||
"content": [ | ||
{ | ||
"type": "text", | ||
"text": "Extract out a table from the image. Only extract out the total number of skiiers.", | ||
}, | ||
{"type": "image_url", "image_url": {"url": url}}, | ||
], | ||
} | ||
], | ||
) | ||
``` | ||
|
||
We can then call it as seen below | ||
|
||
```python | ||
url = "https://cdn.statcdn.com/Infographic/images/normal/16330.jpeg" | ||
tables = extract_table_from_image(url) | ||
for table in tables: | ||
print(table.caption, end="\n") | ||
print(table.dataframe.to_markdown()) | ||
``` | ||
|
||
Logfire is able to capture the stack track of the entire call as seen below, profile each part of our application and most importantly capture the raw inputs of the OpenAI call alongside any potential errors. | ||
|
||
![Logfire Image](img/image-logfire.png) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,52 @@ | ||
import enum | ||
from pydantic import BaseModel | ||
from openai import OpenAI | ||
import instructor | ||
import logfire | ||
|
||
|
||
class Labels(str, enum.Enum): | ||
"""Enumeration for single-label text classification.""" | ||
|
||
SPAM = "spam" | ||
NOT_SPAM = "not_spam" | ||
|
||
|
||
class SinglePrediction(BaseModel): | ||
""" | ||
Class for a single class label prediction. | ||
""" | ||
|
||
class_label: Labels | ||
|
||
|
||
openai_client = OpenAI() | ||
logfire.configure(pydantic_plugin=logfire.PydanticPlugin(record="all")) | ||
logfire.instrument_openai(openai_client) | ||
client = instructor.from_openai(openai_client) | ||
|
||
|
||
@logfire.instrument("classification", extract_args=True) | ||
def classify(data: str) -> SinglePrediction: | ||
"""Perform single-label classification on the input text.""" | ||
return client.chat.completions.create( | ||
model="gpt-3.5-turbo-0613", | ||
response_model=SinglePrediction, | ||
messages=[ | ||
{ | ||
"role": "user", | ||
"content": f"Classify the following text: {data}", | ||
}, | ||
], | ||
) | ||
|
||
|
||
if __name__ == "__main__": | ||
emails = [ | ||
"Hello there I'm a Nigerian prince and I want to give you money", | ||
"Meeting with Thomas has been set at Friday next week", | ||
"Here are some weekly product updates from our marketing team", | ||
] | ||
|
||
for email in emails: | ||
classify(email) |
Oops, something went wrong.