mistral.py 14.5 KB
Newer Older
1
2
# SPDX-License-Identifier: Apache-2.0

3
4
5
6
import os
import re
from dataclasses import dataclass
from pathlib import Path
7
from typing import TYPE_CHECKING, Any, Dict, List, Optional, Union, cast
8

9
import huggingface_hub
10
from huggingface_hub import HfApi, hf_hub_download
11
from mistral_common.protocol.instruct.request import ChatCompletionRequest
12
from mistral_common.tokens.tokenizers.base import SpecialTokens
13
14
15
16
17
18
19
20
21
# yapf: disable
from mistral_common.tokens.tokenizers.mistral import (
    MistralTokenizer as PublicMistralTokenizer)
# yapf: enable
from mistral_common.tokens.tokenizers.sentencepiece import (
    SentencePieceTokenizer)
from mistral_common.tokens.tokenizers.tekken import (SpecialTokenPolicy,
                                                     Tekkenizer)

22
from vllm.logger import init_logger
23
from vllm.utils import is_list_of
24

25
if TYPE_CHECKING:
26
    from vllm.entrypoints.chat_utils import ChatCompletionMessageParam
27

28
29
logger = init_logger(__name__)

30
31
32

@dataclass
class Encoding:
33
    input_ids: Union[List[int], List[List[int]]]
34
35


36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
def maybe_serialize_tool_calls(request: ChatCompletionRequest):
    # SEE: https://github.com/vllm-project/vllm/pull/9951
    # Credits go to: @gcalmettes
    # NOTE: There is currently a bug in pydantic where attributes
    # declared as iterables are replaced in in the instances by
    # pydantic-core ValidatorIterator instance. In particular, this
    # affects tool_calls defined in ChatCompletionAssistantMessageParam
    # model:
    # see:
    #   - https://github.com/pydantic/pydantic/issues/9467
    # As a result, tool_calls from assistant messages are never
    # deserialized in the request object if the tool_calls iterator is
    # not consumed. This affect messages passed to the MistralTokenizer
    # since no chat template is applied and therefore the tools_calls
    # iterator is not directly consumed.
    # Issue is tracked on Pydantic side, with resolution planned for
    # v2.11 release. In the meantime, the official workaround is to
    # consume the iterator so the tool_calls are correctly deserialized
    # in the OpenAI ChatCompletionAssistantMessageParam object
    # https://github.com/pydantic/pydantic/issues/9467#issuecomment-2442097291 # noqa: E501
    # Official Pydantic Issues:
    #   - https://github.com/pydantic/pydantic/issues/9541
    # TODO: remove when pydantic v2.11 is released
    for i, message in enumerate(request.messages):
        if message.get("role") == 'assistant':
            tool_calls_validator = message.get("tool_calls", ().__iter__())
            validated_tool_calls = []
            while True:
                try:
                    tool_call = next(tool_calls_validator)  # type: ignore
                    validated_tool_calls.append(tool_call)
                except StopIteration:
                    break

            request.messages[i]["tool_calls"] = validated_tool_calls


73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
def list_local_repo_files(repo_id: str, revision: Optional[str]) -> List[str]:
    repo_cache = os.path.join(
        huggingface_hub.constants.HF_HUB_CACHE,
        huggingface_hub.constants.REPO_ID_SEPARATOR.join(
            ["models", *repo_id.split("/")]))

    if revision is None:
        revision_file = os.path.join(repo_cache, "refs", "main")
        if os.path.isfile(revision_file):
            with open(revision_file) as file:
                revision = file.read()

    if revision:
        revision_dir = os.path.join(repo_cache, "snapshots", revision)
        if os.path.isdir(revision_dir):
            return os.listdir(revision_dir)

    return []


93
94
95
96
97
98
def find_tokenizer_file(files: List[str]):
    file_pattern = re.compile(r"^tokenizer\.model\.v.*$|^tekken\.json$")

    matched_files = [file for file in files if file_pattern.match(file)]
    if len(matched_files) > 1:
        raise OSError(f"Found {len(matched_files)} files matching the "
99
100
                      f"pattern: {file_pattern}. Make sure only one Mistral "
                      f"tokenizer is present in {files}.")
101
102
    elif len(matched_files) == 0:
        raise OSError(f"Found {len(matched_files)} files matching the "
103
104
                      f"pattern: {file_pattern}. Make sure that a Mistral "
                      f"tokenizer is present in {files}.")
105
106
107
108
109
110
111
112
113
114

    return matched_files[0]


class MistralTokenizer:

    def __init__(self, tokenizer: PublicMistralTokenizer) -> None:
        self.mistral = tokenizer
        self.instruct = tokenizer.instruct_tokenizer

115
        tokenizer_ = tokenizer.instruct_tokenizer.tokenizer
116
117
118
        self.is_tekken = isinstance(tokenizer_, Tekkenizer)
        self.is_spm = isinstance(tokenizer_, SentencePieceTokenizer)
        if self.is_tekken:
119
            # Make sure special tokens will not raise
120
            tokenizer_.special_token_policy = SpecialTokenPolicy.IGNORE
121
        elif self.is_spm:
122
            pass
123
124
        else:
            raise TypeError(f"Unsupported tokenizer: {type(tokenizer_)}")
125

126
127
128
129
130
131
132
133
        self._vocab = tokenizer_.vocab()
        # Convert to a Dict[str, int] to match protocol, but this is a lossy
        # conversion. There may be multiple token ids that decode to the same
        # string due to partial UTF-8 byte sequences being converted to �
        self._vocab_dict = {
            token: idx
            for idx, token in enumerate(self._vocab)
        }
134
        self.tokenizer = tokenizer_
135
        self._max_token_id = self.vocab_size - 1
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161

    @classmethod
    def from_pretrained(cls,
                        path_or_repo_id: str,
                        *,
                        revision: Optional[str] = None) -> "MistralTokenizer":
        if not Path(path_or_repo_id).exists():
            assert len(path_or_repo_id.split("/")) == 2, (
                "You have either provided a non-existent path: "
                "{path_or_repo_id} or an invalid HF Hub repo id.")
            tokenizer_file = cls._download_mistral_tokenizer_from_hf(
                path_or_repo_id, revision)
        elif Path(path_or_repo_id).is_dir():
            tokenizer_file_name = find_tokenizer_file(
                os.listdir(path_or_repo_id))
            tokenizer_file = str(Path(path_or_repo_id) / tokenizer_file_name)
        else:
            assert Path(
                path_or_repo_id).is_file(), f"Invalid path: {path_or_repo_id}"

        mistral_tokenizer = PublicMistralTokenizer.from_file(tokenizer_file)
        return cls(mistral_tokenizer)

    @staticmethod
    def _download_mistral_tokenizer_from_hf(tokenizer_name: str,
                                            revision: Optional[str]) -> str:
162
163
164
165
166
167
168
169
170
171
        try:
            hf_api = HfApi()
            files = hf_api.list_repo_files(repo_id=tokenizer_name,
                                           revision=revision)
        except ConnectionError as exc:
            files = list_local_repo_files(repo_id=tokenizer_name,
                                          revision=revision)

            if len(files) == 0:
                raise exc
172
173
174
175
176
177
178
179

        filename = find_tokenizer_file(files)

        tokenizer_file = hf_hub_download(tokenizer_name,
                                         filename=filename,
                                         revision=revision)
        return tokenizer_file

180
181
    # the following attributes are set to fit VLLM's design and are used
    # by the guided structured output backends.
182
183
    @property
    def all_special_tokens_extended(self) -> List[str]:
184
185
186
187
188
189
190
191
192
        # tekken defines its own extended special tokens list
        if hasattr(self.tokenizer, "SPECIAL_TOKENS"):
            special_tokens = self.tokenizer.SPECIAL_TOKENS
        else:
            special_tokens = list(SpecialTokens)
        return [
            s.value if isinstance(s, SpecialTokens) else s
            for s in special_tokens
        ]
193
194
195

    @property
    def all_special_tokens(self) -> List[str]:
196
        return self.all_special_tokens_extended
197
198
199

    @property
    def all_special_ids(self) -> List[int]:
200
201
202
        return [
            self.all_special_tokens.index(t) for t in self.all_special_tokens
        ]
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219

    @property
    def bos_token_id(self) -> int:
        return self.tokenizer.bos_id

    @property
    def eos_token_id(self) -> int:
        return self.tokenizer.eos_id

    @property
    def is_fast(self) -> bool:
        return True

    @property
    def vocab_size(self) -> int:
        return len(self._vocab)

220
221
222
223
    @property
    def max_token_id(self) -> int:
        return self._max_token_id

224
225
226
    def __len__(self) -> int:
        return self.vocab_size

227
228
    def __call__(
        self,
229
        prompt: Union[str, List[str], List[int]],
230
231
232
233
        add_special_tokens: bool = False,
        truncation: bool = False,
        max_length: Optional[int] = None,
    ):
234
235
236
237
238
239
240
241
242
243
244
245
246
247
        input_ids: Union[List[int], List[List[int]]]
        # For List[str], original prompt text
        if is_list_of(prompt, str):
            input_ids_: List[List[int]] = []
            for p in prompt:
                each_input_ids = self.encode_one(p, truncation, max_length)
                input_ids_.append(each_input_ids)
            input_ids = input_ids_
        # For List[int], apply chat template output, already tokens.
        elif is_list_of(prompt, int):
            input_ids = prompt
        # For str, single prompt text
        else:
            input_ids = self.encode_one(prompt, truncation, max_length)
248
249
        return Encoding(input_ids=input_ids)

250
    def get_vocab(self) -> Dict[str, int]:
251
252
253
        # NB: the dictionary form of the vocabulary collapses token ids that map
        # to the same string but have different bytes
        return self._vocab_dict
254
255

    def get_added_vocab(self) -> Dict[str, int]:
