openai_completions.py 16.8 KB
Newer Older
Baber's avatar
Baber committed
1
2
3
4
5
import asyncio
import base64
import copy
import itertools
import json
Jason Phang's avatar
gpt3  
Jason Phang committed
6
import os
Baber Abbasi's avatar
Baber Abbasi committed
7
from functools import cached_property
8
from operator import itemgetter
Baber's avatar
Baber committed
9
from io import BytesIO
Baber Abbasi's avatar
Baber Abbasi committed
10
from typing import Any, Dict, List, Optional, Tuple, Union
11

Baber's avatar
Baber committed
12
13
14
15
16
from PIL import Image
from tenacity import retry, stop_after_attempt, wait_exponential
from tqdm import tqdm

from lm_eval.api.instance import Instance
17
from lm_eval.api.registry import register_model
Baber Abbasi's avatar
Baber Abbasi committed
18
from lm_eval.models.api_models import TemplateAPI
19
from lm_eval.models.utils import handle_stop_sequences
Baber's avatar
Baber committed
20
21
from lm_eval.models.api_models import JsonChatStr, TemplateAPI
from lm_eval.models.utils import Collator
22
from lm_eval.utils import eval_logger
Leo Gao's avatar
Leo Gao committed
23

lintangsutawika's avatar
update  
lintangsutawika committed
24

Baber Abbasi's avatar
Baber Abbasi committed
25
26
@register_model("local-completions")
class LocalCompletionsAPI(TemplateAPI):
lintangsutawika's avatar
lintangsutawika committed
27
28
    def __init__(
        self,
Baber Abbasi's avatar
Baber Abbasi committed
29
30
31
32
33
34
35
        base_url=None,
        tokenizer_backend="huggingface",
        **kwargs,
    ):
        super().__init__(
            base_url=base_url, tokenizer_backend=tokenizer_backend, **kwargs
        )
lintangsutawika's avatar
lintangsutawika committed
36

Baber Abbasi's avatar
Baber Abbasi committed
37
38
39
40
41
    def _create_payload(
        self,
        messages: Union[List[List[int]], List[dict], List[str], str],
        generate=False,
        gen_kwargs: Optional[dict] = None,
42
        seed: int = 1234,
43
        eos=None,
Baber Abbasi's avatar
Baber Abbasi committed
44
45
46
47
        **kwargs,
    ) -> dict:
        if generate:
            gen_kwargs.pop("do_sample", False)
48
49
50
51
            if "max_tokens" in gen_kwargs:
                max_tokens = gen_kwargs.pop("max_tokens")
            else:
                max_tokens = gen_kwargs.pop("max_gen_toks", self._max_gen_toks)
Baber Abbasi's avatar
Baber Abbasi committed
52
            temperature = gen_kwargs.pop("temperature", 0)
53
            stop = handle_stop_sequences(gen_kwargs.pop("until", None), eos)
Baber Abbasi's avatar
Baber Abbasi committed
54
55
56
57
58
59
            return {
                "prompt": messages,
                "model": self.model,
                "max_tokens": max_tokens,
                "temperature": temperature,
                "stop": stop,
60
                "seed": seed,
Baber Abbasi's avatar
Baber Abbasi committed
61
62
                **gen_kwargs,
            }
Baber Abbasi's avatar
Baber Abbasi committed
63
        else:
Baber Abbasi's avatar
Baber Abbasi committed
64
65
66
            return {
                "model": self.model,
                "prompt": messages,
67
                "temperature": 0,
Baber Abbasi's avatar
Baber Abbasi committed
68
69
                "max_tokens": 1,
                "logprobs": 1,
70
                "seed": seed,
Baber Abbasi's avatar
Baber Abbasi committed
71
72
73
74
75
76
77
78
79
                "echo": True,
            }

    @staticmethod
    def parse_logprobs(
        outputs: Union[Dict, List[Dict]],
        tokens: List[List[int]] = None,
        ctxlens: List[int] = None,
        **kwargs,
lintangsutawika's avatar
lintangsutawika committed
80
81
    ) -> List[Tuple[float, bool]]:
        res = []
