Skip to content

[Tokenizers] Add support for HuggingFace BPE Tokenizer format #6901

Open
@shaltielshmid

Description

@shaltielshmid

Is your feature request related to a problem? Please describe.

I'm requesting this feature after trying to use the GPT2-style tokenizer I trained using HuggingFace in my .NET code. I had trained a model and converted the model to ONNX, but the tokenizer didn't transfer. An exact description of the problem is listed down below.

Describe the solution you'd like

Add support for a flag indicating that the tokenizer came from the HuggingFace BPE trainer, and behind the scenes handle the minor changes required.

Describe alternatives you've considered

Currently I have a class I wrote which wraps a BPE trainer and applies the adjustments before every call to the ML.NET BPE tokenizer.

Additional context

In the HuggingFace BPE code they have a dictionary bytes_to_unicode() which is list of utf-8 byte and a mapping to unicode strings. They run every byte in the string through the mapping before running the BPE encoder/decoder. Examples of where it's used can be found here and here and in other places.

Before the encoding, they treat the string as bytes and map all the bytes to representative unicode strings, and the same thing during after the decoding.

Real example:

I trained a BPE tokenizer using HuggingFace's tokenizers.ByteLevelBPETokenizer. The merges.txt and vocab.json can be found here: https://gist.github.com/shaltielshmid/58b7c1109639eefcd714eb6bfc3eb602.

Sample python code:

from transformers import GPT2Tokenizer
tokenizer = GPT2Tokenizer.from_pretrained('/path/to/tokenizer')
print(tokenizer.encode('שלום וברכה')); // [150, 662, 426, 1396]
print(tokenizer.decode([150, 662, 426, 1396])); // שלום וברכה

Sample C# code:

var bpe = new Bpe("/path/to/vocab.json", "/path/to/merges.txt");
string phrase = "שלום וברכה";
Console.WriteLine(string.Join(", ", bpe.Tokenize(phrase).Select(t => t.Id.ToString()))); // 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
string decoded = Bpe.Decoder.Decode(new List<int> { 150, 662, 426, 1396 }.Select(id => bpe.IdToToken(id)!)); 
Console.WriteLine(decoded); // ש׾×ķ×ĿĠ×ķ×ijר׼×Ķ

// with proposed solution from down below
phrase = new string(Encoding.UTF8.GetBytes(phrase).Select(b => hf_encoder[b]).ToArray());
Console.WriteLine(string.Join(", ", bpe.Tokenize(phrase).Select(t => t.Id.ToString()))); // 150, 662, 426, 1396
decoded = Encoding.UTF8.GetString(decoded.Select(c => (byte)hf_decoder[c]).ToArray());
Console.WriteLine(decoded); // שלום וברכה

Proposed Solution

Create a static dictionary in the BPE class, which is initialized once:

var hf_encoder = new Dictionary<int, char>();
for (int c = '!'; c <= '~'; c++) hf_encoder.Add(c, (char)c);
for (int c = '¡'; c <= '¬'; c++) hf_encoder.Add(c, (char)c);
for (int c = '®'; c <= 'ÿ'; c++) hf_encoder.Add(c, (char)c);
int n = 0;
for (int c = 0; c < 256; c++) {
    if (!hf_encoder.ContainsKey(c))
        hf_encoder.Add(c, (char)(256 + n++));
}

var hf_decoder = hf_encoder.ToDictionary(kvp => kvp.Value, kvp => kvp.Key);

Then, in the BPE.cs class in the Tokenize function here, add the following check:

if (_isHFFormat) {
    sequence = new string(Encoding.UTF8.GetBytes(sequence).Select(b => hf_encoder[b]).ToArray())
}

And then in the BPEDecoder.cs file, in the Decode function here

string ret = string.Join("", tokens);
if (_suffix != null)
{
    ret = ret.Replace(_suffix, " ");
}

if (_isHFFormat) {
    ret = Encoding.UTF8.GetString(ret.Select(c => (byte)hf_decoder[c]).ToArray())
}

return ret;

Would be happy to compile this into a PR, if relevant.

@luisquintanilla

Metadata

Metadata

Assignees

No one assigned

    Type

    No type

    Projects

    No projects

    Milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions