Introduction to LLM Metrics
deepeval offers 50+ SOTA, ready-to-use metrics for you to quickly get started with. Essentially, while a test case represents the thing you're trying to measur, the metric acts as the ruler based on a specific criteria of interest.
Quick Summary
Almost all predefined metrics on deepeval uses LLM-as-a-judge, with various techniques such as QAG (question-answer-generation), DAG (deep acyclic graphs), and G-Eval to score test cases, which represents atomic interactions with your LLM app.
All of deepeval's metrics output a score between 0-1 based on its corresponding equation, as well as score reasoning. A metric is only successful if the evaluation score is equal to or greater than threshold, which is defaulted to 0.5 for all metrics.
Custom metrics allow you to define your custom criteria using SOTA implementations of LLM-as-a-Judge metrics in everyday language:
- G-Eval
- DAG (Deep Acyclic Graph)
- Conversational G-Eval
- Conversational DAG
- Arena G-Eval
- Do it yourself, 100% self-coded metrics (e.g. if you want to use BLEU, ROUGE)
You should aim to have at least one custom metric in your LLM evals pipeline.
RAG (retrieval augmented generation) metrics focus on the retriever and generator components independently.
-
Retriever:
- Contextual Relevancy
- Contextual Precision
- Contextual Recall
-
Generator:
- Answer Relevancy
- Faithfulness
Agentic metrics evaluates the overall execution flow of your agent. In deepeval, there are six main agentic metrics:
- Task Completion
- Argument Correctness
- Tool Correctness
- Step Efficiency
- Plan Adherence
- Plan Quality
The task completion metric does not require a test case and will take an LLM trace to evaluate task completion (i.e. you'll have to setup LLM tracing).
Multi-turn metrics' main use case are for evaluating chatbots and uses a ConversationalTestCase instead. They include:
- Knowledge Retention
- Role Adherence
- Conversation Completeness
- Conversation Relevancy
Multi-turn metrics evaluates conversations as a whole and takes prior context into consideration when doing so.
Safety metrics concerns more on LLM security. They include:
- Bias
- Toxicity
- Non-Advice
- Misuse
- PIILeakage
- Role Violation
For those looking for a full-blown LLM red teaming orchestration frameowork, checkout DeepTeam. DeepTeam is deepeval but for red teaming LLMs specifically.
Metrics in deepeval are multi-modal by default, metrics targetting images are metrics that definitely expects an image in the test case. They include:
- Image Coherence
- Image Helpfulness
- Image Reference
- Text-to-Image
- Image-Editing
Note that multi-modal metrics requires MLLMImages in LLMTestCases.
Not use case specific, but still useful for some use cases:
- Hallucination
- Json Correctness
- Summarization
- Ragas
Your LLM app can be evaluated end-to-end (component-level example further below) by providing a list of metrics and test cases:
from deepeval.test_case import LLMTestCase
from deepeval.metrics import AnswerRelevancyMetric
from deepeval import evaluate
evaluate(
metrics=[AnswerRelevancyMetric()],
test_cases=[LLMTestCase(input="What's `deepeval`?", actual_output="Your favorite eval framework's favorite evals framework.")]
)If you're logged into Confident AI before running an evaluation (deepeval login or deepeval view in the CLI), you'll also get entire testing reports on the platform:
More information on everything can be found on the Confident AI evaluation docs.
Why deepeval Metrics?
Apart from the variety of metrics offered, deepeval's metrics are a step up to other implementations because they:
- Are research-backed LLM-as-as-Judge (
GEval) - One of the most used in the world (20 million+ daily evaluations)
- Make deterministic metric scores possible (when using
DAGMetric) - Are extra reliable as LLMs are only used for extremely confined tasks during evaluation to greatly reduce stochasticity and flakiness in scores
- Provide a comprehensive reason for the scores computed
- Integrated 100% with Confident AI
Create Your First Metric
Custom Metrics
deepeval provides G-Eval, a state-of-the-art LLM evaluation framework for anyone to create a custom LLM-evaluated metric using natural language. G-Eval is available for all single-turn, multi-turn, and multimodal evals.
from deepeval.test_case import LLMTestCase, SingleTurnParams
from deepeval.metrics import GEval
test_case = LLMTestCase(input="...", actual_output="...", expected_output="...")
correctness = GEval(
name="Correctness",
criteria="Correctness - determine if the actual output is correct according to the expected output.",
evaluation_params=[SingleTurnParams.ACTUAL_OUTPUT, SingleTurnParams.EXPECTED_OUTPUT],
strict_mode=True
)
correctness.measure(test_case)
print(correctness.score, correctness.reason)from deepeval.test_case import Turn, MultiTurnParams, ConversationalTestCase
from deepeval.metrics import ConversationalGEval
convo_test_case = ConversationalTestCase(turns=[Turn(role="...", content="..."), Turn(role="...", content="...")])
professionalism_metric = ConversationalGEval(
name="Professionalism",
criteria="Determine whether the assistant has acted professionally based on the content."
evaluation_params=[MultiTurnParams.CONTENT],
strict_mode=True
)
professionalism_metric.measure(convo_test_case)
print(professionalism_metric.score, professionalism_metric.reason)Under the hood, deepeval first generates a series of evaluation steps, before using these steps in conjunction with information in an LLMTestCase for evaluation. For more information, visit the G-Eval documentation page.
Default Metrics
The most used RAG metrics include:
- Answer Relevancy: Evaluates if the generated answer is relevant to the user query
- Faithfulness: Measures if the generated answer is factually consistent with the provided context
- Contextual Relevancy: Assesses if the retrieved context is relevant to the user query
- Contextual Recall: Evaluates if the retrieved context contains all relevant information
- Contextual Precision: Measures if the retrieved context is precise and focused
Which can be simply imported from the deepeval.metrics module:
from deepeval.test_case import LLMTestCase
from deepeval.metrics import AnswerRelevancyMetric
test_case = LLMTestCase(input="...", actual_output="...")
relevancy = AnswerRelevancyMetric(threshold=0.5)
relevancy.measure(test_case)
print(relevancy.score, relevancy.reason)The most used agentic metrics include:
- Task Completion: Assesses if the agent successfully completed a given task for a given LLM trace
- Tool Correctness: Evaluates if tools were called and used correctly
There's not a lot of metrics required for agents since most is taken care of by task completion. To use the task completion metric, you have to setup tracing (just like for component-level evals shown above):
from deepeval.metrics import TaskCompletionMetric
from deepeval.tracing import observe
from deepeval.dataset import Golden
from deepeval import evaluate
task_completion = TaskCompletionMetric(threshold=0.5)
@observe(metrics=[task_completion])
def trip_planner_agent(input):
@observe()
def itinerary_generator(destination, days):
return ["Eiffel Tower", "Louvre Museum", "Montmartre"][:days]
return itinerary_generator("Paris", 2)
evaluate(observed_callback=trip_planner_agent, goldens=[Golden(input="Paris, 2")])Chatbots require "conversational" (or multi-turn) metrics and they include:
- Conversation Completeness: Evaluates if conversation satisify user needs.
- Conversation Relevancy: Measures if the generated outputs are relevant to user inputs.
- Role Adherence: Assesses if the chatbot stays in character throughout a conversation.
- Knowledge Retention: Evaluates if the chatbot is able to retain knowledge learnt throughout a conversation.
You'll need to also use ConversationalTestCases instead of regular LLMTestCase for conversational metrics:
from deepeval.test_case import Turn, ConversationalTestCase
from deepeval.metrics import ConversationalGEval
convo_test_case = ConversationalTestCase(turns=[Turn(role="...", content="..."), Turn(role="...", content="...")])
role_adherence = RoleAdherenceMetric(threshold=0.5)
role_adherence.measure(convo_test_case)
print(role_adherence.score, role_adherence.reason)from deepeval.test_case import LLMTestCase, MLLMImage
from deepeval.metrics import ImageCoherenceMetric
test_case = LLMTestCase(input=f"What does thsi image say? {MLLMImage(...)}", actual_output="No idea!")
image_coherence = ImageCoherenceMetric(threshold=0.5)
image_coherence.measure(m_test_case)
print(image_coherence.score, image_coherence.reason)from deepeval.test_case import LLMTestCase
from deepeval.metrics import BiasMetric
test_case = LLMTestCase(input="...", actual_output="...")
bias = BiasMetric(threshold=0.5)
bias.measure(test_case)
print(bias.score, bias.reason)Choosing Your Metrics
These are the metric categories to consider when choosing your metrics:
- Custom metrics are use case specific and architecture agnostic:
- G-Eval – best for subjective criteria like correctness, coherence, or tone; easy to set up.
- DAG – decision-tree metric for objective or mixed criteria (e.g., verify format before tone).
- Start with G-Eval for simplicity; use DAG for more control. You can also subclass
BaseMetricto create your own.
- Generic metrics are system specific and use case agnostic:
- RAG metrics: measures retriever and generator separately
- Agent metrics: evaluate tool usage and task completion
- Multi-turn metrics: measure overall dialogue quality
- Combine these for multi-component LLM systems.
- Reference vs. Referenceless:
- Reference-based metrics need ground truth (e.g., contextual recall or tool correctness).
- Referenceless metrics work without labeled data, ideal for online or production evaluation.
- Check each metric’s docs for required parameters.
When deciding on metrics, no matter how tempting, try to limit yourself to no more than 5 metrics, with this breakdown:
- 2-3 generic, system-specific metrics (e.g. contextual precision for RAG, tool correctness for agents)
- 1-2 custom, use case-specific metrics (e.g. helpfulness for a medical chatbot, format correctness for summarization)
The goal is to force yourself to prioritize and clearly define your evaluation criteria. This will not only help you use deepeval, but also help you understand what you care most about in your LLM application.
Here are some additional ideas if you're not sure:
- RAG: Focus on the
AnswerRelevancyMetric(evaluatesactual_outputalignment with theinput) andFaithfulnessMetric(checks for hallucinations againstretrieved_context) - Agents: Use the
ToolCorrectnessMetricto verify proper tool selection and usage - Chatbots: Implement a
ConversationCompletenessMetricto assess overall conversation quality - Custom Requirements: When standard metrics don't fit your needs, create custom evaluations with
G-EvalorDAGframeworks
In some cases, where your LLM model is doing most of the heavy lifting, it is not uncommon to have more use case specific metrics.
Configure LLM Judges
You can use ANY LLM judge in deepeval, including OpenAI, Azure OpenAI, Ollama, Anthropic, Gemini, LiteLLM, etc. You can also wrap your own LLM API in deepeval's DeepEvalBaseLLM class to use ANY model of your choice. Click here for full guide.
To use OpenAI for deepeval's LLM metrics, supply your OPENAI_API_KEY in the CLI:
export OPENAI_API_KEY=<your-openai-api-key>Alternatively, if you're working in a notebook environment (Jupyter or Colab), set your OPENAI_API_KEY in a cell:
%env OPENAI_API_KEY=<your-openai-api-key>deepeval also allows you to use Azure OpenAI for metrics that are evaluated using an LLM. Run the following command in the CLI to configure your deepeval environment to use Azure OpenAI for all LLM-based metrics.
deepeval set-azure-openai \
--base-url=<endpoint> \ # e.g. https://example-resource.azure.openai.com/
--model=<model_name> \ # e.g. gpt-4.1
--deployment-name=<deployment_name> \ # e.g. Test Deployment
--api-version=<api_version> \ # e.g. 2025-01-01-preview
--model-version=<model_version> # e.g. 2024-11-20Note that the model-version is optional. If you ever wish to stop using Azure OpenAI and move back to regular OpenAI, simply run:
deepeval unset-azure-openaiTo use Ollama models for your metrics, run deepeval set-ollama --model=<model> in your CLI. For example:
deepeval set-ollama --model=deepseek-r1:1.5bOptionally, you can specify the base URL of your local Ollama model instance if you've defined a custom port. The default base URL is set to http://localhost:11434.
deepeval set-ollama --model=deepseek-r1:1.5b \
--base-url="http://localhost:11434"To stop using your local Ollama model and move back to OpenAI, run:
deepeval unset-ollamaTo use Gemini models with deepeval, run the following command in your CLI.
deepeval set-gemini \
--model=<model_name> # e.g. "gemini-2.0-flash-001"deepeval allows you to use ANY custom LLM for evaluation. This includes LLMs from langchain's chat_model module, Hugging Face's transformers library, or even LLMs in GGML format.
This includes any of your favorite models such as:
- Azure OpenAI
- Claude via AWS Bedrock
- Google Vertex AI
- Mistral 7B
All the examples can be found here, but down below is a quick example of a custom Azure OpenAI model through langchain's AzureChatOpenAI module for evaluation:
from langchain_openai import AzureChatOpenAI
from deepeval.models.base_model import DeepEvalBaseLLM
class AzureOpenAI(DeepEvalBaseLLM):
def __init__(
self,
model
):
self.model = model
def load_model(self):
return self.model
def generate(self, prompt: str) -> str:
chat_model = self.load_model()
return chat_model.invoke(prompt).content
async def a_generate(self, prompt: str) -> str:
chat_model = self.load_model()
res = await chat_model.ainvoke(prompt)
return res.content
def get_model_name(self):
return "Custom Azure OpenAI Model"
# Replace these with real values
custom_model = AzureChatOpenAI(
openai_api_version=api_version,
azure_deployment=azure_deployment,
azure_endpoint=azure_endpoint,
openai_api_key=openai_api_key,
)
azure_openai = AzureOpenAI(model=custom_model)
print(azure_openai.generate("Write me a joke"))When creating a custom LLM evaluation model you should ALWAYS:
- inherit
DeepEvalBaseLLM. - implement the
get_model_name()method, which simply returns a string representing your custom model name. - implement the
load_model()method, which will be responsible for returning a model object. - implement the
generate()method with one and only one parameter of type string that acts as the prompt to your custom LLM. - the
generate()method should return the final output string of your custom LLM. Note that we calledchat_model.invoke(prompt).contentto access the model generations in this particular example, but this could be different depending on the implementation of your custom model object. - implement the
a_generate()method, with the same function signature asgenerate(). Note that this is an async method. In this example, we calledawait chat_model.ainvoke(prompt), which is an asynchronous wrapper provided by LangChain's chat models.
Lastly, to use it for evaluation for an LLM-Eval:
from deepeval.metrics import AnswerRelevancyMetric
...
metric = AnswerRelevancyMetric(model=azure_openai)Using Metrics
There are three ways you can use metrics:
- End-to-end evals, treating your LLM system as a black-box and evaluating the system inputs and outputs.
- Component-level evals, placing metrics on individual components in your LLM app instead.
- One-off (or standalone) evals, where you would use a metric to execute it individually.
For End-to-End Evals
To run end-to-end evaluations of your LLM system using any metric of your choice, simply provide a list of test cases to evaluate your metrics against:
from deepeval.test_case import LLMTestCase
from deepeval.metrics import AnswerRelevancyMetric
from deepeval import evaluate
test_case = LLMTestCase(input="...", actual_output="...")
evaluate(test_cases=[test_case], metrics=[AnswerRelevancyMetric()])The evaluate() function or deepeval test run is the best way to run evaluations. They offer tons of features out of the box, including caching, parallelization, cost tracking, error handling, and integration with Confident AI.
For Component-Level Evals
To run component-level evaluations of your LLM system using any metric of your choice, simply decorate your components with @observe and create test cases at runtime:
from deepeval.dataset import EvaluationDataset, Golden
from deepeval.tracing import observe, update_current_span
from deepeval.metrics import AnswerRelevancyMetric
# 1. observe() decorator traces LLM components
@observe()
def llm_app(input: str):
# 2. Supply metric at any component
@observe(metrics=[AnswerRelevancyMetric()])
def nested_component():
# 3. Create test case at runtime
update_current_span(test_case=LLMTestCase(...))
pass
nested_component()
# 4. Create dataset
dataset = EvaluationDataset(goldens=[Golden(input="Test input")])
# 5. Loop through dataset
for goldens in dataset.evals_iterator():
# Call LLM app
llm_app(golden.input)For One-Off Evals
You can also execute each metric individually. All metrics in deepeval, including custom metrics that you create:
- can be executed via the
metric.measure()method - can have its score accessed via
metric.score, which ranges from 0 - 1 - can have its score reason accessed via
metric.reason - can have its status accessed via
metric.is_successful() - can be used to evaluate test cases or entire datasets, with or without Pytest
- has a
thresholdthat acts as the threshold for success.metric.is_successful()is only true ifmetric.scoreis above/belowthreshold - has a
strict_modeproperty, which when turned on enforcesmetric.scoreto a binary one - has a
verbose_modeproperty, which when turned on prints metric logs whenever a metric is executed
In addition, all metrics in deepeval execute asynchronously by default. You can configure this behavior using the async_mode parameter when instantiating a metric.
Here's a quick example:
from deepeval.metrics import AnswerRelevancyMetric
from deepeval.test_case import LLMTestCase
# Initialize a test case
test_case = LLMTestCase(...)
# Initialize metric with threshold
metric = AnswerRelevancyMetric(threshold=0.5)
metric.measure(test_case)
print(metric.score, metric.reason)All of deepeval's metrics give a reason alongside its score.
Using Metrics Async
When a metric's async_mode=True (which is the default for all metrics), invocations of metric.measure() will execute internal algorithms concurrently. However, it's important to note that while operations INSIDE measure() execute concurrently, the metric.measure() call itself still blocks the main thread.
To measure multiple metrics at once and NOT block the main thread, use the asynchronous a_measure() method instead.
import asyncio
...
# Remember to use async
async def long_running_function():
# These will all run at the same time
await asyncio.gather(
metric1.a_measure(test_case),
metric2.a_measure(test_case),
metric3.a_measure(test_case),
metric4.a_measure(test_case)
)
print("Metrics finished!")
asyncio.run(long_running_function())Debug A Metric Judgement
You can turn on verbose_mode for ANY deepeval metric at metric initialization to debug a metric whenever the measure() or a_measure() method is called:
...
metric = AnswerRelevancyMetric(verbose_mode=True)
metric.measure(test_case)Customize Metric Prompts
All of deepeval's metrics use LLM-as-a-judge evaluation with unique default prompt templates for each metric. While deepeval has well-designed algorithms for each metric, you can customize these prompt templates to improve evaluation accuracy and stability. Simply provide a custom template class as the evaluation_template parameter to your metric of choice (example below).
You'll find this particularly valuable when using a custom LLM, as deepeval's default metrics are optimized for OpenAI's models, which are generally more powerful than most custom LLMs.
Here's a quick example of how you can define a custom AnswerRelevancyTemplate and inject it into the AnswerRelevancyMetric through the evaluation_params parameter:
from deepeval.metrics import AnswerRelevancyMetric
from deepeval.metrics.answer_relevancy import AnswerRelevancyTemplate
# Define custom template
class CustomTemplate(AnswerRelevancyTemplate):
@staticmethod
def generate_statements(actual_output: str):
return f"""Given the text, breakdown and generate a list of statements presented.
Example:
Our new laptop model features a high-resolution Retina display for crystal-clear visuals.
{{
"statements": [
"The new laptop model has a high-resolution Retina display."
]
}}
===== END OF EXAMPLE ======
Text:
{actual_output}
JSON:
"""
# Inject custom template to metric
metric = AnswerRelevancyMetric(evaluation_template=CustomTemplate)
metric.measure(...)What About Non-LLM-as-a-judge Metrics?
If you're looking to use something like ROUGE, BLEU, or BLEURT, etc. you can create a custom metric and use the scorer module available in deepeval for scoring by following this guide.
The scorer module is available but not documented because our experience tells us these scorers are not useful as LLM metrics where outputs require a high level of reasoning to evaluate.