Baber Abbasi's avatar
Baber Abbasi committed
82
83
84
        if not isinstance(outputs, list):
            outputs = [outputs]
        for out in outputs:
85
86
87
            for choice, ctxlen in zip(
                sorted(out["choices"], key=itemgetter("index")), ctxlens
            ):
Baber Abbasi's avatar
Baber Abbasi committed
88
89
                assert ctxlen > 0, "Context length must be greater than 0"
                logprobs = sum(choice["logprobs"]["token_logprobs"][ctxlen:-1])
90
                tokens_logprobs = choice["logprobs"]["token_logprobs"][ctxlen:-1]
Baber Abbasi's avatar
Baber Abbasi committed
91
92
                top_logprobs = choice["logprobs"]["top_logprobs"][ctxlen:-1]
                is_greedy = True
93
94
                for tok, top in zip(tokens_logprobs, top_logprobs):
                    if tok != max(top.values()):
Baber Abbasi's avatar
Baber Abbasi committed
95
96
97
98
99
100
101
                        is_greedy = False
                        break
                res.append((logprobs, is_greedy))
        return res

    @staticmethod
    def parse_generations(outputs: Union[Dict, List[Dict]], **kwargs) -> List[str]:
lintangsutawika's avatar
lintangsutawika committed
102
        res = []
Baber Abbasi's avatar
Baber Abbasi committed
103
104
105
        if not isinstance(outputs, list):
            outputs = [outputs]
        for out in outputs:
106
            tmp = [None] * len(out["choices"])
Baber Abbasi's avatar
Baber Abbasi committed
107
            for choices in out["choices"]:
108
109
                tmp[choices["index"]] = choices["text"]
            res = res + tmp
Baber Abbasi's avatar
Baber Abbasi committed
110
        return res
lintangsutawika's avatar
lintangsutawika committed
111

Baber Abbasi's avatar
Baber Abbasi committed
112
113
114
    @property
    def api_key(self):
        return os.environ.get("OPENAI_API_KEY", "")
lintangsutawika's avatar
lintangsutawika committed
115
116


Baber Abbasi's avatar
Baber Abbasi committed
117
118
119
120
121
122
123
124
125
@register_model("local-chat-completions")
class LocalChatCompletion(LocalCompletionsAPI):
    def __init__(
        self,
        base_url=None,
        tokenizer_backend=None,
        tokenized_requests=False,
        **kwargs,
    ):
126
127
128
        eval_logger.warning(
            "chat-completions endpoint requires the `--apply_chat_template` flag."
        )
Baber Abbasi's avatar
Baber Abbasi committed
129
130
131
132
133
134
135
136
137
        super().__init__(
            base_url=base_url,
            tokenizer_backend=tokenizer_backend,
            tokenized_requests=tokenized_requests,
            **kwargs,
        )
        if self._batch_size > 1:
            eval_logger.warning(
                "Chat completions does not support batching. Defaulting to batch size 1."
lintangsutawika's avatar
lintangsutawika committed
138
            )
Baber Abbasi's avatar
Baber Abbasi committed
139
140
141
            self._batch_size = 1

    def _create_payload(
142
143
144
145
146
        self,
        messages: List[Dict],
        generate=False,
        gen_kwargs: dict = None,
        seed=1234,
147
        eos=None,
148
        **kwargs,
Baber Abbasi's avatar
Baber Abbasi committed
149
    ) -> dict:
150
151
152
        assert (
            type(messages) is not str
        ), "chat-completions require the --apply_chat_template flag."
Baber Abbasi's avatar
Baber Abbasi committed
153
        gen_kwargs.pop("do_sample", False)
154
155
156
157
        if "max_tokens" in gen_kwargs:
            max_tokens = gen_kwargs.pop("max_tokens")
        else:
            max_tokens = gen_kwargs.pop("max_gen_toks", self._max_gen_toks)
Baber Abbasi's avatar
Baber Abbasi committed
158
        temperature = gen_kwargs.pop("temperature", 0)
159
        stop = handle_stop_sequences(gen_kwargs.pop("until", None), eos)
