Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Instruction-tuning Support #196

Open
wants to merge 81 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 62 commits
Commits
Show all changes
81 commits
Select commit Hold shift + click to select a range
59f0191
Add example config for the construction of chat templates
rrutmann Jul 15, 2024
8b60a83
chore: add chat template config based on jinja2
lllAlexanderlll Jul 15, 2024
ba2f65c
chore: update chat template config based on jinja2
lllAlexanderlll Jul 15, 2024
47e71c3
chore: Add apply chat template feature with role mapping
lllAlexanderlll Jul 15, 2024
3303147
chore: extend to multiple chat templates
lllAlexanderlll Jul 16, 2024
0c6bbf5
fix: data driven chat tempalte key retrieval
lllAlexanderlll Jul 16, 2024
32f5756
chore: Add 'index' to output JSONL
lllAlexanderlll Jul 16, 2024
482f7af
fix: Add süecical token to be kept during treinaing to allow for earl…
lllAlexanderlll Jul 16, 2024
1d72770
chore: Update output file
lllAlexanderlll Jul 16, 2024
0bd9bfa
build: Add jsonlines dependency
rrutmann Jul 16, 2024
ed2f4ce
chore: integration of collator wrapper with loss masking functionalit…
lllAlexanderlll Jul 16, 2024
6e24ea2
chore: Use SFT config replaction with uuid as file pair identification.
lllAlexanderlll Jul 18, 2024
6e716b4
chore: Add loss masking test
lllAlexanderlll Jul 18, 2024
a0376a6
chore: Merge branch 'main' into sft_loss_masking
lllAlexanderlll Jul 18, 2024
70dc498
fix: copy raw config file for truly original content
lllAlexanderlll Jul 19, 2024
242e429
chore: add pbin file for testing loss masking
Jul 22, 2024
bddcf8b
chore: add pbin file with more data for testing loss masking
Jul 22, 2024
f86b6ed
chore: use a hash not uuid for showing which config belongs to whoch …
lllAlexanderlll Jul 22, 2024
7632a02
chore: add pbin file for testing loss masking
Jul 22, 2024
15719a3
chore: Fix loss masking when starting within an assistant answer
lllAlexanderlll Jul 22, 2024
ab0f34c
chore: add lost collator wrappr again
lllAlexanderlll Jul 22, 2024
0a545ca
chore: fix the loss masking test and the implementation. Improve docu…
lllAlexanderlll Jul 22, 2024
c6d0a61
chore: Merge commit '15ed069beaa2c83dcd15b087e4d0864b1aec4caa' into s…
lllAlexanderlll Jul 22, 2024
0c52856
chore: Merge branch 'sft_with_main' into sft_loss_masking
lllAlexanderlll Jul 22, 2024
12c74bc
feat(sft): Do not reuse last targets for Instruction Tuning
rrutmann Jul 23, 2024
fc7bec1
Merge remote-tracking branch 'origin/sft_with_main' into sft_sample_g…
rrutmann Jul 23, 2024
25fdcd7
refactor(sft): Make reuse_last_target optional
rrutmann Jul 23, 2024
01109e2
docs: Correct spelling
rrutmann Jul 23, 2024
7148e1e
Update comment
rrutmann Jul 23, 2024
75611dd
Merge pull request #193 from Modalities/sft_sample_generator
rrutmann Jul 23, 2024
1f36d64
chore: apply review changes: only single chat template, do raise erro…
lllAlexanderlll Jul 23, 2024
9921ed3
chore: Merge branch 'sft_with_main' into sft_loss_masking
lllAlexanderlll Jul 23, 2024
0f49355
chore: run loss masking with padded and truncated pbin file. Refine e…
lllAlexanderlll Jul 23, 2024
bf2f1a3
chore: restore original lorem ipsum config, as we have our own sft co…
lllAlexanderlll Jul 23, 2024
99e7ac3
Merge pull request #192 from Modalities/sft_loss_masking
lllAlexanderlll Jul 24, 2024
76b34ab
feat: added fixed number of elements to ResumableBatchSampler
le1nux Jul 11, 2024
2f4bdb2
feat: added fixed number of batches to dataloader
le1nux Jul 11, 2024
9840e0a
fix: fixed error in fixed_num_batches calculation
le1nux Jul 17, 2024
3906845
feat: implemented test for fixed_num_batches in dataloader
le1nux Jul 17, 2024
cc57125
refactor: removed fixed_num_batches from config_lorem_ipsum.yaml
le1nux Jul 17, 2024
603b367
refactor: moved SequenceDataset to test
le1nux Jul 19, 2024
78f4d03
refactor: added another check that fixed_num_batches > skip_num_batches
le1nux Jul 19, 2024
8372b21
fix: check for fixed_num_batches is not None before comparison
lllAlexanderlll Jul 29, 2024
c8ba69c
chore: add missing reuse_last_target: False in example SFT config
lllAlexanderlll Jul 29, 2024
60e9894
chore: Merge branch 'main' into sft_with_main
le1nux Jul 30, 2024
b0c69a9
fix: removed sequence_length from MemMapDataset
le1nux Jul 31, 2024
d041900
chore: added reraise of exception in PackedMemMapDatasetBase
le1nux Jul 31, 2024
3912e36
chore: fixed typo
le1nux Jul 31, 2024
cc4eef7
refactor: made non-public methods related to apply_chat_template private
le1nux Jul 31, 2024
00a35ee
chore: minor style improvement
le1nux Jul 31, 2024
009bb8a
chore: fix suggestions of PR first round
lllAlexanderlll Aug 5, 2024
a47dc8c
chore: Move collate functions to dataloader package
lllAlexanderlll Aug 5, 2024
705101c
chore: renamed MaskingTokenConfig to LossMaskingTokenConfig
lllAlexanderlll Aug 5, 2024
c399b94
chore: Add explaination to vectorized loss masking
lllAlexanderlll Aug 5, 2024
483ea83
chore: moved SFTConfig and tests
lllAlexanderlll Aug 5, 2024
4ea1d76
chore: update SFT README
lllAlexanderlll Aug 12, 2024
3945426
chore: test for reuse last target, update sft readme, create folder f…
lllAlexanderlll Aug 12, 2024
4d7e53d
chore: Merge branch 'main' into sft_with_main
lllAlexanderlll Aug 12, 2024
eee2bac
chore: fix tokenization tests and renaming of loss masking config field
lllAlexanderlll Aug 12, 2024
4f53f0c
chore: Update SFT_README.md
lllAlexanderlll Aug 13, 2024
ed50d2f
docs: add doc strings
lllAlexanderlll Aug 13, 2024
d5867a4
chore: update instruction tuning e2e test with output artifact check
lllAlexanderlll Aug 13, 2024
94d89cb
chore: Update readme
lllAlexanderlll Aug 13, 2024
42cf6ce
chore: refine names of helper functions and doc strings
lllAlexanderlll Aug 19, 2024
d98a26a
fix: apply renaming
lllAlexanderlll Aug 19, 2024
d95bd46
chore: Update SFT_README
lllAlexanderlll Aug 19, 2024
c6b0e4c
chore(sft): Improve check on correctness of loss masked sequences
rrutmann Aug 20, 2024
b9fbcec
chore(sft): Change special tokens used for instruction tuning
rrutmann Aug 20, 2024
6538072
chore: Add artifacts to .gitignore
rrutmann Aug 20, 2024
2748887
chore(sft): Add splitting functionality and introduce a new entry poi…
lllAlexanderlll Aug 20, 2024
72ed828
fix(sft): do not append hash twice
lllAlexanderlll Aug 20, 2024
6594633
test(sft): Use special tokens already existing in tokenizers vocabulary
rrutmann Sep 9, 2024
66f0bea
test(sft): Add data and config for tests
rrutmann Sep 9, 2024
0daec5b
test(sft): Add documentation for test
rrutmann Sep 9, 2024
125311f
chore: Pass black check
rrutmann Sep 9, 2024
9cbc8f3
chore: Merge remote-tracking branch 'origin/main' into sft_with_main
rrutmann Sep 9, 2024
b121eea
chore: improve error message and readme
lllAlexanderlll Sep 16, 2024
396aba5
chore: Update SFT_README.md
lllAlexanderlll Sep 16, 2024
8416c9d
Update SFT_README.md
lllAlexanderlll Sep 16, 2024
b08f02f
chore: Merge branch 'main' into sft_with_main
rrutmann Sep 23, 2024
eb658c9
test: fix failing sft e2e test
davidkaczer Nov 4, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
227 changes: 227 additions & 0 deletions SFT_README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,227 @@
# Supervised Fine-tuning with Modalities

