Notes on Transformers Book Ch. 5
- Introduction
- The Challenge with Generating Coherent Text
- Greedy Search Decoding
- Beam Search Decoding
- Sampling Methods
- Top-k and Nucleus Sampling
- Which Decoding Method Is Best?
- References
import transformers
import datasets
import accelerate
# Only print error messages
transformers.logging.set_verbosity_error()
datasets.logging.set_verbosity_error()
transformers.__version__, datasets.__version__, accelerate.__version__
('4.11.3', '1.16.1', '0.5.1')
import ast
# https://astor.readthedocs.io/en/latest/
import astor
import inspect
import textwrap
def print_source(obj, exclude_doc=True):
# Get source code
= inspect.getsource(obj)
source # Remove any common leading whitespace from every line
= textwrap.dedent(source)
cleaned_source # Parse the source into an AST node.
= ast.parse(cleaned_source)
parsed
for node in ast.walk(parsed):
# Skip any nodes that are not class or function definitions
if not isinstance(node, (ast.FunctionDef, ast.ClassDef, ast.AsyncFunctionDef)):
continue
if exclude_doc and len(node.body) > 1: node.body = node.body[1:]
print(astor.to_source(parsed))
Introduction
- Transformer-based language models like GPT-2 and GPT-3 can generate text almost indistinguishable from text written by humans.
- Such models acquire a broad set of skills and pattern recognition abilities by learning to predict the next word in the text of millions of web pages.
- We can activate these skills with different kinds of input prompts.
- Language models are exposed to sequences of tasks during pretraining that we can adapt during inference.
The Challenge with Generating Coherent Text
- Converting a model’s probabilistic output to text requires a decoding method.
- The decoding process is iterative and involves significantly more computing than passing inputs once through the forward pass of a model.
- The quality and diversity of the generated text depend on the choice of decoding method and associated hyperparameters.
GPT-2 Pretraining Process
- GPT-2 is pretrained to estimate the probability \(P\left(y \vert x \right)\) of a sequence of tokens \(y = y_{1},y_{2},\ldots,y_{t}\) occurring in the text \(x = x_{1},x_{2},\ldots,x_{k}\), given some initial prompt or context sequence.
- It is common to use the chain rule of probability to factorize it as a product of conditional probabilities.
\[P\left(y_{1},\ldots,y_{t} \vert x \right) = \prod^{N}_{t=1}{P\left(y_{t} \vert y_{ \ < \ t},x \right)}\]
- where \(y_{ \ < \ t}\) is the shorthand notation for the sequence \(y_{1},\ldots,y_{t-1}\)
- We can adapt this token prediction task to generate sequences of arbitrary length by feeding the model a prompt.
- We then iteratively add the next predicted token to the prompt and feed the new prompt to the model.
- Some call this type of text generation conditional text generation since the output sequence depends on the choice of input prompt.
Decoding
- A decoding method determines which token to select at each timestep.
- The language model produces a logit \(z_{t,i}\) per token in the vocabulary at each time step.
- We can get the probability distribution over the next possible token \(w_{i}\) by taking the softmax.
\[P\left(y_{t} = w_{i} \vert y_{ \ < \ t},x \right) = softmax \left( z_{t,i} \right)\]
- Most decoder methods search for the most likely overall sequence by picking a \(\hat{y}\) such that
\[\hat{y} = \underset{y}{argmax} P\left(y \vert x \right)\]
- We use approximations for \(\hat{y}\) instead of finding it directly.
Greedy Search Decoding
- The simplest decoding method is to greedily select the token with the highest probability at each timestep.
\[\hat{y}_{t} = \underset{y}{argmax} {P\left(y_{t} \vert y_{ \ < \ t},x \right)}\]
- Greedy search decoding tends to produce repetitive output sequences.
- Greedy search can miss sequences whose overall probability is higher when low probability words precede high-probability words.
- Greedy search is not suitable for text generation tasks that require diversity.
- Greedy search is better suited for producing short sequences like arithmetic that require deterministic and factually correct output.
import torch
from transformers import AutoTokenizer, AutoModelForCausalLM
Load the 1.5-billion-parameter version of GPT-2 with a language modeling head
Note: The model takes up around 8GB of VRAM.
= "cuda" if torch.cuda.is_available() else "cpu"
device = "gpt2-xl"
model_name = AutoTokenizer.from_pretrained(model_name)
tokenizer = AutoModelForCausalLM.from_pretrained(model_name).to(device) model
import pandas as pd
'max_colwidth', None)
pd.set_option('display.max_rows', None)
pd.set_option('display.max_columns', None) pd.set_option(
Prepare Input
= "Transformers are the"
input_txt = tokenizer(input_txt, return_tensors="pt")["input_ids"].to(device) input_ids
0]) tokenizer.convert_ids_to_tokens(input_ids[
['Transform', 'ers', 'Ġare', 'Ġthe']
Perform Greedy Search Decoding
= []
iterations = 8
n_steps = 5
choices_per_step
with torch.no_grad():
for _ in range(n_steps):
= dict()
iteration "Input"] = tokenizer.decode(input_ids[0])
iteration[= model(input_ids=input_ids)
output # Select logits of the first batch and the last token and apply softmax
= output.logits[0, -1, :]
next_token_logits = torch.softmax(next_token_logits, dim=-1)
next_token_probs = torch.argsort(next_token_probs, dim=-1, descending=True)
sorted_ids # Store tokens with highest probabilities
for choice_idx in range(choices_per_step):
= sorted_ids[choice_idx]
token_id = next_token_probs[token_id].cpu().numpy()
token_prob = (
token_choice f"{tokenizer.decode(token_id)} ({100 * token_prob:.2f}%)"
)f"Choice {choice_idx+1}"] = token_choice
iteration[# Append predicted next token to input
= torch.cat([input_ids, sorted_ids[None, 0, None]], dim=-1)
input_ids
iterations.append(iteration)
pd.DataFrame(iterations)
Input | Choice 1 | Choice 2 | Choice 3 | Choice 4 | Choice 5 | |
---|---|---|---|---|---|---|
0 | Transformers are the | most (8.53%) | only (4.96%) | best (4.65%) | Transformers (4.37%) | ultimate (2.16%) |
1 | Transformers are the most | popular (16.78%) | powerful (5.37%) | common (4.96%) | famous (3.72%) | successful (3.20%) |
2 | Transformers are the most popular | toy (10.63%) | toys (7.23%) | Transformers (6.60%) | of (5.46%) | and (3.76%) |
3 | Transformers are the most popular toy | line (34.38%) | in (18.20%) | of (11.71%) | brand (6.10%) | line (2.69%) |
4 | Transformers are the most popular toy line | in (46.28%) | of (15.09%) | , (4.94%) | on (4.40%) | ever (2.72%) |
5 | Transformers are the most popular toy line in | the (65.99%) | history (12.42%) | America (6.91%) | Japan (2.44%) | North (1.40%) |
6 | Transformers are the most popular toy line in the | world (69.26%) | United (4.55%) | history (4.29%) | US (4.23%) | U (2.30%) |
7 | Transformers are the most popular toy line in the world | , (39.73%) | . (30.64%) | and (9.87%) | with (2.32%) | today (1.74%) |
Note: The generated sentence indicates that GPT-2 internalized some knowledge about the Transformers media franchise during pretraining.
GenerationMixin.generate
- Documentation
- Generates sequences of token ids for models with a language modeling head.
- greedy decoding:
num_beams=1
anddo_sample=False
- multinomial decoding:
num_beams=1
anddo_sample=True
- beam-search decoding:
num_beams>1
anddo_sample=False
- beam-search multinomial sampling:
num_beams>1
anddo_sample=True
- diverse beam-search decoding:
num_beams>1
and `num_beam_groups>1
- constrained beam-search decoding:
constraints!=None
orforce_words_ids!=None
Perform Greedy Search Decoding with the generate()
function
= tokenizer(input_txt, return_tensors="pt")["input_ids"].to(device)
input_ids = model.generate(input_ids, max_new_tokens=n_steps, do_sample=False)
output print(tokenizer.decode(output[0]))
Transformers are the most popular toy line in the world,
Try to perform arithmetic with Greedy Search Decoding
= 20
max_length = """5 + 8 => 13 \n 7 + 2 => 9 \n 1 + 0 =>"""
input_txt = tokenizer(input_txt, return_tensors="pt")["input_ids"].to(device)
input_ids = model.generate(input_ids, max_length=max_length,
output_greedy =False)
do_sampleprint(tokenizer.decode(output_greedy[0]))
5 + 8 => 13
7 + 2 => 9
1 + 0 => 1
= 20
max_length = """5 + 8 => 13 \n 7 + 2 => 9 \n 2 * 10 =>"""
input_txt = tokenizer(input_txt, return_tensors="pt")["input_ids"].to(device)
input_ids = model.generate(input_ids, max_length=max_length,
output_greedy =False)
do_sampleprint(tokenizer.decode(output_greedy[0]))
5 + 8 => 13
7 + 2 => 9
2 * 10 => 20
= 20
max_length = """5 + 8 => 13 \n 7 + 2 => 9 \n 2 * 13 =>"""
input_txt = tokenizer(input_txt, return_tensors="pt")["input_ids"].to(device)
input_ids = model.generate(input_ids, max_length=max_length,
output_greedy =False)
do_sampleprint(tokenizer.decode(output_greedy[0]))
5 + 8 => 13
7 + 2 => 9
2 * 13 => 13
Note: Not perfect.
Try to replicate the OpenAI Unicorn story with Greedy Search Decoding
= 128
max_length = """In a shocking finding, scientist discovered \
input_txt a herd of unicorns living in a remote, previously unexplored \
valley, in the Andes Mountains. Even more surprising to the \
researchers was the fact that the unicorns spoke perfect English.\n\n
"""
= tokenizer(input_txt, return_tensors="pt")["input_ids"].to(device)
input_ids = model.generate(input_ids, max_length=max_length,
output_greedy =False)
do_sampleprint(tokenizer.decode(output_greedy[0]))
In a shocking finding, scientist discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English.
The researchers, from the University of California, Davis, and the University of Colorado, Boulder, were conducting a study on the Andean cloud forest, which is home to the rare species of cloud forest trees.
The researchers were surprised to find that the unicorns were able to communicate with each other, and even with humans.
The researchers were surprised to find that the unicorns were able
Note: The results demonstrate the repetitive output that is characteristic of greedy search decoding.
Beam Search Decoding
- Beam search keeps track of the \(top-b\) most probable next tokens, where \(b\) is the number of beams or partial hypotheses.
- There is a tradeoff between output quality and speed when choosing the number of beams.
- We choose the next set of beams by considering all possible next-token extensions of the existing ones and selecting the \(b\) most likely extensions.
- We repeat this process until we reach the maximum length or an EOS token.
- We select the most likely sequence by ranking the \(b\) beams according to their log probabilities.
\[\log{P\left(y_{1},\ldots,y_{t} \vert x \right)} = \sum^{N}_{t=1}{\log{P\left(y_{t} \vert y_{ \ < \ t},x \right)}}\]
Note: We use the log probabilities to avoid numerical instability due to floating-point precision.
0.5 ** 1024
5.562684646268003e-309
import numpy as np
sum([np.log(0.5)] * 1024)
-709.7827128933695
import torch.nn.functional as F
log_softmax
- Documentation
- Mathematically equivalent to
log(softmax(x))
\[\text{LogSoftmax}(x_{i}) = \log\left(\frac{\exp(x_i)}{ \sum_j \exp(x_j)} \right)\]
Define a function to calculate the log probability of a single token
def log_probs_from_logits(logits, labels):
# Normalize the logits with softmax before taking the log
= F.log_softmax(logits, dim=-1)
logp = torch.gather(logp, 2, labels.unsqueeze(2)).squeeze(-1)
logp_label return logp_label
Define a function to calculate the log probabilities of a sequence
def sequence_logprob(model, labels, input_len=0):
with torch.no_grad():
= model(labels)
output = log_probs_from_logits(
log_probs # We dont need the last logit since we don't have a ground truth token for it
# We don't have a logit for the first token
-1, :], labels[:, 1:])
output.logits[:, :# Sum the log probabilities for each token
# Ignore the log probabilities of the input sequence
= torch.sum(log_probs[:, input_len:])
seq_log_prob return seq_log_prob.cpu().numpy()
Compare the log probabilities texts generated by greedy and beam search
= sequence_logprob(model, output_greedy, input_len=len(input_ids[0]))
logp print(tokenizer.decode(output_greedy[0]))
print(f"\nlog-prob: {logp:.2f}")
In a shocking finding, scientist discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English.
The researchers, from the University of California, Davis, and the University of Colorado, Boulder, were conducting a study on the Andean cloud forest, which is home to the rare species of cloud forest trees.
The researchers were surprised to find that the unicorns were able to communicate with each other, and even with humans.
The researchers were surprised to find that the unicorns were able
log-prob: -87.43
= model.generate(input_ids, max_length=max_length, num_beams=5,
output_beam =False)
do_sample= sequence_logprob(model, output_beam, input_len=len(input_ids[0]))
logp print(tokenizer.decode(output_beam[0]))
print(f"\nlog-prob: {logp:.2f}")
In a shocking finding, scientist discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English.
The discovery of the unicorns was made by a team of scientists from the University of California, Santa Cruz, and the National Geographic Society.
The scientists were conducting a study of the Andes Mountains when they discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English
log-prob: -55.23
Note: * A higher log probability is better. * Beam search still suffers from repetitive text. * We can impose an n-gram penalty that tracks which n-grams are already present in the output.
Test beam search with an n-gram penalty
= model.generate(input_ids, max_length=max_length, num_beams=5,
output_beam =False, no_repeat_ngram_size=2)
do_sample= sequence_logprob(model, output_beam, input_len=len(input_ids[0]))
logp print(tokenizer.decode(output_beam[0]))
print(f"\nlog-prob: {logp:.2f}")
In a shocking finding, scientist discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English.
The discovery was made by a team of scientists from the University of California, Santa Cruz, and the National Geographic Society.
According to a press release, the scientists were conducting a survey of the area when they came across the herd. They were surprised to find that they were able to converse with the animals in English, even though they had never seen a unicorn in person before. The researchers were
log-prob: -93.12
Note: * There are no repetitions, and the text remains coherent despite a lower log probability. * We can use beam search with an n-gram penalty to find a tradeoff between focusing on high-probability tokens while reducing repetitions.
Sampling Methods
- The simplest sampling method is to randomly sample from the probability distribution of the model’s outputs over the entire vocabulary at each timestep.
\[P\left(y_{t} = w_{i} \vert y_{ \ < \ t},x \right) = \text{softmax} \left( z_{t,i} \right) = \frac{\exp(z_{t,i})}{ \sum^{|V|}_{j=1} \exp(z_{t,j})}\]
- where \(\vert V \vert\) denotes the cardinality of the vocabulary
- We can control the diversity of the output by adding a temperature parameter \(T\) that rescales the logits before taking the softmax.
\[\left(y_{t} = w_{i} \vert y_{ \ < \ t},x \right) = \text{softmax} \left( z_{t,i} \right) = \frac{\frac{\exp(z_{t,i})}{T}}{ \sum^{|V|}_{j=1} \frac{\exp(z_{t,j}}{T})}\]
- We can tune the temperature parameter to control the shape of the probability distribution.
- A \(T\) value much less than \(1\) suppresses the rare tokens.
- A \(T\) value much greater than \(1\) causes each token to become equally likely.
import matplotlib.pyplot as plt
Define a softmax function with a temperature parameter
def softmax(logits, T=1):
= np.exp(logits / T)
e_x return e_x / e_x.sum()
Plot a distribution of randomly generated token probabilities for three selected temperatures
= np.exp(np.random.random(1000))
logits = np.sort(logits)[::-1]
sorted_logits = np.arange(1000)
x
for T in [0.5, 1.0, 2.0]:
=f"T={T}")
plt.step(x, softmax(sorted_logits, T), label="best")
plt.legend(loc"Sorted token probabilities")
plt.xlabel("Probability")
plt.ylabel( plt.show()
Reset random seed
42); torch.manual_seed(
Sample generated text with a high temperature
= model.generate(input_ids, max_length=max_length, do_sample=True,
output_temp =2.0, top_k=0)
temperatureprint(tokenizer.decode(output_temp[0]))
In a shocking finding, scientist discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English.
While the station aren protagonist receive Pengala nostalgiates tidbitRegarding Jenny loclonju AgreementCON irrational �rite Continent seaf A jer Turner Dorbecue WILL Pumpkin mere Thatvernuildagain YoAniamond disse * Runewitingkusstemprop});b zo coachinginventorymodules deflation press Vaticanpres Wrestling chargesThingsctureddong Ty physician PET KimBi66 graz Oz at aff da temporou MD6 radi iter
Note: Sampling with a high temperature produces gibberish.
Reset random seed
42); torch.manual_seed(
Sample generated text with a low temperature
= model.generate(input_ids, max_length=max_length, do_sample=True,
output_temp =0.5, top_k=0)
temperatureprint(tokenizer.decode(output_temp[0]))
In a shocking finding, scientist discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English.
The scientists were searching for the source of the mysterious sound, which was making the animals laugh and cry.
The unicorns were living in a remote valley in the Andes mountains
'When we first heard the noise of the animals, we thought it was a lion or a tiger,' said Luis Guzman, a researcher from the University of Buenos Aires, Argentina.
'But when
Note: Sampling with a low temperature produces a much more coherent output.
Top-k and Nucleus Sampling
Reset random seed
42); torch.manual_seed(
= """In a shocking finding, scientist discovered \
input_txt a herd of unicorns living in a remote, previously unexplored \
valley, in the Andes Mountains. Even more surprising to the \
researchers was the fact that the unicorns spoke perfect English.\n\n
"""
= tokenizer(input_txt, return_tensors="pt")["input_ids"].to(device) input_ids
Calculate the probability distribution of the model’s outputs at \(T=1\)
with torch.no_grad():
= model(input_ids=input_ids)
output = output.logits[:, -1, :]
next_token_logits = F.softmax(next_token_logits, dim=-1).detach().cpu().numpy() probs
Plot the cumulative probability distribution of the model’s outputs at \(T=1\)
= plt.subplots(1, 2, figsize=(10, 3.5))
fig, axes
0].hist(probs[0], bins=np.logspace(-10, -1, 100), color="C0", edgecolor="C0")
axes[0].set_xscale("log")
axes[0].set_yscale("log")
axes[0].set_title("Probability distribution")
axes[0].set_xlabel("Probability")
axes[0].set_ylabel("Count")
axes[
1].plot(np.cumsum(np.sort(probs[0])[::-1]), color="black")
axes[1].set_xlim([0, 10000])
axes[1].set_ylim([0.75, 1.01])
axes[1].set_title("Cumulative probability")
axes[1].set_ylabel("Probability")
axes[1].set_xlabel("Token (descending probability)")
axes[1].minorticks_on()
axes[
= 'top-k threshold (k=2000)'
top_k_label = 'nucleus threshold (p=0.95)'
top_p_label 1].vlines(x=2000, ymin=0, ymax=2, color='C0', label=top_k_label)
axes[1].hlines(y=0.95, xmin=0, xmax=10000, color='C1', label=top_p_label, linestyle='--')
axes[1].legend(loc='lower right')
axes[ plt.tight_layout()
Note: * The histogram has a peak around 10^-8 and a second smaller peak around 10^-4, followed by a sharp drop. * The probability of picking the token with the highest likelihood is 1 in 10. * There are 50,257 tokens in GPT-2’s vocabulary. * The curved line in the Cumulative probability chart represents the probability of picking any of the preceding tokens. * There is a 1 in 100 chance of not picking any of the tokens that are not n the top 2000. * There is a significant chance of picking an unlikely token when sampling hundreds of times. * Picking such tokens can negatively impact the quality of the generated text.
Top-k Sampling
- The idea behind top-k sampling is to avoid low-probability choices by only choosing from the k tokens with the highest probability.
- We can find a good value for k by looking at some text quality metrics.
Reset random seed
42); torch.manual_seed(
Generate text using the 50 tokens with the highest probability
= model.generate(input_ids, max_length=max_length, do_sample=True,
output_topk =50)
top_kprint(tokenizer.decode(output_topk[0]))
In a shocking finding, scientist discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English.
The wild unicorns roam the Andes Mountains in the region of Cajamarca, on the border with Argentina (Picture: Alamy/Ecole Nationale Supérieure d'Histoire Naturelle)
The researchers came across about 50 of the animals in the valley. They had lived in such a remote and isolated area at that location for nearly a thousand years that
### Nucleus (top-p) Sampling
* The idea behind nucleus sampling is to cut off the long tail of the distribution after reaching a certain probability mass in the selection.
* We order all tokens in descending order by probability and add one token after another from the top of the list until the sum of the probabilities of the selected tokens reaches the target mass.
Reset random seed
42); torch.manual_seed(
Generate text using top-p sampling
= model.generate(input_ids, max_length=max_length, do_sample=True,
output_topp =0.90)
top_pprint(tokenizer.decode(output_topp[0]))
In a shocking finding, scientist discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English.
The scientists studied the DNA of the animals and came to the conclusion that the herd are descendants of a prehistoric herd that lived in Argentina about 50,000 years ago.
According to the scientific analysis, the first humans who migrated to South America migrated into the Andes Mountains from South Africa and Australia, after the last ice age had ended.
Since their migration, the animals have been adapting to
Note: Top-p sampling also produces a coherent story.
Reset random seed
42); torch.manual_seed(
Generate text using top-k and top-p sampling
= model.generate(input_ids, max_length=max_length, do_sample=True,
output_topp =50, top_p=0.90)
top_kprint(tokenizer.decode(output_topp[0]))
In a shocking finding, scientist discovered a herd of unicorns living in a remote, previously unexplored valley, in the Andes Mountains. Even more surprising to the researchers was the fact that the unicorns spoke perfect English.
The scientists studied the DNA of the animals and came to the conclusion that the herd are descendants of a prehistoric herd that lived in Argentina about 50,000 years ago.
According to the scientific analysis, the first humans who migrated to South America migrated into the Andes Mountains from South Africa and Australia, after the last ice age had ended.
Since their migration, the animals have been adapting to
Which Decoding Method Is Best?
- The best approach depends on the nature of the task.
- Lower the temperature or use deterministic methods to perform a precise task like arithmetic or providing an answer to a specific question.
- Switch to sampling methods and increase the temperature when you want the model to generate longer text and be more creative.
References
Previous: Notes on Transformers Book Ch. 4
Next: Notes on Transformers Book Ch. 6
- I’m Christian Mills, a deep learning consultant specializing in computer vision and practical AI implementations.
- I help clients leverage cutting-edge AI technologies to solve real-world problems.
- Learn more about me or reach out via email at [email protected] to discuss your project.