Baber Abbasi's avatar
Baber Abbasi committed
160
161
162
163
164
165
166
167
        if not isinstance(stop, (list, tuple)):
            stop = [stop]
        return {
            "messages": messages,
            "model": self.model,
            "max_tokens": max_tokens,
            "temperature": temperature,
            "stop": stop[:4],
168
            "seed": seed,
Baber Abbasi's avatar
Baber Abbasi committed
169
170
171
172
173
174
175
176
177
            **gen_kwargs,
        }

    @staticmethod
    def parse_generations(outputs: Union[Dict, List[Dict]], **kwargs) -> List[str]:
        res = []
        if not isinstance(outputs, list):
            outputs = [outputs]
        for out in outputs:
178
            tmp = [None] * len(out["choices"])
Baber Abbasi's avatar
Baber Abbasi committed
179
            for choices in out["choices"]:
180
181
                tmp[choices["index"]] = choices["message"]["content"]
            res = res + tmp
Baber Abbasi's avatar
Baber Abbasi committed
182
183
184
185
186
187
188
189
190
191
        return res

    def tok_encode(
        self,
        string: Union[str, Any],
        left_truncate_len=None,
        add_special_tokens=None,
        **kwargs,
    ) -> Union[List[str], List[int], Any]:
        return string
lintangsutawika's avatar
lintangsutawika committed
192

Baber Abbasi's avatar
Baber Abbasi committed
193
    def loglikelihood(self, requests, **kwargs):
Baber Abbasi's avatar
Baber Abbasi committed
194
195
196
        raise NotImplementedError(
            "Loglikelihood is not supported for chat completions. Consider using the completions API instead."
        )
lintangsutawika's avatar
lintangsutawika committed
197
198


Baber Abbasi's avatar
Baber Abbasi committed
199
200
201
202
@register_model(
    "openai-completions",
)
class OpenAICompletionsAPI(LocalCompletionsAPI):
203
    def __init__(
204
        self,
Baber Abbasi's avatar
Baber Abbasi committed
205
206
        base_url="https://api.openai.com/v1/completions",
        tokenizer_backend="tiktoken",
207
        **kwargs,
Baber Abbasi's avatar
Baber Abbasi committed
208
209
210
211
    ):
        super().__init__(
            base_url=base_url, tokenizer_backend=tokenizer_backend, **kwargs
        )
212

Baber Abbasi's avatar
Baber Abbasi committed
213
214
215
216
217
218
    @cached_property
    def api_key(self):
        """Override this property to return the API key for the API request."""
        key = os.environ.get("OPENAI_API_KEY", None)
        if key is None:
            raise ValueError(
219
                "API key not found. Please set the `OPENAI_API_KEY` environment variable."
220
            )
Baber Abbasi's avatar
Baber Abbasi committed
221
        return key
222

Baber Abbasi's avatar
Baber Abbasi committed
223
    def loglikelihood(self, requests, **kwargs):
Baber Abbasi's avatar
Baber Abbasi committed
224
        assert (
225
226
227
228
229
230
            self.model
            in [
                "babbage-002",
                "davinci-002",
            ]
        ), f"Prompt loglikelihoods are only supported by OpenAI's API for {['babbage-002', 'davinci-002']}."
Baber Abbasi's avatar
Baber Abbasi committed
231
        return super().loglikelihood(requests, **kwargs)
232

233
234
235
    def chat_template(self, chat_template: Union[bool, str] = False) -> Optional[str]:
        return ""

236

Baber Abbasi's avatar
Baber Abbasi committed
237
@register_model("openai-chat-completions")
Baber Abbasi's avatar
Baber Abbasi committed
238
239
240
241
242
243
244
245
class OpenAIChatCompletion(LocalChatCompletion):
    def __init__(
        self,
        base_url="https://api.openai.com/v1/chat/completions",
        tokenizer_backend=None,
        tokenized_requests=False,
        **kwargs,
    ):
246
247
248
249
        if "o1" in kwargs.get("model", ""):
            eval_logger.warning(
                "o1 models do not support `stop` and only support temperature=1"
            )