Currently supported are Instruction-tuning and Low-rank Adaption (LorA), as explained in more detail next.

## Instruction-tuning
> Instruction-tuning currently only works with fast and slow Huggingface tokenizers, as the special tokens need to be added to the tokenizer.

The goal of instruction-tuning is to let the model learn instruction-following capabilites, so that it acts as an helpful assistant within an chat environment.
For this, we need to further fine-tune the model on conversational instruction data.
Specifically, we need the model to only learn to act as the assistant. Hence, we allow full attention on conversation, but calculate the loss only on the assistants untterances.

For example, we only calculate the loss for the words not struck-trough:

> ~~You are Mody, a helpful assistant trained by the modalities team. Answer friendly and informatively to the user's messages.\nUser1: What is the best way to learn a new language?\nAssistant:^~~The best way to learn a new language is to practice regularly, immerse yourself in the language, and use a variety of resources like books, apps, and language classes. It's also helpful to practice with native speakers.\n°~~$User1: Thank you for the advice.\nAssistant:^~~You're welcome! Learning a new language can be a rewarding experience. If you have any more questions, feel free to ask.\n°~~$~~

### Create Prompts from Conversations
To prepare the instruction-tuning data we created a new entry point `apply_chat_template`, which requires a [configuration file](./config_files/data_preparation/apply_chat_template_config.yaml). Within it we define:
* the path to instruction-tuning dataset as a JSONL file wereas each line contains a structured conversation as an array of dictionaries.
lllAlexanderlll marked this conversation as resolved.
Show resolved Hide resolved
* A [jinja2](https://jinja.palletsprojects.com/en/3.1.x/) chat template which defines the rules how to glue `chat_template_data` and the data within the JSONL together to one `chat` string.

As part of the `chat_template_data`, we require the special tokens `b_include_to_loss_token` and `e_include_to_loss_token`.
To prepare the instruction-tuning data we created a new entry point `apply_chat_template`, which requires a [configuration file](./config_files/data_preparation/apply_chat_template_config.yaml). Within it we define the path to instruction-tuning dataset as a JSONL file, in which each line contains a structured conversation as an array of dictionaries.

##### Example

Input JSONL file entry:
```json
{
"id": 16,
"conversations": [
{
"from": "human_1",
"value": "What is the best way to learn a new language?"
},
{
"from": "gpt",
"value": "The best way to learn a new language is to practice regularly, immerse yourself in the language, and use a variety of resources like books, apps, and language classes. It's also helpful to practice with native speakers."
},
{
"from": "human_1",
"value": "Thank you for the advice."
},
{
"from": "gpt",
"value": "You're welcome! Learning a new language can be a rewarding experience. If you have any more questions, feel free to ask."
}
]
}
```

Config:
```yaml
settings:
src_path: data/lorem_ipsum_sft.jsonl
dst_path: data/lorem_ipsum_sft_converted.jsonl
conversations_key: conversations

instruction_data_transformation:
role_mapping:
human_1: User1
human_2: User2
gpt: Assistant

...

chat_template_data:
...
special_tokens:
b_include_to_loss_token: ^
e_include_to_loss_token: $
```
lllAlexanderlll marked this conversation as resolved.
Show resolved Hide resolved

Created JSONL file entry:
```json
{
"id": 16,
"conversations": [
{
"from": "User1",
"value": "What is the best way to learn a new language?"
},
{
"from": "Assistant",
"value": "The best way to learn a new language is to practice regularly, immerse yourself in the language, and use a variety of resources like books, apps, and language classes. It's also helpful to practice with native speakers."
},
{
"from": "User1",
"value": "Thank you for the advice."
},
{
"from": "Assistant",
"value": "You're welcome! Learning a new language can be a rewarding experience. If you have any more questions, feel free to ask."
}
],
"chat": "You are Mody, a helpful assistant trained by the modalities team. Answer friendly and informatively to the user's messages.\nUser1: What is the best way to learn a new language?\nAssistant:^The best way to learn a new language is to practice regularly, immerse yourself in the language, and use a variety of resources like books, apps, and language classes. It's also helpful to practice with native speakers.\n°$User1: Thank you for the advice.\nAssistant:^You're welcome! Learning a new language can be a rewarding experience. If you have any more questions, feel free to ask.\n°$"
}
```

Run the `apply_chat_template` entry point with:
```bash
modalities data apply_chat_template --config_file_path config_files/data_preparation/apply_chat_template_config.yaml
```

This will create two files
1. The new JSONL file with a new attribute `chat` containing the conversations e.g. `lorem_ipsum_sft_40e0699/lorem_ipsum_sft_converted.40e0699.jsonl`
2. The config used to generate the `chat` e.g. `lorem_ipsum_sft_40e0699/sft_chat_template_config.40e0699.yaml`

> Both files names contain the first 7 symbols of the hash of the config file, to group files which belong together!
> Also, a new directory with the original dataset file name and the hash in it its name is created.

### Create idx and pbin files
Before continuing with the instruction-tuning you need to index the created JSONL and convert it to a tokenized binary file.

> Make sure to use the same hash for correct grouping when defining the output file names!

For example:
```bash
# create idx file
modalities data create_raw_index --index_path data/lorem_ipsum_sft_40e0699/lorem_ipsum_sft_converted.40e0699.idx data/lorem_ipsum_sft_40e0699/lorem_ipsum_sft_converted.40e0699.jsonl

# create pbin file
modalities data pack_encoded_data --config_file_path config_files/data_preparation/packed_chat_dataset_config.yaml
```

> The [packed_chat_dataset_config.yaml](config_files/data_preparation/packed_chat_dataset_config.yaml) must use truncation and padding!

In summary, the automatically created folder for all files related to the instruction-tuning data, should look as follows (the hash value might be different depending on your intial apply chat template configuration file):

> lorem_ipsum_sft_40e0699
> ├── lorem_ipsum_sft_converted.40e0699.idx
> ├── lorem_ipsum_sft_converted.40e0699.jsonl
> ├── lorem_ipsum_sft_converted.40e0699.pbin
> ├── packed_chat_dataset_config.40e0699.yaml
> └── sft_chat_template_config.40e0699.yaml

### Instruction-Tuning

With your prepared instruction-tuning data as pbin file, you can now instruction-tune.

Make sure to use the wrapped collate function.

* You need to look up the `b_include_to_loss_token` and `e_include_to_loss_token` as defined within your `sft_chat_template_config.40e0699.yaml`. If configured the pbin creation correctly, you only need to check for matching hash suffixes.
* Set the `loss_ignore_index` which gets ignored by your loss function. In torch this is usually -100.
* We need a tokenizer to tokenize the `b_include_to_loss_token` and `e_include_to_loss_token`
* We need to not re-use the last token

For example (Copied from [config_files/training/config_lorem_ipsum_sft.yaml](config_files/training/config_lorem_ipsum_sft.yaml)):
```yaml
collate_fn:
component_key: collate_fn
variant_key: mask_loss_collator_wrapper
config:
wrapped_collate_fn:
component_key: collate_fn
variant_key: gpt_2_llm_collator
config:
sample_key: ${settings.referencing_keys.sample_key}
target_key: ${settings.referencing_keys.target_key}
target_keys_to_mask:
- ${settings.referencing_keys.target_key}
loss_ignore_index: -100
mask_tokens:
b_include_to_loss_token: ^
e_include_to_loss_token: $
tokenizer:
instance_key: tokenizer
pass_type: BY_REFERENCE
```

with

```yaml
train_dataset:
component_key: dataset
variant_key: packed_mem_map_dataset_continuous
config:
raw_data_path: ./data/lorem_ipsum_sft_40e0699/lorem_ipsum_sft_converted.40e0699.pbin
sequence_length: ${settings.training.sequence_length}
sample_key: ${settings.referencing_keys.sample_key}
reuse_last_target: true
```

and with

```yaml
tokenizer:
component_key: tokenizer
variant_key: pretrained_hf_tokenizer
config:
pretrained_model_name_or_path: data/tokenizer/hf_gpt2
padding: false
truncation: false
special_tokens:
additional_special_tokens:
- "^"
- "$"
```

Finally, run the instruction-tuning with the `run` entry point:
```bash
torch.distributed.run --nnodes 1 --nproc_per_node 2 --rdzv-endpoint=0.0.0.0:29555 src/modalities/__main__.py run --config_file_path config_files/training/config_lorem_ipsum_sft.yaml
```

> Note, that it is advised to add a special token (which is already known as non-special token to the tokenizers' voabulary) to indicate the end of an assistant turn within the `b_include_to_loss_token` and `e_include_to_loss_token` in your chat template. Change your chat template accordingly and make sure to inlcude this token as special token in the tokenizer configuration for the pbin file creation step and model training!
lllAlexanderlll marked this conversation as resolved.
Show resolved Hide resolved

#### A Note on Tokanization in Huggingface
lllAlexanderlll marked this conversation as resolved.
Show resolved Hide resolved
The special tokens are added to a [Trie](https://en.wikipedia.org/wiki/Trie), so that longer special tokens are split first and then shorter special tokens.
lllAlexanderlll marked this conversation as resolved.
Show resolved Hide resolved
Example from the huggingface documentation:

```python
>>> trie = Trie()
>>> trie.split("[CLS] This is a extra_id_100")
["[CLS] This is a extra_id_100"]

>>> trie.add("[CLS]")
>>> trie.add("extra_id_1")
>>> trie.add("extra_id_100")
>>> trie.split("[CLS] This is a extra_id_100")
["[CLS]", " This is a ", "extra_id_100"]
```

When we add a special token, which exists within the tokenizer voabulary already, HF only marks it as special token (adds it to the trie).
This means, if the sequence we add as special token already exists in the vocab, there is no need to resize the embedding matrix!
Copy link
Member

@le1nux le1nux Aug 13, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

so basically we represent the special token by at least on of the original tokens in the vocabulary?

For example, assuming "spec_tok_1" is supposed to be a special token, it gets tokenized to e.g., ["spec", "_", "tok", "_", "1"] ? Since it has been added to the Trie, we therefore still know that it is a special token but during training and inference the special token would still be represented by a 5 token sequence and therefore the embedding matrix can remain unchanged, right?

What happens if the special token cannot be represented?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not quite.
Given spec_tok_1 as a special token, which is part of the trie and given the text to tokenize being Hi there spec_tok_1!:
First, the split method of the trie is used to extract all special tokens. In our case, the split would look like this: ["Hi there", "spec_tok_1", "!"], then all strings that are not part of the trie, i.e., the normal tokens, are tokenized leading to e.g. ["Hi", "there", "spec_tok_1", "!"]


## Low-rank Adaption (LorA)

TBD
42 changes: 42 additions & 0 deletions config_files/data_preparation/apply_chat_template_config.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
settings:
src_path: data/lorem_ipsum_sft.jsonl
dst_path: data/lorem_ipsum_sft_converted.jsonl
conversations_key: conversations

instruction_data_transformation:
role_mapping:
human_1: User1
human_2: User2
gpt: Assistant

# The b_include_to_loss_token, e_include_to_loss_token are required to be part of each chat template for proper loss masking!
jinja2_chat_template: |
le1nux marked this conversation as resolved.
Show resolved Hide resolved
{{ chat_template_data.system_instruction + '\n' }}
{% for turn in conversation %}
{{ turn.from + ':' }}
{% if turn.from == chat_template_data.assistant_role %}
{{ chat_template_data.special_tokens.b_include_to_loss_token}}
{% else %}
{{ " " }}
{% endif %}
{{ turn.value + '\n'}}
{% if turn.from == chat_template_data.assistant_role %}
{{ chat_template_data.special_tokens.e_assistant_token}}
{{ chat_template_data.special_tokens.e_include_to_loss_token}}
{% endif %}
{% endfor %}

# The key-value pairs of chat_template_data are passed to the Jinja2 template and
# are not type checked for full compliance with the chat tempalate!
chat_template_data:
assistant_role: Assistant
system_instruction: "You are Mody, a helpful assistant trained by the modalities team. Answer friendly and informatively to the user's messages."
# Currently only works with HF tokenizers, as the special tokens are added to the tokenizer
special_tokens:
# tokens to indicate the beginning and end of the assistant's response, only needed for proper loss masking
b_include_to_loss_token: ^
e_include_to_loss_token: $
# the user must need to know that any special tokens must be re-introduced in the pbin file creation config and the training config
# if not the training will run, but the e_assistant_token might not be learned correctly, as it is per default not a special token in the tokenizer
# the e_assistant_token is used to mask the end of the assistant's response. It will be trained and can be used as early stopping criterion during generation in inference mode
e_assistant_token: °
le1nux marked this conversation as resolved.
Show resolved Hide resolved
26 changes: 26 additions & 0 deletions config_files/data_preparation/packed_chat_dataset_config.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
settings:
src_path: data/lorem_ipsum_sft_40e0699/lorem_ipsum_sft_converted.40e0699.jsonl
dst_path: data/lorem_ipsum_sft_40e0699/lorem_ipsum_sft_converted.40e0699.pbin
index_path: data/lorem_ipsum_sft_40e0699/lorem_ipsum_sft_converted.40e0699.idx
jq_pattern: .chat
num_cpus: 1
eod_token: <|endoftext|>
processing_batch_size: 5
raw_samples_queue_size: 300
processed_samples_queue_size: 300
sequence_length: 2048

tokenizer:
component_key: tokenizer
variant_key: pretrained_hf_tokenizer
config:
pretrained_model_name_or_path: data/tokenizer/hf_gpt2
padding: max_length
truncation: true
max_length: ${settings.sequence_length}
special_tokens:
pad_token: ${settings.eod_token}
additional_special_tokens:
- "^"
- "$"
- "°"
Loading