Skip to main content

Choosing the best GPT model: benchmark on your own data

This guide will walk you through how to compare OpenAI's GPT-4o and GPT-4o-mini, top contenders for the most powerful and effective GPT models. This testing framework will give you the chance to test the models' reasoning capabilities, cost, and latency.

New model releases often score well on benchmarks. But generic benchmarks are for generic use cases. If you're building an LLM app, you should evaluate these models on your own data and make an informed decision based on your specific needs.

The end result will be a side-by-side comparison that looks like this:

gpt 4o-mini vs gpt 4o

Prerequisites

To start, make sure you have:

Step 1: Setup

Create a dedicated directory for your comparison project:

npx promptfoo@latest init gpt-comparison

Edit promptfooconfig.yaml to include both models:

providers:
- openai:gpt-4o-mini
- openai:gpt-4

Step 2: Crafting the prompts

For our comparison, we'll use a simple prompt:

prompts:
- 'Solve this riddle: {{riddle}}'

Feel free to add multiple prompts and tailor to your use case.

Step 3: Create test cases

Above, we have a {{riddle}} placeholder variable. Each test case runs the prompts with a different riddle:

tests:
- vars:
riddle: 'I speak without a mouth and hear without ears. I have no body, but I come alive with wind. What am I?'
- vars:
riddle: 'You see a boat filled with people. It has not sunk, but when you look again you don’t see a single person on the boat. Why?'
- vars:
riddle: 'The more of this there is, the less you see. What is it?'

Step 4: Run the comparison

Execute the comparison with the following command:

npx promptfoo@latest eval

This will process the riddles against both GPT-3.5 and GPT-4, providing you with side-by-side results in your command line interface:

npx promptfoo@latest view

Step 5: Automatic evaluation

To streamline the evaluation process, you can add various types of assertions to your test cases. Assertions verify if the model's output meets certain criteria, marking the test as pass or fail accordingly.

In this case, we're especially interested in cost and latency assertions given the tradeoffs between the two models:

tests:
- vars:
riddle: 'I speak without a mouth and hear without ears. I have no body, but I come alive with wind. What am I?'
assert:
# Make sure the LLM output contains this word
- type: contains
value: echo
# Inference should always cost less than this (USD)
- type: cost
threshold: 0.001
# Inference should always be faster than this (milliseconds)
- type: latency
threshold: 5000
# Use model-graded assertions to enforce free-form instructions
- type: llm-rubric
value: Do not apologize
- vars:
riddle: 'You see a boat filled with people. It has not sunk, but when you look again you don’t see a single person on the boat. Why?'
assert:
- type: cost
threshold: 0.002
- type: latency
threshold: 3000
- type: llm-rubric
value: explains that the people are below deck
- vars:
riddle: 'The more of this there is, the less you see. What is it?'
assert:
- type: contains
value: darkness
- type: cost
threshold: 0.0015
- type: latency
threshold: 4000

After setting up your assertions, rerun the promptfoo eval command. This automated process helps quickly determine which model best fits your reasoning task requirements.

For more info on available assertion types, see assertions & metrics.

Cleanup

Finally, we'll use defaultTest to clean things up a bit and apply global latency and cost requirements. Here's the final eval config:

providers:
- openai:gpt-4o-mini
- openai:gpt-4

prompts:
- 'Solve this riddle: {{riddle}}'

defaultTest:
assert:
# Inference should always cost less than this (USD)
- type: cost
threshold: 0.001
# Inference should always be faster than this (milliseconds)
- type: latency
threshold: 3000

tests:
- vars:
riddle: "I speak without a mouth and hear without ears. I have no body, but I come alive with wind. What am I?"
assert:
- type: contains
value: echo
- vars:
riddle: "You see a boat filled with people. It has not sunk, but when you look again you don’t see a single person on the boat. Why?"
assert:
- type: llm-rubric
value: explains that the people are below deck
- vars:
riddle: "The more of this there is, the less you see. What is it?"
assert:
- type: contains
value: darkness

For more info on setting up the config, see the configuration guide.

Conclusion

In the end, you will see a result like this:

gpt 4o-mini vs gpt 4o

In this particular eval, the models performed very similarly in terms of answers, but it looks like GPT-4o-mini exceeded our maximum latency. Notably, 4o was about 35x more expensive compared to 4o-mini.

Of course, this is a limited example test set. The tradeoff between cost, latency, and accuracy is going to be tailored for each application. That's why it's important to run your own eval.

I encourage you to experiment with your own test cases and use this guide as a starting point. To learn more, see Getting Started.