Baber Abbasi's avatar
Baber Abbasi committed
250
251
252
253
254
255
        super().__init__(
            base_url=base_url,
            tokenizer_backend=tokenizer_backend,
            tokenized_requests=tokenized_requests,
            **kwargs,
        )
256

Baber Abbasi's avatar
Baber Abbasi committed
257
258
259
260
261
262
    @cached_property
    def api_key(self):
        """Override this property to return the API key for the API request."""
        key = os.environ.get("OPENAI_API_KEY", None)
        if key is None:
            raise ValueError(
263
                "API key not found. Please set the `OPENAI_API_KEY` environment variable."
264
            )
Baber Abbasi's avatar
Baber Abbasi committed
265
        return key
266
267
268
269
270

    def loglikelihood(self, requests, **kwargs):
        raise NotImplementedError(
            "Loglikelihood (and therefore `multiple_choice`-type tasks) is not supported for chat completions as OpenAI does not provide prompt logprobs. See https://github.com/EleutherAI/lm-evaluation-harness/issues/942#issuecomment-1777836312 or https://github.com/EleutherAI/lm-evaluation-harness/issues/1196 for more background on this limitation."
        )
Baber's avatar
Baber committed
271

272
273
274
275
276
277
    def _create_payload(
        self,
        messages: List[Dict],
        generate=False,
        gen_kwargs: dict = None,
        seed=1234,
278
        eos="<|endoftext|>",
279
280
281
282
283
284
285
286
287
288
289
        **kwargs,
    ) -> dict:
        assert (
            type(messages) is not str
        ), "chat-completions require the --apply_chat_template flag."
        gen_kwargs.pop("do_sample", False)
        if "max_tokens" in gen_kwargs:
            max_tokens = gen_kwargs.pop("max_tokens")
        else:
            max_tokens = gen_kwargs.pop("max_gen_toks", self._max_gen_toks)
        temperature = gen_kwargs.pop("temperature", 0)
290
        stop = handle_stop_sequences(gen_kwargs.pop("until", ["<|endoftext|>"]), eos)
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
        if not isinstance(stop, (list, tuple)):
            stop = [stop]
        output = {
            "messages": messages,
            "model": self.model,
            "max_completion_tokens": max_tokens,
            "temperature": temperature,
            "stop": stop[:4],
            "seed": seed,
            **gen_kwargs,
        }
        if "o1" in self.model:
            output.pop("stop")
            output["temperature"] = 1
        return output
Baber's avatar
Baber committed
306

Baber's avatar
Baber committed
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488

