Skip to content

Commit e3493e2

Browse files
committed
feat(client): add ._request_id property to object responses
1 parent 4d45eb5 commit e3493e2

File tree

5 files changed

+109
-3
lines changed

5 files changed

+109
-3
lines changed

src/openai/_legacy_response.py

Lines changed: 4 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@
2525

2626
from ._types import NoneType
2727
from ._utils import is_given, extract_type_arg, is_annotated_type
28-
from ._models import BaseModel, is_basemodel
28+
from ._models import BaseModel, is_basemodel, add_request_id
2929
from ._constants import RAW_RESPONSE_HEADER
3030
from ._streaming import Stream, AsyncStream, is_stream_class_type, extract_stream_chunk_type
3131
from ._exceptions import APIResponseValidationError
@@ -138,6 +138,9 @@ class MyModel(BaseModel):
138138
if is_given(self._options.post_parser):
139139
parsed = self._options.post_parser(parsed)
140140

141+
if isinstance(parsed, BaseModel):
142+
add_request_id(parsed, self.request_id)
143+
141144
self._parsed_by_type[cache_key] = parsed
142145
return parsed
143146

src/openai/_models.py

Lines changed: 33 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
import os
44
import inspect
5-
from typing import TYPE_CHECKING, Any, Type, Union, Generic, TypeVar, Callable, cast
5+
from typing import TYPE_CHECKING, Any, Type, Union, Generic, TypeVar, Callable, Optional, cast
66
from datetime import date, datetime
77
from typing_extensions import (
88
Unpack,
@@ -94,6 +94,23 @@ def model_fields_set(self) -> set[str]:
9494
class Config(pydantic.BaseConfig): # pyright: ignore[reportDeprecated]
9595
extra: Any = pydantic.Extra.allow # type: ignore
9696

97+
if TYPE_CHECKING:
98+
_request_id: Optional[str] = None
99+
"""The ID of the request, returned via the X-Request-ID header. Useful for debugging requests and reporting issues to OpenAI.
100+
101+
This will **only** be set for the top-level response object, it will not be defined for nested objects. For example:
102+
103+
```py
104+
completion = await client.chat.completions.create(...)
105+
completion._request_id # req_id_xxx
106+
completion.usage._request_id # raises `AttributeError`
107+
```
108+
109+
Note: unlike other properties that use an `_` prefix, this property
110+
*is* public. Unless documented otherwise, all other `_` prefix properties,
111+
methods and modules are *private*.
112+
"""
113+
97114
def to_dict(
98115
self,
99116
*,
@@ -662,6 +679,21 @@ def set_pydantic_config(typ: Any, config: pydantic.ConfigDict) -> None:
662679
setattr(typ, "__pydantic_config__", config) # noqa: B010
663680

664681

682+
def add_request_id(obj: BaseModel, request_id: str | None) -> None:
683+
obj._request_id = request_id
684+
685+
# in Pydantic v1, using setattr like we do above causes the attribute
686+
# to be included when serializing the model which we don't want in this
687+
# case so we need to explicitly exclude it
688+
if not PYDANTIC_V2:
689+
try:
690+
exclude_fields = obj.__exclude_fields__ # type: ignore
691+
except AttributeError:
692+
cast(Any, obj).__exclude_fields__ = {"_request_id", "__exclude_fields__"}
693+
else:
694+
cast(Any, obj).__exclude_fields__ = {*(exclude_fields or {}), "_request_id", "__exclude_fields__"}
695+
696+
665697
# our use of subclasssing here causes weirdness for type checkers,
666698
# so we just pretend that we don't subclass
667699
if TYPE_CHECKING:

src/openai/_response.py

Lines changed: 7 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@
2626

2727
from ._types import NoneType
2828
from ._utils import is_given, extract_type_arg, is_annotated_type, extract_type_var_from_base
29-
from ._models import BaseModel, is_basemodel
29+
from ._models import BaseModel, is_basemodel, add_request_id
3030
from ._constants import RAW_RESPONSE_HEADER, OVERRIDE_CAST_TO_HEADER
3131
from ._streaming import Stream, AsyncStream, is_stream_class_type, extract_stream_chunk_type
3232
from ._exceptions import OpenAIError, APIResponseValidationError
@@ -315,6 +315,9 @@ class MyModel(BaseModel):
315315
if is_given(self._options.post_parser):
316316
parsed = self._options.post_parser(parsed)
317317

318+
if isinstance(parsed, BaseModel):
319+
add_request_id(parsed, self.request_id)
320+
318321
self._parsed_by_type[cache_key] = parsed
319322
return parsed
320323

@@ -419,6 +422,9 @@ class MyModel(BaseModel):
419422
if is_given(self._options.post_parser):
420423
parsed = self._options.post_parser(parsed)
421424

425+
if isinstance(parsed, BaseModel):
426+
add_request_id(parsed, self.request_id)
427+
422428
self._parsed_by_type[cache_key] = parsed
423429
return parsed
424430

tests/test_legacy_response.py

Lines changed: 21 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,27 @@ def test_response_parse_custom_model(client: OpenAI) -> None:
6666
assert obj.bar == 2
6767

6868

69+
def test_response_basemodel_request_id(client: OpenAI) -> None:
70+
response = LegacyAPIResponse(
71+
raw=httpx.Response(
72+
200,
73+
headers={"x-request-id": "my-req-id"},
74+
content=json.dumps({"foo": "hello!", "bar": 2}),
75+
),
76+
client=client,
77+
stream=False,
78+
stream_cls=None,
79+
cast_to=str,
80+
options=FinalRequestOptions.construct(method="get", url="/foo"),
81+
)
82+
83+
obj = response.parse(to=CustomModel)
84+
assert obj._request_id == "my-req-id"
85+
assert obj.foo == "hello!"
86+
assert obj.bar == 2
87+
assert obj.to_dict() == {"foo": "hello!", "bar": 2}
88+
89+
6990
def test_response_parse_annotated_type(client: OpenAI) -> None:
7091
response = LegacyAPIResponse(
7192
raw=httpx.Response(200, content=json.dumps({"foo": "hello!", "bar": 2})),

tests/test_response.py

Lines changed: 44 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
import pydantic
88

99
from openai import OpenAI, BaseModel, AsyncOpenAI
10+
from openai._compat import PYDANTIC_V2
1011
from openai._response import (
1112
APIResponse,
1213
BaseAPIResponse,
@@ -156,6 +157,49 @@ async def test_async_response_parse_custom_model(async_client: AsyncOpenAI) -> N
156157
assert obj.bar == 2
157158

158159

160+
def test_response_basemodel_request_id(client: OpenAI) -> None:
161+
response = APIResponse(
162+
raw=httpx.Response(
163+
200,
164+
headers={"x-request-id": "my-req-id"},
165+
content=json.dumps({"foo": "hello!", "bar": 2}),
166+
),
167+
client=client,
168+
stream=False,
169+
stream_cls=None,
170+
cast_to=str,
171+
options=FinalRequestOptions.construct(method="get", url="/foo"),
172+
)
173+
174+
obj = response.parse(to=CustomModel)
175+
assert obj._request_id == "my-req-id"
176+
assert obj.foo == "hello!"
177+
assert obj.bar == 2
178+
assert obj.to_dict() == {"foo": "hello!", "bar": 2}
179+
180+
181+
@pytest.mark.asyncio
182+
async def test_async_response_basemodel_request_id(client: OpenAI) -> None:
183+
response = AsyncAPIResponse(
184+
raw=httpx.Response(
185+
200,
186+
headers={"x-request-id": "my-req-id"},
187+
content=json.dumps({"foo": "hello!", "bar": 2}),
188+
),
189+
client=client,
190+
stream=False,
191+
stream_cls=None,
192+
cast_to=str,
193+
options=FinalRequestOptions.construct(method="get", url="/foo"),
194+
)
195+
196+
obj = await response.parse(to=CustomModel)
197+
assert obj._request_id == "my-req-id"
198+
assert obj.foo == "hello!"
199+
assert obj.bar == 2
200+
assert obj.to_dict() == {"foo": "hello!", "bar": 2}
201+
202+
159203
def test_response_parse_annotated_type(client: OpenAI) -> None:
160204
response = APIResponse(
161205
raw=httpx.Response(200, content=json.dumps({"foo": "hello!", "bar": 2})),

0 commit comments

Comments
 (0)