Skip to content

Commit

Permalink
Change guide to HuggingFace model benchmarking (#133)
Browse files Browse the repository at this point in the history
Now advertises memos as the canonical way of using caching+lazy loading
in combination for memory efficiency.

Changes the example structure to three standalone Python files instead
of a project. This is because we disentangled the files by removing the
argument parser, and trivialized the runner.py file.

Changes the guide name to mention HuggingFace as the tech of interest in
this example.
  • Loading branch information
nicholasjng authored Mar 27, 2024
1 parent 7df827d commit 4de465a
Show file tree
Hide file tree
Showing 11 changed files with 115 additions and 183 deletions.
109 changes: 0 additions & 109 deletions docs/tutorials/artifact_benchmarking.md

This file was deleted.

107 changes: 107 additions & 0 deletions docs/tutorials/huggingface.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
# Benchmarking HuggingFace models on a dataset
There is a high likelihood that you, at some point, find yourself wanting to benchmark previously trained models.
This guide shows you how to do it for a HuggingFace model with nnbench.

## Example: Named Entity Recognition
We start with a small tangent about the example setup that we will use in this guide.
If you are only interested in the application of nnbench, you can skip this section.

There are lots of reasons why you could want to retrieve saved models for benchmarking.
Among them these are reviewing the work of colleagues, comparing model performance to an existing benchmark, or dealing with models that require significant compute such that in-place retraining is impractical.

For this example, we look at a named entity recognition (NER) model that is based on the pre-trained encoder-decoder transformer [BERT](https://arxiv.org/abs/1810.04805) from HuggingFace.
The model is trained on the [CoNLLpp dataset](https://huggingface.co/datasets/conllpp) which consists of sentences from news stories where words were tagged with Person, Organization, Location, or Miscellaneous if they referred to entities.
Words are assigned an out-of-entity label if they do not represent an entity.

## Model Training
You find the code to train the model in the nnbench [repository](https://github.com/aai-institute/nnbench/tree/main/examples/huggingface).
If you want to skip running the training script but still want to reproduce this example, you can take any BERT model fine tuned for NER with the CoNLL dataset family.
You find many on the Huggingface model hub, for example [this one](https://huggingface.co/dslim/bert-base-NER). You need to download the `model.safetensors`, `config.json`, `tokenizer_config.json`, and `tokenizer.json` files.
If you want to train your own model, continue below.

There is some necessary preprocessing and data wrangling to train the model.
We will not go into the details here, but if you are interested in a more thorough walkthrough, look into this [resource](https://huggingface.co/learn/nlp-course/chapter7/2?fw=pt) by Huggingface which served as the basis for this example.

It is not feasible to train the model on a CPU. If you do not have access to a GPU, you can use free GPU instances on [Google Colab](https://colab.research.google.com/).
When opening a new Colab notebook, make sure to select a GPU instance in the upper right corner.
Then, you can upload the `training.py`. You can ignore any warnings about the data not being persisted.

Next, install the necessary dependencies: `!pip install datasets transformers[torch]`. Google Colab comes with some dependencies already installed in the environment.
Hence, if you are working with a different GPU instance, make sure to install everything from the `pyproject.toml` in the `examples/artifact_benchmarking` folder.

Finally, you can execute the `training.py` with `!python training.py`.
This will train two BERT models ("bert-base-uncased" and "distilbert-base-uncased") which we can compare using nnbench.
If you want, you can adapt the training script to train other models by editing the tuples in the `tokenizers_and_models` list at the bottom of the training script.
The training of the models takes around 10 minutes.

Once it is done, download the respective files and save them to your disk.
They should be the same mentioned above. We will need the paths to the files for benchmarking later.

## The benchmarks

The benchmarking code is found in the `examples/huggingface/benchmark.py`.
We calculate precision, recall, accuracy, and f1 scores for the whole test set and specific labels.
Additionally, we obtain information about the model such as its memory footprint and inference time.

We are not walking through the whole file but instead point out certain design choices as an inspiration to you.
If you are interested in a more detailed walkthrough on how to set up benchmarks, you can find it [here](../guides/benchmarks.md).

Notable design choices in this benchmark are that we factored out the evaluation loop as it is necessary for all evaluation metrics.
We cache it using the `functools.cache` decorator so the evaluation loop runs only once per benchmark run instead of once per metric which greatly reduces runtime.

We also use `nnbench.parametrize` to get the per-class metrics.
As the parametrization method needs the same arguments for each benchmark, we use Python's builtin `functools.partial` to fill the arguments.

```python
--8<-- "examples/huggingface/benchmark.py:131:139"
```

!!! Tip
In this parametrization, the model path is hardcoded to "dslim/distilbert-NER" on the HuggingFace hub.
When benchmarking other models, be sure to change this path to the actual model you want to benchmark.

After this, the benchmarking code is actually very simple, as in most of the other examples.
You find it in the nnbench repository in `examples/huggingface/runner.py`.

## Custom memo classes

The parametrization contains a list of models, which are each instances of a `TokenClassificationModelMemo` a custom class we implemented which inherits from the `nnbench.Memo` class.
A big advantage of a memo in this case is its ability to lazy-load models and later evict the loaded models again from a cache.

```python
--8<-- "examples/huggingface/benchmark.py:23:35"
```

The `Memo` class is a generic wrapper around serialized data of any kind.
It allows for lazy deserialization of artifacts from uniquely identifying metadata like storage paths, checksums, or model names on HuggingFace Hub in our case.
In our derived class, we have to override the `Memo.__call__()` method to properly load the memoized value into memory.

We do similar with the CoNLLpp dataset.

```python
--8<-- "examples/huggingface/benchmark.py:51:64"
```

In this case, we lazy-load the `datasets.Dataset` object.
In the following `IndexLabelMapMemo` class, we store a dictionary mapping the label ID to a semantic string.

```python
--8<-- "examples/huggingface/benchmark.py:67:82"
```

!!! Info
There is no need to type-hint `TokenClassificationModelMemo`s in the corresponding benchmarks -
the benchmark runner takes care of filling in the memoized values for the memos themselves.

Because we implemented our memoized values as four different memo class types, this modularizes the benchmark input parameters -
we only need to reference memos when they are actually used. Considering the recall benchmarks:

```python
--8<-- "examples/huggingface/benchmark.py:174:204"
```

we see that the memoized `index_label_mapping` argument is only necessary in the per-class benchmark, so it is never passed to the main computation.

!!! Tip
When implementing memos for a benchmark workload, using only one value per memo at the cost of another class definition is often worth it,
since you have more direct control over what goes into your benchmarks, and you can avoid having unused parameters altogether with this approach.
2 changes: 1 addition & 1 deletion docs/tutorials/index.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,4 @@ Click any of the links below for inspiration on how to use nnbench in your proje
* [Using a streamlit web app to dispatch benchmarks](streamlit.md)
* [Analyzing benchmark results at scale with duckDB](duckdb.md)
* [Streaming benchmark results to a cloud database (Google BigQuery)](bq.md)
* [How to benchmark previously trained models via artifacts](artifact_benchmarking.md)
* [How to benchmark pre-trained HuggingFace models with memos](huggingface.md)
19 changes: 0 additions & 19 deletions examples/artifact_benchmarking/pyproject.toml

This file was deleted.

47 changes: 0 additions & 47 deletions examples/artifact_benchmarking/requirements.txt

This file was deleted.

Empty file.
Empty file.
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
from datasets import Dataset, load_dataset
from torch.nn import Module
from torch.utils.data import DataLoader
from training.training import tokenize_and_align_labels
from training import tokenize_and_align_labels
from transformers import (
AutoModelForTokenClassification,
AutoTokenizer,
Expand All @@ -17,14 +17,14 @@
from transformers.tokenization_utils_base import PreTrainedTokenizerBase

import nnbench
from nnbench.types import Memo
from nnbench.types import Memo, cached_memo


class TokenClassificationModelMemo(Memo[Module]):
def __init__(self, path: str):
self.path = path

@cache
@cached_memo
def __call__(self) -> Module:
model: Module = AutoModelForTokenClassification.from_pretrained(
self.path, use_safetensors=True
Expand All @@ -39,7 +39,7 @@ class TokenizerMemo(Memo[PreTrainedTokenizerBase]):
def __init__(self, path: str):
self.path = path

@cache
@cached_memo
def __call__(self) -> PreTrainedTokenizerBase:
tokenizer: PreTrainedTokenizerBase = AutoTokenizer.from_pretrained(self.path)
return tokenizer
Expand All @@ -53,7 +53,7 @@ def __init__(self, path: str, split: str):
self.path = path
self.split = split

@cache
@cached_memo
def __call__(self) -> Dataset:
dataset = load_dataset(self.path)
path = dataset.cache_files[self.split][0]["filename"]
Expand All @@ -69,7 +69,7 @@ def __init__(self, path: str, split: str):
self.path = path
self.split = split

@cache
@cached_memo
def __call__(self) -> dict[int, str]:
dataset = load_dataset(self.path)
path = dataset.cache_files[self.split][0]["filename"]
Expand Down
File renamed without changes.
File renamed without changes.
2 changes: 1 addition & 1 deletion mkdocs.yml
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ nav:
- guides/transforms.md
- Examples:
- tutorials/index.md
- tutorials/artifact_benchmarking.md
- tutorials/huggingface.md
- tutorials/mnist.md
- tutorials/prefect.md
- tutorials/streamlit.md
Expand Down

0 comments on commit 4de465a

Please sign in to comment.