Tasks¶
In this section we will see what's a Task
and the list of tasks available in distilabel
.
Task¶
The Task
class takes charge of setting how the LLM behaves, deciding whether it acts as a generator or a labeller. To accomplish this, the Task
class creates a prompt using a template that will be sent to the LLM
. It specifies the necessary input arguments for generating the prompt and identifies the output arguments to be extracted from the LLM
response. The Task
class yields a Prompt
that can generate a string with the format needed, depending on the specific LLM
used.
All the Task
s defines a system_prompt
which serves as the initial instruction given to the LLM, guiding it on what kind of information or output is expected, and the following methods:
generate_prompt
: This method will be used by theLLM
to create the prompts that will be fed to the model.parse_output
: After theLLM
has generated the content, this method will be called on the raw outputs of the model to extract the relevant content (scores, rationales, etc).input_args_names
andoutput_args_names
: These methods are used in thePipeline
to process the datasets. The first one defines the columns that will be extracted from the dataset to build the prompt in case of aLLM
that acts as a generator or labeller alone, or the columns that should be placed in the dataset to be processed by the labellerLLM
, in the case of aPipeline
that has both a generator and a labeller. The second one is in charge of inserting the defined fields as columns of the dataset generated dataset.
After defining a task, the only action required is to pass it to the corresponding LLM
. All the intricate processes are then handled internally:
from distilabel.llm import TransformersLLM
from distilabel.tasks import TextGenerationTask
# This snippet uses `TransformersLLM`, but is the same for every other `LLM`.
generator = TransformersLLM(
model=...,
tokenizer=...,
task=TextGenerationTask(),
)
Given this explanation, distilabel
distinguishes between two primary categories of tasks: those focused on text generation and those centered around labelling. These Task
classes delineate the LLM's conduct, be it the creation of textual content or the assignment of labels to text, each with precise guidelines tailored to their respective functionalities. Users can seamlessly leverage these distinct task types to tailor the LLM's behavior according to their specific application needs.
Text Generation¶
These set of classes are designed to steer a LLM
in generating text with specific guidelines. They provide a structured approach to instruct the LLM on generating content in a manner tailored to predefined criteria.
TextGenerationTask¶
This is the base class for text generation, and includes the following fields for guiding the generation process:
system_prompt
, which serves as the initial instruction or query given to the LLM, guiding it on what kind of information or output is expected.- A list of
principles
to inject on thesystem_prompt
, which by default correspond to those defined in the UltraFeedback paper1, - and lastly a distribution for these principles so the
LLM
can be directed towards the different principles with a more customized behaviour.
For the API reference visit TextGenerationTask.
SelfInstructTask¶
The task specially designed to build the prompts following the Self-Instruct paper: SELF-INSTRUCT: Aligning Language Models with Self-Generated Instructions.
From the original repository:
The Self-Instruct process is an iterative bootstrapping algorithm that starts with a seed set of manually-written instructions and uses them to prompt the language model to generate new instructions and corresponding input-output instances, so this Task
is specially interesting for generating new datasets from a set of predefined topics.
import os
from distilabel.llm import OpenAILLM
from distilabel.tasks import SelfInstructTask
generator = OpenAILLM(
task=SelfInstructTask(
system_prompt="You are a question-answering assistant for...",
application_description="AI assistant",
num_instructions=3,
criteria_for_query_generation="Design queries to be... ",
),
openai_api_key=os.getenv("OPENAI_API_KEY"),
)
For the API reference visit SelfInstructTask.
Customise your SelfInstructTask¶
You can personalize the way in which your SelfInstructTask behaves by changing the default values of the parameters to something that suits your use case. Let's go through them:
- System Prompt: you can control the overall behaviour and expectations of your model.
- Application Description: a description of the AI application. By default, we use "AI Assistant".
- Number of instructions: number of instructions in the prompt.
- Criteria for Query Generation: the criteria for query generation that we want our model to have. The default value covers default behaviour for SelfInstructTask. This value is passed to the .jinja template, where extra instructions are added to ensure correct output format.
Let's see an example of how to customise a SelfInstructTask to create Haikus in the snippet below. You can take a look at this dataset as an example of a Haiku DPO dataset.
from distilabel.tasks import SelfInstructTask
system_prompt: str = "You are an expert Haiku writer, writing the best and most diverse Haikus given topics as inputs."
application_description = (
"An AI assistant adept at writing Haiku.\n"
"It expects complete suggestions from users providing details of the kind of haiku they want.\n"
"The AI assistant will help users write haiku about particular topics and is willing to accept requests related to a specific subject or object or a more abstract request"
"based on an emotion, theme or vibe.\n"
)
criteria_queries = (
"Incorporate a diverse range of verbs, avoiding repetition.\n"
"Ensure queries are compatible with AI model's text generation functions and are limited to 1-2 sentences.\n"
"Design queries to be self-contained and standalone."
)
instruction_task = SelfInstructTask(
num_instructions=15,
application_description=application_description,
criteria_for_query_generation=criteria_queries,
)
# Let's print the generated prompt to see the input of the LLM model
print(instruction_task.generate_prompt("mountain peaks").formatted_prompt)
"""
# Task Description
Develop 15 user queries that can be received by the given AI application and applicable to the provided context. Emphasize diversity in verbs and linguistic structures within the model’s textual capabilities.
# Criteria for Queries
Incorporate a diverse range of verbs, avoiding repetition.
Ensure queries are compatible with AI model’s text generation functions and are limited to 1-2 sentences.
Design queries to be self-contained and standalone.
Write each query on a separate line and avoid using numbered lists or bullet points.
# AI Application
An AI assistant adept at writing Haiku.
It expects complete suggestions from users providing details of the kind of haiku they want.
The AI assistant will help users write haiku about particular topics and is willing to accept requests related to a specific subject or object or a more abstract requestbased on an emotion, theme or vibe.
# Context
mountain peaks
# Output
"""
EvolInstructTask¶
The task is specially designed to build the prompts following the Evol-Instruct strategy proposed in: WizardLM: Empowering Large Language Models to Follow Complex Instructions.
From the original repository:
Evol-Instruct is a novel method using LLMs instead of humans to automatically mass-produce open-domain instructions of various difficulty levels and skill range, to improve the performance of LLMs.
Use this Task
to build more complete and complex datasets starting from simple ones.
import os
from distilabel.llm import OpenAILLM
from distilabel.tasks import EvolInstructTask
generator = OpenAILLM(
task=EvolInstructTask(),
openai_api_key=os.getenv("OPENAI_API_KEY"),
)
You can take a look at a sample dataset generated using the script the following script: examples/pipeline-evol-instruct-alpaca.py.
Note
The original definition of EvolInstruct
considers an elimination evolving step with different
situations to remove the responses considered failures. Section 3.2, Elimination Evolving in WizardLM paper shows these steps. We have implemented steps 2-4 as part of this task, but not step one. Step 1 of the elimination process can be implemented using labellers in distilabel
, an example implementation can be found in the following script: examples/pipeline-openai-wizardl-equal-prompts.py.
For the API reference visit EvolInstructTask.
Labelling¶
Instead of generating text, you can instruct the LLM
to label datasets. The existing tasks are designed specifically for creating both PreferenceTask
and CritiqueTask
datasets.
Preference¶
Preference datasets for Language Models (LLMs) are sets of information that show how people rank or prefer one thing over another in a straightforward and clear manner. These datasets help train language models to understand and generate content that aligns with user preferences, enhancing the model's ability to generate contextually relevant and preferred outputs.
Contrary to the TextGenerationTask
, the PreferenceTask
is not intended for direct use. It implements the default methods input_args_names
and output_args_names
, but generate_prompt
and parse_output
are specific to each PreferenceTask
. Examining the output_args_names
reveals that the generation will encompass both the rating and the rationale that influenced that rating.
UltraFeedbackTask¶
This task is specifically designed to build the prompts following the format defined in the "UltraFeedback: Boosting Language Models With High Quality Feedback" paper.
From the original repository: To collect high-quality preference and textual feedback, we design a fine-grained annotation instruction, which contains 4 different aspects, namely instruction-following, truthfulness, honesty and helpfulness. This Task
is designed to label datasets following the different aspects defined for the UltraFeedback dataset creation.
The following snippet can be used as a simplified UltraFeedback Task, for which we define 3 different ratings, but take into account the predefined versions are intended to be used out of the box:
from textwrap import dedent
from distilabel.tasks.preference.ultrafeedback import Rating, UltraFeedbackTask
task_description = dedent(
"""
# General Text Quality Assessment
Evaluate the model's outputs based on various criteria:
1. **Correctness & Informativeness**: Does the output provide accurate and helpful information?
2. **Honesty & Uncertainty**: How confidently does the model convey its information, and does it express uncertainty appropriately?
3. **Truthfulness & Hallucination**: Does the model introduce misleading or fabricated details?
4. **Instruction Following**: Does the model's output align with given instructions and the user's intent?
Your role is to provide a holistic assessment considering all the above factors.
**Scoring**: Rate outputs 1 to 3 based on the overall quality, considering all aspects:
"""
)
ratings = [
Rating(value=1, description="Low Quality"),
Rating(value=2, description="Moderate Quality"),
Rating(value=3, description="Good Quality"),
]
ultrafeedback_task = UltraFeedbackTask(
system_prompt="Your role is to evaluate text quality based on given criteria",
task_description=task_description,
ratings=ratings,
)
The following example creates a UltraFeedback task to emphasize helpfulness, that is overall quality and correctness of the output:
The following example creates a UltraFeedback task to emphasize truthfulness and hallucination assessment:
The following example creates a UltraFeedback task to emphasize honesty and uncertainty expression assessment:
The following example creates a UltraFeedback task to emphasize the evaluation of alignment between output and intent:
Additionally, we at Argilla created a custom subtask for UltraFeedback, that generates an overall score evaluating all the aspects mentioned above but within a single subtask. Otherwise, in order to get an overall score, all the subtasks above should be run and the average of those scores to be calculated.
The following example uses a LLM
to examinate the data for our custom overall quality criteria, which includes the different criteria from UltraFeedback (Correctness & Informativeness, Honesty & Uncertainty, Truthfulness & Hallucination and Instruction Following):
For the API reference visit UltraFeedbackTask.
JudgeLMTask¶
The task specially designed to build the prompts following the UltraFeedback paper: JudgeLM: Fine-tuned Large Language Models Are Scalable Judges. This task is designed to evaluate the performance of AI assistants.
import os
from distilabel.llm import OpenAILLM
from distilabel.tasks import JudgeLMTask
labeller = OpenAILLM(task=JudgeLMTask(), openai_api_key=os.getenv("OPENAI_API_KEY"))
For the API reference visit JudgeLMTask.
UltraJudgeTask¶
This class implements a PreferenceTask
specifically for a better evaluation using AI Feedback. The task is defined based on both UltraFeedback and JudgeLM, but with several improvements / modifications.
It introduces an additional argument to differentiate various areas for processing. While these areas can be customized, the default values are as follows:
from distilabel.tasks import UltraJudgeTask
# To see the complete system_prompt and task_description please take a look at the UltraJudgeTask definition
ultrajudge_task = UltraJudgeTask(
system_prompt="You are an evaluator tasked with assessing AI assistants' responses from the perspective of typical user preferences...",
task_description="Your task is to rigorously evaluate the performance of...",
areas=[
"Practical Accuracy",
"Clarity & Transparency",
"Authenticity & Reliability",
"Compliance with Intent",
],
)
Which can be directly used in the following way:
import os
from distilabel.llm import OpenAILLM
from distilabel.tasks import UltraJudgeTask
labeller = OpenAILLM(task=UltraJudgeTask(), openai_api_key=os.getenv("OPENAI_API_KEY"))
For the API reference visit UltraJudgeTask.
Critique¶
The CritiqueTask
is designed to be a labeller for generated text, while not only adding scores based on a rubric, but also critiques explaining the reasons why those scores have been provided. The critique can either be using a reference answer (gold answer) as e.g. Prometheus does, or just by generating the critique per each of the N provided generations.
The resulting datasets after running a pipeline with the CritiqueTask
are useful towards either training a model to generate critiques based on the critiques generated by a more powerful model as e.g. GPT-4 from OpenAI, or to be used directly for DPO fine-tuning. The fact that the critique is generated per each pair, a balanced dataset could be generated with individual critiques and their scores, so that then we can e.g. define a threshold on what's considered chosen and rejected, to then run DPO fine-tunes.
While the CritiqueTask
may seem fairly similar to the PreferenceTask
, there is a core difference, which is the fact that the critiques are provided per each response or even to a single response, with no need to compare or rate them against each other.
UltraCMTask¶
This task is specifically designed to build the prompts following the format defined in the "UltraFeedback: Boosting Language Models With High Quality Feedback" paper.
UltraCM is a model that has been fine-tuned using the UltraFeedback dataset, so as to produce critiques for the generated content, as the authors claim in their paper: "Moreover, since ULTRAFEEDBACK provides detailed textual feedback, we also fine-tune a model that could critique model responses automatically. Our critique model, UltraCM, generates reasonable and detailed comments on various tasks.".
Ideally, the UltraCMTask
will be more consistent when used with either their fine-tuned model UltraCM or with OpenAI, as both have been proven to produce successfully the structured content following the prompt formatting, and not only structured, but also meaningful and reasonable.
See the following snippet, with an example on how to instantiate the UltraCMTask
which only requires the system prompt, and it can be modified based on how is the critique intended to be formulated, while the system prompt shown below is the default one as of the UltraFeedback paper.
from distilabel.tasks import UltraCMTask
task = UltraCMTask(
system_prompt="User: A one-turn chat between a curious user and an artificial intelligence assistant. The assistant gives helpful, very detailed, and polite answers to the user's questions.</s>",
)
PrometheusTask¶
This task is specifically designed to build the prompts following the format defined in the "Prometheus: Inducing Fine-grained Evaluation Capability in Language Models" paper.
Ideally, the PrometheusTask
should only be used to format the prompts for the Prometheus models as those are the ones that have been fine-tuned to follow the same formatting and will produce consistent results compared to other base models or fine-tuned with different formats. In this case, since the formatting used by Prometheus follows the Llama 2 format, those are recommended. Otherwise, OpenAI has also proved to produce consistent results.
The following snippet can be used out of the box to define a simple PrometheusTask
with the system prompt, the scoring criteria and the score descriptions, but those can be modified while keeping in mind that Prometheus always expects 5 scores from 1-5 with a meaningful description, as well as with a criteria relevant to the scores defined.
from distilabel.tasks import PrometheusTask
task = PrometheusTask(
system_prompt="You are a fair evaluator language model.",
scoring_criteria="Relevance, Grammar, Informativeness, Engagement",
score_descriptions={
1: "The response is not relevant to the prompt.",
2: "The response is relevant to the prompt, but it is not grammatical.",
3: "The response is relevant to the prompt and it is grammatical, but it is not informative.",
4: "The response is relevant to the prompt, it is grammatical, and it is informative, but it is not engaging.",
5: "The response is relevant to the prompt, it is grammatical, it is informative, and it is engaging.",
},
)
-
The principles can be found here in the codebase. More information on the Principle Sampling can be found in the UltraFeedfack repository. ↩