mirror of
				https://github.com/meta-llama/llama-stack.git
				synced 2025-10-26 09:15:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			198 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			198 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright (c) Meta Platforms, Inc. and affiliates.
 | |
| # All rights reserved.
 | |
| #
 | |
| # This source code is licensed under the terms described in the LICENSE file in
 | |
| # the root directory of this source tree.
 | |
| 
 | |
| from collections.abc import Collection, Iterator, Sequence, Set
 | |
| from logging import getLogger
 | |
| from pathlib import Path
 | |
| from typing import (
 | |
|     Literal,
 | |
|     cast,
 | |
| )
 | |
| 
 | |
| import tiktoken
 | |
| 
 | |
| from llama_stack.models.llama.tokenizer_utils import load_bpe_file
 | |
| 
 | |
| logger = getLogger(__name__)
 | |
| 
 | |
| 
 | |
| # The tiktoken tokenizer can handle <=400k chars without
 | |
| # pyo3_runtime.PanicException.
 | |
| TIKTOKEN_MAX_ENCODE_CHARS = 400_000
 | |
| 
 | |
| # https://github.com/openai/tiktoken/issues/195
 | |
| # Here we iterate over subsequences and split if we exceed the limit
 | |
| # of max consecutive non-whitespace or whitespace characters.
 | |
| MAX_NO_WHITESPACES_CHARS = 25_000
 | |
| 
 | |
| 
 | |
| _INSTANCE = None
 | |
| 
 | |
| 
 | |
| class Tokenizer:
 | |
|     """
 | |
|     Tokenizing and encoding/decoding text using the Tiktoken tokenizer.
 | |
|     """
 | |
| 
 | |
|     special_tokens: dict[str, int]
 | |
| 
 | |
|     num_reserved_special_tokens = 256
 | |
| 
 | |
|     pat_str = r"(?i:'s|'t|'re|'ve|'m|'ll|'d)|[^\r\n\p{L}\p{N}]?\p{L}+|\p{N}{1,3}| ?[^\s\p{L}\p{N}]+[\r\n]*|\s*[\r\n]+|\s+(?!\S)|\s+"  # noqa: E501
 | |
| 
 | |
|     @classmethod
 | |
|     def get_instance(cls):
 | |
|         global _INSTANCE
 | |
| 
 | |
|         if _INSTANCE is None:
 | |
|             _INSTANCE = Tokenizer(Path(__file__).parent / "tokenizer.model")
 | |
|         return _INSTANCE
 | |
| 
 | |
|     def __init__(self, model_path: Path):
 | |
|         """
 | |
|         Initializes the Tokenizer with a Tiktoken model.
 | |
| 
 | |
|         Args:
 | |
|             model_path (str): The path to the Tiktoken model file.
 | |
|         """
 | |
|         if not model_path.exists():
 | |
|             raise FileNotFoundError(f"Tokenizer model file not found: {model_path}")
 | |
| 
 | |
|         mergeable_ranks = load_bpe_file(model_path)
 | |
|         num_base_tokens = len(mergeable_ranks)
 | |
|         special_tokens = [
 | |
|             "<|begin_of_text|>",
 | |
|             "<|end_of_text|>",
 | |
|             "<|reserved_special_token_0|>",
 | |
|             "<|reserved_special_token_1|>",
 | |
|             "<|finetune_right_pad_id|>",
 | |
|             "<|step_id|>",
 | |
|             "<|start_header_id|>",
 | |
|             "<|end_header_id|>",
 | |
|             "<|eom_id|>",  # end of message
 | |
|             "<|eot_id|>",  # end of turn
 | |
|             "<|python_tag|>",
 | |
|             "<|image|>",
 | |
|         ]
 | |
|         reserved_tokens = [
 | |
|             f"<|reserved_special_token_{2 + i}|>" for i in range(self.num_reserved_special_tokens - len(special_tokens))
 | |
|         ]
 | |
|         special_tokens = special_tokens + reserved_tokens
 | |
| 
 | |
|         self.special_tokens = {token: num_base_tokens + i for i, token in enumerate(special_tokens)}
 | |
|         self.model = tiktoken.Encoding(
 | |
|             name=model_path.name,
 | |
|             pat_str=self.pat_str,
 | |
|             mergeable_ranks=mergeable_ranks,
 | |
|             special_tokens=self.special_tokens,
 | |
|         )
 | |
| 
 | |
|         self.n_words: int = num_base_tokens + len(special_tokens)
 | |
|         # BOS / EOS token IDs
 | |
|         self.bos_id: int = self.special_tokens["<|begin_of_text|>"]
 | |
|         self.eos_id: int = self.special_tokens["<|end_of_text|>"]
 | |
