from chatlas import ChatAnthropic, ChatOpenAI
= """
question How can I compute the mean and median of variables a, b, c, and so on,
all the way up to z, grouped by age and sex.
"""
Prompt design
This article gives you some advice about how to use chatlas to write prompts. We’ll work through two hopefully relevant examples: a prompt that generates code and another that extracts structured data. If you’ve never written a prompt, I’d highly recommend reading Ethan Mollick’s Getting started with AI: Good enough prompting. I think understanding his analogy about how AI works will really help you get started:
Treat AI like an infinitely patient new coworker who forgets everything you tell them each new conversation, one that comes highly recommended but whose actual abilities are not that clear. … Two parts of this are analogous to working with humans (being new on the job and being a coworker) and two of them are very alien (forgetting everything and being infinitely patient). We should start with where AIs are closest to humans, because that is the key to good-enough prompting
As well as learning general prompt design skills, it’s also a good idea to read any specific advice for the model that you’re using. Here are some pointers to the prompt design guides of some of the most popular models:
If you have a claude account, you can use its prompt-generator. It’s specifically tailored for Claude, but I suspect it will help you with many other LLMs, or at least give you some ideas as to what else to include in your prompt.
Best practices
It’s highly likely that you’ll end up writing long, possibly multi-page prompts. To ensure your success with this task, we have two recommendations. First, put each prompt its own, separate file. Second, write the prompts using markdown. The reason to use markdown is that it’s quite readable to LLMs (and humans), and it allows you to do things like use headers to divide up a prompt into sections and itemised lists to enumerate multiple options. You can see some examples of this style of prompt here:
- https://github.com/posit-dev/shiny-assistant/blob/main/shinyapp/app_prompt_python.md
- https://github.com/jcheng5/py-sidebot/blob/main/prompt.md
- https://github.com/simonpcouch/pal/tree/main/inst/prompts
- https://github.com/cpsievert/aidea/blob/main/inst/app/prompt.md
In terms of file names, if you only have one prompt in your project, call it prompt.md
. If you have multiple prompts, give them informative names like prompt-extract-metadata.md
or prompt-summarize-text.md
. If you’re writing a package, put your prompt(s) in a prompts
directory, otherwise it’s fine to put them in the project’s root directory.
Your prompts are going to change over time, so we’d highly recommend commiting them to a git repo. That will ensure that you can easily see what has changed, and that if you accidentally make a mistake you can easily roll back to a known good verison.
If your prompt includes dynamic data, you could use something like f-strings to insert variables, but using {
and }
for templating won’t work well when the prompt contains JSON. Instead, consider using chatlas.interpolate()
(or chatlas.interpolate_file()
), which uses { }
instead of { }
to make it easier to work with JSON.
As you iterate on the prompt, it’s a good idea to build up a small set of challenging examples that you can regularly re-check with your latest version of the prompt. Currently you’ll need to do this by hand, but we hope to eventually also provide tools that’ll help you do this a little more formally.
Unfortunately, you won’t see these best practices in action in this article since we’re keeping the prompts short and inline to make it easier for you to grok what’s going on.
Code generation
Let’s explore prompt design for a simple code generation task:
Basic flavour
When I don’t provide a system prompt, I sometimes get answers in a different language (like R):
= ChatAnthropic()
chat = chat.chat(question) _
Here’s how to compute mean and median for variables a through z, grouped by age and sex:
# Using dplyr
library(dplyr)
%>%
df group_by(age, sex) %>%
summarise(across(a:z, list(
mean = ~mean(., na.rm = TRUE),
median = ~median(., na.rm = TRUE)
)))
# Alternative base R approach
aggregate(. ~ age + sex, data = df[,c("age", "sex", letters[1:26])],
FUN = function(x) c(mean = mean(x), median = median(x)))
This will: 1. Group the data by age and sex 2. Calculate both mean and median for each variable a through z 3. Handle missing values with na.rm = TRUE 4. Return a dataframe with results for each age-sex combination
The output will have columns for age, sex, and mean/median values for each variable.
I can ensure that I always get Python code by providing a system prompt:
= "You are a helpful Python (not R) programming assistant."
chat.system_prompt = chat.chat(question) _
Here’s how to compute mean and median for variables a through z using Python, likely with pandas:
import pandas as pd
# Assuming your data is in a DataFrame called 'df'
# and variables a through z are column names
# Create a list of columns from 'a' to 'z'
= list('abcdefghijklmnopqrstuvwxyz')
cols
# Group by age and sex, then calculate mean and median
= df.groupby(['age', 'sex'])[cols].agg(['mean', 'median'])
result
# If you want to reset the index to make age and sex regular columns
= result.reset_index()
result
# Alternative more explicit way:
= df.groupby(['age', 'sex']).agg({
result 'mean', 'median'] for col in cols
col: [ })
This will: 1. Group your data by age and sex 2. Calculate both mean and median for each variable from a to z 3. Return a DataFrame with multi-level columns - First level: variable names (a through z) - Second level: statistics (mean and median) - Index: age and sex combinations
The result will look something like this:
age sex a b ...
mean median mean median ...
20 M 5.2 5.0 3.1 3.0 ...
F 4.8 4.5 3.3 3.0 ...
21 M 5.1 5.0 3.2 3.0 ...
Note: If you have missing values, pandas handles them automatically, but you can explicitly specify:
= df.groupby(['age', 'sex'])[cols].agg(['mean', 'median'], skipna=True) result
Note that I’m using both a system prompt (which defines the general behaviour) and a user prompt (which asks the specific question). You could put all of the content in the user prompt and get similar results, but I think it’s helpful to use both to cleanly divide the general framing of the response from the specific questions that you want to ask.
Since I’m mostly interested in the code, I ask it to drop the explanation:
= """
chat.system_prompt You are a helpful Python (not R) programming assistant.
Just give me the code without any text explanation.
"""
= chat.chat(question) _
import pandas as pd
= list('abcdefghijklmnopqrstuvwxyz')
cols = df.groupby(['age', 'sex'])[cols].agg(['mean', 'median']) result
In this case, I seem to mostly get pandas code. But if you want a different style, you can ask for it:
= """
chat.system_prompt You are a helpful Python (not R) programming assistant who prefers polars to pandas.
Just give me the code without any text explanation.
"""
= chat.chat(question) _
import polars as pl
= list('abcdefghijklmnopqrstuvwxyz')
cols = (df.groupby(['age', 'sex'])
result
.agg([
pl.col(cols).mean(),
pl.col(cols).median() ]))
Be explicit
If there’s something about the output that you don’t like, you can try being more explicit about it. For example, the code isn’t styled quite how I like, so I provide more details about what I do want:
= """
chat.system_prompt You are a helpful Python (not R) programming assistant who prefers siuba to pandas.
Just give me the code. I don't want any explanation or sample data.
* Spread long function calls across multiple lines.
* Where needed, always indent function calls with two spaces.
* Always use double quotes for strings.
"""
= chat.chat(question) _
from siuba import _, group_by, summarize
from siuba.siu import mean, median
= list('abcdefghijklmnopqrstuvwxyz')
cols
= (df
result >> group_by(_.age, _.sex)
>> summarize(**{
f"{col}_mean": mean(_[col]) for col in cols
| {
} f"{col}_median": median(_[col]) for col in cols
}))
This still doesn’t yield exactly the code that I’d write, but it’s prety close.
You could provide a different prompt if you were looking for more explanation of the code:
= """
chat.system_prompt You are an an expert Python (not R) programmer and a warm and supportive teacher.
Help me understand the code you produce by explaining each function call with
a brief comment. For more complicated calls, add documentation to each
argument. Just give me the code without any text explanation.
"""
= chat.chat(question) _
import pandas as pd
import string
# Create list of lowercase letters a-z
= list(string.ascii_lowercase)
cols
# Group by age and sex, calculate mean and median for all letter columns
= df.groupby(['age', 'sex'])[cols].agg(['mean', 'median'])
result
# Optional: flatten multi-level column names
= [f'{col}_{stat}' for col, stat in result.columns] result.columns
Teach it about new features
You can imagine LLMs as being a sort of an average of the internet at a given point in time. That means they will provide popular answers, which will tend to reflect older coding styles (either because the new features aren’t in their index, or the older features are so much more popular). So if you want your code to use specific newer language features, you might need to provide the examples yourself:
= """
chat.system_prompt You are an expert R programmer.
Just give me the code; no explanation in text.
Use the `.by` argument rather than `group_by()`.
dplyr 1.1.0 introduced per-operation grouping with the `.by` argument.
e.g., instead of:
transactions |>
group_by(company, year) |>
mutate(total = sum(revenue))
write this:
transactions |>
mutate(
total = sum(revenue),
.by = c(company, year)
)
"""
= chat.chat(question) _
|>
df summarise(
across(a:z, list(
mean = ~mean(., na.rm = TRUE),
median = ~median(., na.rm = TRUE)
)),.by = c(age, sex)
)
Structured data
Providing a rich set of examples is a great way to encourage the output to produce exactly what you want. This is known as multi-shot prompting. Below we’ll work through a prompt that I designed to extract structured data from recipes, but the same ideas apply in many other situations.
Getting started
My overall goal is to turn a list of ingredients, like the following, into a nicely structured JSON that I can then analyse in Python (e.g. compute the total weight, scale the recipe up or down, or convert the units from volumes to weights).
= """
ingredients ¾ cup (150g) dark brown sugar
2 large eggs
¾ cup (165g) sour cream
½ cup (113g) unsalted butter, melted
1 teaspoon vanilla extract
¾ teaspoon kosher salt
⅓ cup (80ml) neutral oil
1½ cups (190g) all-purpose flour
150g plus 1½ teaspoons sugar
"""
= ChatOpenAI(model="gpt-4o-mini") chat
(This isn’t the ingredient list for a real recipe but it includes a sampling of styles that I encountered in my project.)
If you don’t have strong feelings about what the data structure should look like, you can start with a very loose prompt and see what you get back. I find this a useful pattern for underspecified problems where the heavy lifting lies with precisely defining the problem you want to solve. Seeing the LLM’s attempt to create a data structure gives me something to react to, rather than having to start from a blank page.
= """
instruct_json You're an expert baker who also loves JSON. I am going to give you a list of
ingredients and your job is to return nicely structured JSON. Just return the
JSON and no other commentary.
"""
= instruct_json
chat.system_prompt = chat.chat(ingredients) _
{
"ingredients": [
{
"name": "dark brown sugar",
"amount": "¾ cup",
"weight": "150g"
},
{
"name": "large eggs",
"amount": "2"
},
{
"name": "sour cream",
"amount": "¾ cup",
"weight": "165g"
},
{
"name": "unsalted butter",
"amount": "½ cup",
"weight": "113g",
"preparation": "melted"
},
{
"name": "vanilla extract",
"amount": "1 teaspoon"
},
{
"name": "kosher salt",
"amount": "¾ teaspoon"
},
{
"name": "neutral oil",
"amount": "⅓ cup",
"volume": "80ml"
},
{
"name": "all-purpose flour",
"amount": "1½ cups",
"weight": "190g"
},
{
"name": "sugar",
"amount": "150g plus 1½ teaspoons"
}
]
}
(I don’t know if the additional colour, “You’re an expert baker who also loves JSON”, does anything, but I like to think this helps the LLM get into the right mindset of a very nerdy baker.)
Provide examples
This isn’t a bad start, but I prefer to cook with weight and I only want to see volumes if weight isn’t available so I provide a couple of examples of what I’m looking for. I was pleasantly suprised that I can provide the input and output examples in such a loose format.
= """
instruct_weight Here are some examples of the sort of output I'm looking for:
¾ cup (150g) dark brown sugar
{"name": "dark brown sugar", "quantity": 150, "unit": "g"}
⅓ cup (80ml) neutral oil
{"name": "neutral oil", "quantity": 80, "unit": "ml"}
2 t ground cinnamon
{"name": "ground cinnamon", "quantity": 2, "unit": "teaspoon"}
"""
= instruct_json + "\n" + instruct_weight
chat.system_prompt = chat.chat(ingredients) _
{
"ingredients": [
{
"name": "dark brown sugar",
"quantity": 150,
"unit": "g"
},
{
"name": "large eggs",
"quantity": 2,
"unit": "count"
},
{
"name": "sour cream",
"quantity": 165,
"unit": "g"
},
{
"name": "unsalted butter",
"quantity": 113,
"unit": "g",
"preparation": "melted"
},
{
"name": "vanilla extract",
"quantity": 1,
"unit": "teaspoon"
},
{
"name": "kosher salt",
"quantity": 0.75,
"unit": "teaspoon"
},
{
"name": "neutral oil",
"quantity": 80,
"unit": "ml"
},
{
"name": "all-purpose flour",
"quantity": 190,
"unit": "g"
},
{
"name": "sugar",
"quantity": "150g plus 1½ teaspoons"
}
]
}
Just providing the examples seems to work remarkably well. But I found it useful to also include a description of what the examples are trying to accomplish. I’m not sure if this helps the LLM or not, but it certainly makes it easier for me to understand the organisation and check that I’ve covered the key pieces I’m interested in.
= """
instruct_weight * If an ingredient has both weight and volume, extract only the weight:
¾ cup (150g) dark brown sugar
[
{"name": "dark brown sugar", "quantity": 150, "unit": "g"}
]
* If an ingredient only lists a volume, extract that.
2 t ground cinnamon
⅓ cup (80ml) neutral oil
[
{"name": "ground cinnamon", "quantity": 2, "unit": "teaspoon"},
{"name": "neutral oil", "quantity": 80, "unit": "ml"}
]
"""
This structure also allows me to give the LLMs a hint about how I want multiple ingredients to be stored, i.e. as an JSON array.
I then iterated on the prompt, looking at the results from different recipes to get a sense of what the LLM was getting wrong. Much of this felt like I was iterating on my own understanding of the problem as I didn’t start by knowing exactly how I wanted the data. For example, when I started out I didn’t really think about all the various ways that ingredients are specified. For later analysis, I always want quantities to be number, even if they were originally fractions, or the if the units aren’t precise (like a pinch). It made me to realise that some ingredients are unitless.
= """
instruct_unit * If the unit uses a fraction, convert it to a decimal.
⅓ cup sugar
½ teaspoon salt
[
{"name": "dark brown sugar", "quantity": 0.33, "unit": "cup"},
{"name": "salt", "quantity": 0.5, "unit": "teaspoon"}
]
* Quantities are always numbers
pinch of kosher salt
[
{"name": "kosher salt", "quantity": 1, "unit": "pinch"}
]
* Some ingredients don't have a unit.
2 eggs
1 lime
1 apple
[
{"name": "egg", "quantity": 2},
{"name": "lime", "quantity": 1},
{"name", "apple", "quantity": 1}
]
"""
You might want to take a look at the full prompt to see what I ended up with.
Structured data
Now that I’ve iterated to get a data structure I like, it seems useful to formalise it and tell the LLM exactly what I’m looking for when dealing with structured data. This guarantees that the LLM will only return JSON, that the JSON will have the fields that you expect, and that chatlas will convert it into an Python data structure for you.
from pydantic import BaseModel, Field
class Ingredient(BaseModel):
"Ingredient name"
str = Field(description="Ingredient name")
name: float
quantity: str | None = Field(description="Unit of measurement")
unit:
class Ingredients(BaseModel):
list[Ingredient]
items:
= instruct_json + "\n" + instruct_weight
chat.system_prompt =Ingredients) chat.extract_data(ingredients, data_model
{'items': [{'name': 'dark brown sugar', 'quantity': 150, 'unit': 'g'},
{'name': 'large eggs', 'quantity': 2, 'unit': 'count'},
{'name': 'sour cream', 'quantity': 165, 'unit': 'g'},
{'name': 'unsalted butter', 'quantity': 113, 'unit': 'g'},
{'name': 'vanilla extract', 'quantity': 1, 'unit': 'teaspoon'},
{'name': 'kosher salt', 'quantity': 0.75, 'unit': 'teaspoon'},
{'name': 'neutral oil', 'quantity': 80, 'unit': 'ml'},
{'name': 'all-purpose flour', 'quantity': 190, 'unit': 'g'},
{'name': 'sugar', 'quantity': 150, 'unit': 'g'}]}
Capturing raw input
One thing that I’d do next time would also be to include the raw ingredient names in the output. This doesn’t make much difference in this simple example but it makes it much easier to align the input with the output and to start developing automated measures of how well my prompt is doing.
= """
instruct_weight_input * If an ingredient has both weight and volume, extract only the weight:
¾ cup (150g) dark brown sugar
[
{"name": "dark brown sugar", "quantity": 150, "unit": "g", "input": "¾ cup (150g) dark brown sugar"}
]
* If an ingredient only lists a volume, extract that.
2 t ground cinnamon
⅓ cup (80ml) neutral oil
[
{"name": "ground cinnamon", "quantity": 2, "unit": "teaspoon", "input": "2 t ground cinnamon"},
{"name": "neutral oil", "quantity": 80, "unit": "ml", "input": "⅓ cup (80ml) neutral oil"}
]
"""
I think this is particularly important if you’re working with even less structured text. For example, imagine you had this text:
= """
recipe In a large bowl, cream together one cup of softened unsalted butter and a
quarter cup of white sugar until smooth. Beat in an egg and 1 teaspoon of
vanilla extract. Gradually stir in 2 cups of all-purpose flour until the
dough forms. Finally, fold in 1 cup of semisweet chocolate chips. Drop
spoonfuls of dough onto an ungreased baking sheet and bake at 350°F (175°C)
for 10-12 minutes, or until the edges are lightly browned. Let the cookies
cool on the baking sheet for a few minutes before transferring to a wire
rack to cool completely. Enjoy!
"""
Including the input text in the output makes it easier to see if it’s doing a good job:
= instruct_json + "\n" + instruct_weight_input
chat.system_prompt = chat.chat(ingredients) _
{
"ingredients": [
{
"name": "dark brown sugar",
"quantity": 150,
"unit": "g"
},
{
"name": "large eggs",
"quantity": 2,
"unit": "count"
},
{
"name": "sour cream",
"quantity": 165,
"unit": "g"
},
{
"name": "unsalted butter",
"quantity": 113,
"unit": "g",
"preparation": "melted"
},
{
"name": "vanilla extract",
"quantity": 1,
"unit": "teaspoon"
},
{
"name": "kosher salt",
"quantity": 0.75,
"unit": "teaspoon"
},
{
"name": "neutral oil",
"quantity": 80,
"unit": "ml"
},
{
"name": "all-purpose flour",
"quantity": 190,
"unit": "g"
},
{
"name": "sugar",
"quantity": "150g plus 1½ teaspoons"
}
]
}
When I ran it while writing this vignette, it seemed to be working out the weight of the ingredients specified in volume, even though the prompt specifically asks it not to. This may suggest I need to broaden my examples.
Token usage
from chatlas import token_usage
token_usage()
[{'name': 'Anthropic', 'input': 5887, 'output': 1102},
{'name': 'OpenAI', 'input': 3036, 'output': 1050}]