@register_model("pixtral-api")
class PixtralAPI(LocalChatCompletion):
    MULTIMODAL = True
    DEFAULT_IMAGE_PLACEHOLDER = "<image>"

    def __init__(
        self,
        max_images: int = 999,
        **kwargs,
    ):
        self.max_images = max_images
        super().__init__(
            tokenizer_backend=None,
            tokenized_requests=False,
            model="mistralai/Pixtral-12B-2409",
            **kwargs,
        )

    def generate_until(
        self, requests: List[Instance], disable_tqdm: bool = False
    ) -> List[str]:
        res = []

        def _collate_gen(_requests):
            # sort by the length of the non-tokenized contexts
            return -len(_requests[0])

        # Let the API deal with tokenization
        requests, all_gen_kwargs, aux_args = zip(*(req.args for req in requests))
        if self.tokenized_requests:
            encodings_list = self.tok_encode(
                requests, add_special_tokens=self.add_bos_token
            )
        else:
            requests = [
                self.update_json_chat_str_with_image(req, pil_image["visual"])
                for req, pil_image in zip(requests, aux_args)
            ]
            encodings_list = [None] * len(requests)
        requests = [
            (a, b, c) for a, b, c in zip(requests, all_gen_kwargs, encodings_list)
        ]

        re_ord = Collator(
            requests,
            sort_fn=_collate_gen,
            group_by="gen_kwargs",
        )
        chunked = re_ord.get_batched(
            n=self._batch_size if self._concurrent <= 1 else 0, batch_fn=None
        )
        if self._concurrent <= 1:
            pbar = tqdm(desc="Requesting API", total=len(requests))
            for chunk in chunked:
                contexts, all_gen_kwargs, encodings_list = zip(*chunk)
                req = encodings_list if self.tokenized_requests else contexts
                outputs = retry(
                    stop=stop_after_attempt(self.max_retries),
                    wait=wait_exponential(multiplier=0.5, min=1, max=10),
                    reraise=True,
                )(self.model_call)(
                    messages=req,
                    generate=True,
                    gen_kwargs=copy.deepcopy(all_gen_kwargs[0]),
                )
                for generated_text, context in zip(
                    self.parse_generations(
                        outputs=outputs,
                        contexts=contexts,
                    ),
                    contexts,
                ):
                    if generated_text is not None:
                        res.append(generated_text)

                        # partial caching
                        if context is not None:
                            self.cache_hook.add_partial(
                                "generate_until",
                                (context, all_gen_kwargs[0]),
                                generated_text,
                            )
                            pbar.update(1)
        else:
            for chunk in chunked:
                contexts, all_gen_kwargs, encodings_list = zip(*chunk)
                req = encodings_list if self.tokenized_requests else contexts
                results = itertools.chain.from_iterable(
                    asyncio.run(
                        self.get_batched_requests(
                            req,
                            cache_keys=[(ctx, all_gen_kwargs[0]) for ctx in contexts],
                            generate=True,
                            gen_kwargs=copy.deepcopy(all_gen_kwargs[0]),
                        )
                    )
                )
                res.extend(results)

        return re_ord.get_original(res)

    @staticmethod
    def encode_pillow_image(img):
        if img.mode == "P":
            img = img.convert("RGB")
        if img.mode == "RGBA":
            # Create a white background
            background = Image.new("RGB", img.size, (255, 255, 255))
            # Paste the image on the background.
            # The alpha channel is automatically used as mask
            background.paste(img, mask=img.split()[3])
            img = background

        buffered = BytesIO()
        img.save(buffered, format="JPEG")

        return base64.b64encode(buffered.getvalue()).decode("utf-8")

    def update_json_chat_str_with_image(
        self, json_chat_str, pil_images: Union["Image.Image", List["Image.Image"]]
    ):
        # Parse the JSON string
        chat_data = json.loads(json_chat_str.prompt)

        # Convert single image to list for consistency
        if not isinstance(pil_images, list):
            pil_images = [pil_images]

        # Encode the Pillow image(s)
        base64_images = [self.encode_pillow_image(img) for img in pil_images]

        # Update the image_url(s) in the chat data
        image_index = 0
        for message in chat_data:
            if message["role"] == "user":
                for content in message["content"]:
                    if content["type"] == "image_url":
                        if image_index < len(base64_images):
                            content["image_url"] = {
                                "url": f"data:image/jpeg;base64,{base64_images[image_index]}"
                            }
                            image_index += 1
                        else:
                            # If we run out of images, set to None or handle as needed
                            content["image_url"] = None

        # Update the JsonChatStr object with the new JSON string
        json_chat_str = JsonChatStr(json.dumps(chat_data))

        return json_chat_str

    def apply_chat_template(
        self, chat_history: List[Dict[str, str]]
    ) -> Union[str, JsonChatStr]:
        """Applies a chat template to a list of chat history between user and model."""
        if self.tokenizer_backend == "huggingface" and self.tokenized_requests:
            return self.tokenizer.apply_chat_template(
                chat_history, tokenize=False, add_generation_prompt=True
            )
        else:
            # bit of a hack. We'll load back before sending to the API
            new_messages = []
            for message in chat_history:
                if message["role"] == "user":
                    # Split the content at <image> placeholder
                    parts = message["content"].split("<image>")
                    new_content = [
                        {"type": "text", "text": parts[0].strip()},
                        {"type": "image_url", "image_url": None},
                    ]
                    if len(parts) > 1:
                        new_content.append({"type": "text", "text": parts[1].strip()})

                    new_messages.append(
                        {"role": message["role"], "content": new_content}
                    )
                else:
                    # For non-user messages, keep the format as is
                    new_messages.append(message)

            return JsonChatStr(json.dumps(new_messages))