|         self.eot_id: int = self.special_tokens["<|eot_id|>"]
 | |
|         self.eom_id: int = self.special_tokens["<|eom_id|>"]
 | |
|         self.python_tag_id = self.special_tokens["<|python_tag|>"]
 | |
|         self.pad_id: int = self.special_tokens["<|finetune_right_pad_id|>"]
 | |
|         self.stop_tokens = [
 | |
|             self.eos_id,
 | |
|             self.special_tokens["<|eom_id|>"],
 | |
|             self.special_tokens["<|eot_id|>"],
 | |
|         ]
 | |
| 
 | |
|     def encode(
 | |
|         self,
 | |
|         s: str,
 | |
|         *,
 | |
|         bos: bool,
 | |
|         eos: bool,
 | |
|         allowed_special: Literal["all"] | Set[str] | None = None,
 | |
|         disallowed_special: Literal["all"] | Collection[str] = (),
 | |
|     ) -> list[int]:
 | |
|         """
 | |
|         Encodes a string into a list of token IDs.
 | |
| 
 | |
|         Args:
 | |
|             s (str): The input string to be encoded.
 | |
|             bos (bool): Whether to prepend the beginning-of-sequence token.
 | |
|             eos (bool): Whether to append the end-of-sequence token.
 | |
|             allowed_special ("all"|set[str]): allowed special tokens in string
 | |
|             disallowed_special ("all"|set[str]): special tokens that raise an error when in string
 | |
| 
 | |
|         Returns:
 | |
|             list[int]: A list of token IDs.
 | |
| 
 | |
|         By default, setting disallowed_special=() encodes a string by ignoring
 | |
|         special tokens. Specifically:
 | |
|         - Setting `disallowed_special` to () will cause all text corresponding
 | |
|           to special tokens to be encoded as natural text (insteading of raising
 | |
|           an error).
 | |
|         - Setting `allowed_special` to "all" will treat all text corresponding
 | |
|           to special tokens to be encoded as special tokens.
 | |
|         """
 | |
|         if allowed_special is None:
 | |
|             allowed_special = set()
 | |
|         assert type(s) is str
 | |
| 
 | |
|         substrs = (
 | |
|             substr
 | |
|             for i in range(0, len(s), TIKTOKEN_MAX_ENCODE_CHARS)
 | |
|             for substr in self._split_whitespaces_or_nonwhitespaces(
 | |
|                 s[i : i + TIKTOKEN_MAX_ENCODE_CHARS], MAX_NO_WHITESPACES_CHARS
 | |
|             )
 | |
|         )
 | |
|         t: list[int] = []
 | |
|         for substr in substrs:
 | |
|             t.extend(
 | |
|                 self.model.encode(
 | |
|                     substr,
 | |
|                     allowed_special=allowed_special,
 | |
|                     disallowed_special=disallowed_special,
 | |
|                 )
 | |
|             )
 | |
|         if bos:
 | |
|             t.insert(0, self.bos_id)
 | |
|         if eos:
 | |
|             t.append(self.eos_id)
 | |
|         return t
 | |
| 
 | |
|     def decode(self, t: Sequence[int]) -> str:
 | |
|         """
 | |
|         Decodes a list of token IDs into a string.
 | |
| 
 | |
|         Args:
 | |
|             t (List[int]): The list of token IDs to be decoded.
 | |
| 
 | |
|         Returns:
 | |
|             str: The decoded string.
 | |
|         """
 | |
|         # Typecast is safe here. Tiktoken doesn't do anything list-related with the sequence.
 | |
|         return self.model.decode(cast(list[int], t))
 | |
| 
 | |
|     @staticmethod
 | |
|     def _split_whitespaces_or_nonwhitespaces(s: str, max_consecutive_slice_len: int) -> Iterator[str]:
 | |
|         """
 | |
|         Splits the string `s` so that each substring contains no more than `max_consecutive_slice_len`
 | |
|         consecutive whitespaces or consecutive non-whitespaces.
 | |
|         """
 | |
|         current_slice_len = 0
 | |
|         current_slice_is_space = s[0].isspace() if len(s) > 0 else False
 | |
|         slice_start = 0
 | |
| 
 | |
|         for i in range(len(s)):
 | |
|             is_now_space = s[i].isspace()
 | |
| 
 | |
|             if current_slice_is_space ^ is_now_space:
 | |
|                 current_slice_len = 1
 | |
|                 current_slice_is_space = is_now_space
 | |
|             else:
 | |
|                 current_slice_len += 1
 | |
|                 if current_slice_len > max_consecutive_slice_len:
 | |
|                     yield s[slice_start:i]
 | |
|                     slice_start = i
 | |
|                     current_slice_len = 1
 | |
|         yield s[slice_start:]
 |