256
        # Mistral tokenizers have no added vocabulary
257
        return {}
258

259
260
261
262
263
264
265
266
267
268
269
270
271
    def encode_one(
        self,
        prompt: str,
        truncation: bool = False,
        max_length: Optional[int] = None,
    ) -> List[int]:
        # Mistral Tokenizers should not add special tokens
        input_ids = self.encode(prompt)

        if truncation:
            input_ids = input_ids[:max_length]
        return input_ids

272
    def encode(self, prompt: str) -> List[int]:
273
        # `encode` should only be used for prompt completion
274
275
276
277
278
        # it should never be used for chat_completion.
        # For chat completion use `apply_chat_template`
        return self.tokenizer.encode(prompt, bos=True, eos=False)

    def apply_chat_template(self,
279
                            messages: List["ChatCompletionMessageParam"],
280
281
282
                            tools: Optional[Dict[str, Any]] = None,
                            **kwargs) -> List[int]:

283
        last_message = cast(Dict[str, Any], messages[-1])
284
285
286
        if last_message["role"] == "assistant":
            last_message["prefix"] = True

287
288
        request = ChatCompletionRequest(messages=messages,
                                        tools=tools)  # type: ignore[type-var]
289
290
291
292
293
294
        encoded = self.mistral.encode_chat_completion(request)

        # encode-decode to get clean prompt
        return encoded.tokens

    def convert_tokens_to_string(self, tokens: List[str]) -> str:
295
        if self.is_tekken:
296
297
            tokens = [
                t for t in tokens
298
299
                if (t is SpecialTokens.tool_calls
                    or t not in self.tokenizer._all_special_tokens)
300
301
302
303
304
            ]

            if any(isinstance(t, bytes) for t in tokens):
                # we need to encode and decode all tokens again
                shift = self.tokenizer.num_special_tokens
305
306
307
308
309
310
311
312
313
314
315
316
317
318

                def _token_to_id(t: str):
                    t_bytes = t.encode("utf-8") \
                        if not isinstance(t, bytes) else t
                    try:
                        return shift + \
                            self.tokenizer._tekken_token2id_nospecial[t_bytes]
                    except KeyError:
                        logger.warning(
                            "Failed to convert token %s to id,"
                            " replacing with <unk>", t_bytes)
                        return self.tokenizer.unk_id

                ids = [_token_to_id(t) for t in tokens]
319
320
321
                decoded = self.tokenizer.decode(ids)
            else:
                decoded = "".join(tokens)
322
        else:
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
            # make sure certain special tokens like Tool calls are
            # not decoded
            special_tokens = {SpecialTokens.tool_calls}
            regular_tokens: List[str] = []
            decoded_list = []

            for token in tokens:
                if token in special_tokens:
                    if regular_tokens:
                        decoded_list.append(
                            self.tokenizer.decode(regular_tokens))
                        regular_tokens = []
                    decoded_list.append(token)
                else:
                    regular_tokens.append(token)

            if regular_tokens:
                decoded_list.append(
341
                    self.tokenizer.decode(regular_tokens))  # type: ignore
342
343

            decoded = ''.join(decoded_list)
344
345

        return decoded
346

347
348
349
    # WARN: Outlines logits processors can overwrite this method.
    # See: guided_decoding/outlines_logits_processors.py::_adapt_tokenizer
    # for more.
350
351
352
353
354
    def decode(self,
               ids: Union[List[int], int],
               skip_special_tokens: bool = True) -> str:
        assert (
            skip_special_tokens
355
        ), "skip_special_tokens=False is not supported for Mistral tokenizers."
356

357
358
359
360
361
        if isinstance(ids, int):
            ids = [ids]
        return self.tokenizer.decode(ids)

    def convert_ids_to_tokens(
362
363
364
365
        self,
        ids: List[int],
        skip_special_tokens: bool = True,
    ) -> List[str]:
366
367
368
        # TODO(Patrick) - potentially allow special tokens to not be skipped
        assert (
            skip_special_tokens
369
        ), "skip_special_tokens=False is not supported for Mistral tokenizers."
370

371
        assert self.is_tekken or self.is_spm, type(self.tokenizer)
372

373
        if self.is_tekken:
374
375
376
377
378
            # skip special tokens except tool call
            ids = [
                i for i in ids if i > self.tokenizer.num_special_tokens or i ==
                self.tokenizer.get_control_token(SpecialTokens.tool_calls)
            ]
379

380
        tokens = [self.tokenizer.id_to_piece(id) for id in ids]
381

382
        if any("�" in t for t in tokens) and self.is_tekken:
383
384
            # if a decoded token contains the replacement character, then the
            # token has an incomplete UTF-8 character so we must use bytes
385
            # See: https://github.com/vllm-project/vllm/pull/8640
386
            #      https://github.com/vllm-project/vllm/pull/9625
387
            # if underlying tokenizeir is sentencepiece, we just add "�"
388
389
            tokens = [self.tokenizer.id_to_byte_piece(id) for id in ids]

390
        return tokens