c63a5cfe-b235-4fbe-8bbb-82a9e02a482a-python 0.1.0a3__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- c63a5cfe_b235_4fbe_8bbb_82a9e02a482a_python-0.1.0a3.dist-info/METADATA +396 -0
- c63a5cfe_b235_4fbe_8bbb_82a9e02a482a_python-0.1.0a3.dist-info/RECORD +190 -0
- c63a5cfe_b235_4fbe_8bbb_82a9e02a482a_python-0.1.0a3.dist-info/WHEEL +4 -0
- c63a5cfe_b235_4fbe_8bbb_82a9e02a482a_python-0.1.0a3.dist-info/licenses/LICENSE +201 -0
- digitalocean_genai_sdk/__init__.py +99 -0
- digitalocean_genai_sdk/_base_client.py +1949 -0
- digitalocean_genai_sdk/_client.py +795 -0
- digitalocean_genai_sdk/_compat.py +219 -0
- digitalocean_genai_sdk/_constants.py +14 -0
- digitalocean_genai_sdk/_exceptions.py +108 -0
- digitalocean_genai_sdk/_files.py +123 -0
- digitalocean_genai_sdk/_models.py +805 -0
- digitalocean_genai_sdk/_qs.py +150 -0
- digitalocean_genai_sdk/_resource.py +43 -0
- digitalocean_genai_sdk/_response.py +832 -0
- digitalocean_genai_sdk/_streaming.py +333 -0
- digitalocean_genai_sdk/_types.py +219 -0
- digitalocean_genai_sdk/_utils/__init__.py +57 -0
- digitalocean_genai_sdk/_utils/_logs.py +25 -0
- digitalocean_genai_sdk/_utils/_proxy.py +65 -0
- digitalocean_genai_sdk/_utils/_reflection.py +42 -0
- digitalocean_genai_sdk/_utils/_resources_proxy.py +24 -0
- digitalocean_genai_sdk/_utils/_streams.py +12 -0
- digitalocean_genai_sdk/_utils/_sync.py +86 -0
- digitalocean_genai_sdk/_utils/_transform.py +447 -0
- digitalocean_genai_sdk/_utils/_typing.py +151 -0
- digitalocean_genai_sdk/_utils/_utils.py +422 -0
- digitalocean_genai_sdk/_version.py +4 -0
- digitalocean_genai_sdk/lib/.keep +4 -0
- digitalocean_genai_sdk/py.typed +0 -0
- digitalocean_genai_sdk/resources/__init__.py +145 -0
- digitalocean_genai_sdk/resources/agents/__init__.py +89 -0
- digitalocean_genai_sdk/resources/agents/agents.py +965 -0
- digitalocean_genai_sdk/resources/agents/api_keys.py +581 -0
- digitalocean_genai_sdk/resources/agents/child_agents.py +508 -0
- digitalocean_genai_sdk/resources/agents/functions.py +421 -0
- digitalocean_genai_sdk/resources/agents/knowledge_bases.py +346 -0
- digitalocean_genai_sdk/resources/agents/versions.py +298 -0
- digitalocean_genai_sdk/resources/api_keys/__init__.py +19 -0
- digitalocean_genai_sdk/resources/api_keys/api_keys.py +275 -0
- digitalocean_genai_sdk/resources/api_keys/api_keys_.py +529 -0
- digitalocean_genai_sdk/resources/auth/__init__.py +33 -0
- digitalocean_genai_sdk/resources/auth/agents/__init__.py +33 -0
- digitalocean_genai_sdk/resources/auth/agents/agents.py +102 -0
- digitalocean_genai_sdk/resources/auth/agents/token.py +173 -0
- digitalocean_genai_sdk/resources/auth/auth.py +102 -0
- digitalocean_genai_sdk/resources/chat.py +381 -0
- digitalocean_genai_sdk/resources/embeddings.py +201 -0
- digitalocean_genai_sdk/resources/indexing_jobs.py +543 -0
- digitalocean_genai_sdk/resources/knowledge_bases/__init__.py +33 -0
- digitalocean_genai_sdk/resources/knowledge_bases/data_sources.py +410 -0
- digitalocean_genai_sdk/resources/knowledge_bases/knowledge_bases.py +667 -0
- digitalocean_genai_sdk/resources/models.py +222 -0
- digitalocean_genai_sdk/resources/providers/__init__.py +47 -0
- digitalocean_genai_sdk/resources/providers/anthropic/__init__.py +33 -0
- digitalocean_genai_sdk/resources/providers/anthropic/anthropic.py +102 -0
- digitalocean_genai_sdk/resources/providers/anthropic/keys.py +662 -0
- digitalocean_genai_sdk/resources/providers/openai/__init__.py +33 -0
- digitalocean_genai_sdk/resources/providers/openai/keys.py +658 -0
- digitalocean_genai_sdk/resources/providers/openai/openai.py +102 -0
- digitalocean_genai_sdk/resources/providers/providers.py +134 -0
- digitalocean_genai_sdk/resources/regions.py +191 -0
- digitalocean_genai_sdk/types/__init__.py +57 -0
- digitalocean_genai_sdk/types/agent_create_params.py +39 -0
- digitalocean_genai_sdk/types/agent_create_response.py +16 -0
- digitalocean_genai_sdk/types/agent_delete_response.py +16 -0
- digitalocean_genai_sdk/types/agent_list_params.py +18 -0
- digitalocean_genai_sdk/types/agent_list_response.py +198 -0
- digitalocean_genai_sdk/types/agent_retrieve_response.py +16 -0
- digitalocean_genai_sdk/types/agent_update_params.py +65 -0
- digitalocean_genai_sdk/types/agent_update_response.py +16 -0
- digitalocean_genai_sdk/types/agent_update_status_params.py +16 -0
- digitalocean_genai_sdk/types/agent_update_status_response.py +16 -0
- digitalocean_genai_sdk/types/agents/__init__.py +31 -0
- digitalocean_genai_sdk/types/agents/api_key_create_params.py +15 -0
- digitalocean_genai_sdk/types/agents/api_key_create_response.py +12 -0
- digitalocean_genai_sdk/types/agents/api_key_delete_response.py +12 -0
- digitalocean_genai_sdk/types/agents/api_key_list_params.py +15 -0
- digitalocean_genai_sdk/types/agents/api_key_list_response.py +18 -0
- digitalocean_genai_sdk/types/agents/api_key_regenerate_response.py +12 -0
- digitalocean_genai_sdk/types/agents/api_key_update_params.py +19 -0
- digitalocean_genai_sdk/types/agents/api_key_update_response.py +12 -0
- digitalocean_genai_sdk/types/agents/api_link_knowledge_base_output.py +16 -0
- digitalocean_genai_sdk/types/agents/api_links.py +21 -0
- digitalocean_genai_sdk/types/agents/api_meta.py +15 -0
- digitalocean_genai_sdk/types/agents/child_agent_add_params.py +22 -0
- digitalocean_genai_sdk/types/agents/child_agent_add_response.py +14 -0
- digitalocean_genai_sdk/types/agents/child_agent_delete_response.py +13 -0
- digitalocean_genai_sdk/types/agents/child_agent_update_params.py +24 -0
- digitalocean_genai_sdk/types/agents/child_agent_update_response.py +18 -0
- digitalocean_genai_sdk/types/agents/child_agent_view_response.py +16 -0
- digitalocean_genai_sdk/types/agents/function_create_params.py +25 -0
- digitalocean_genai_sdk/types/agents/function_create_response.py +16 -0
- digitalocean_genai_sdk/types/agents/function_delete_response.py +16 -0
- digitalocean_genai_sdk/types/agents/function_update_params.py +29 -0
- digitalocean_genai_sdk/types/agents/function_update_response.py +16 -0
- digitalocean_genai_sdk/types/agents/knowledge_base_detach_response.py +16 -0
- digitalocean_genai_sdk/types/agents/version_list_params.py +15 -0
- digitalocean_genai_sdk/types/agents/version_list_response.py +118 -0
- digitalocean_genai_sdk/types/agents/version_update_params.py +15 -0
- digitalocean_genai_sdk/types/agents/version_update_response.py +30 -0
- digitalocean_genai_sdk/types/api_agent.py +263 -0
- digitalocean_genai_sdk/types/api_agent_api_key_info.py +22 -0
- digitalocean_genai_sdk/types/api_agreement.py +17 -0
- digitalocean_genai_sdk/types/api_anthropic_api_key_info.py +22 -0
- digitalocean_genai_sdk/types/api_deployment_visibility.py +9 -0
- digitalocean_genai_sdk/types/api_indexing_job.py +43 -0
- digitalocean_genai_sdk/types/api_key_list_params.py +42 -0
- digitalocean_genai_sdk/types/api_key_list_response.py +42 -0
- digitalocean_genai_sdk/types/api_keys/__init__.py +13 -0
- digitalocean_genai_sdk/types/api_keys/api_key_create_params.py +11 -0
- digitalocean_genai_sdk/types/api_keys/api_key_create_response.py +12 -0
- digitalocean_genai_sdk/types/api_keys/api_key_delete_response.py +12 -0
- digitalocean_genai_sdk/types/api_keys/api_key_list_params.py +15 -0
- digitalocean_genai_sdk/types/api_keys/api_key_list_response.py +18 -0
- digitalocean_genai_sdk/types/api_keys/api_key_update_params.py +15 -0
- digitalocean_genai_sdk/types/api_keys/api_key_update_regenerate_response.py +12 -0
- digitalocean_genai_sdk/types/api_keys/api_key_update_response.py +12 -0
- digitalocean_genai_sdk/types/api_keys/api_model_api_key_info.py +22 -0
- digitalocean_genai_sdk/types/api_knowledge_base.py +37 -0
- digitalocean_genai_sdk/types/api_model.py +57 -0
- digitalocean_genai_sdk/types/api_model_version.py +15 -0
- digitalocean_genai_sdk/types/api_openai_api_key_info.py +25 -0
- digitalocean_genai_sdk/types/api_retrieval_method.py +13 -0
- digitalocean_genai_sdk/types/auth/__init__.py +3 -0
- digitalocean_genai_sdk/types/auth/agents/__init__.py +6 -0
- digitalocean_genai_sdk/types/auth/agents/token_create_params.py +13 -0
- digitalocean_genai_sdk/types/auth/agents/token_create_response.py +13 -0
- digitalocean_genai_sdk/types/chat_completion_request_message_content_part_text_param.py +15 -0
- digitalocean_genai_sdk/types/chat_completion_token_logprob.py +57 -0
- digitalocean_genai_sdk/types/chat_create_completion_params.py +208 -0
- digitalocean_genai_sdk/types/chat_create_completion_response.py +81 -0
- digitalocean_genai_sdk/types/embedding_create_params.py +28 -0
- digitalocean_genai_sdk/types/embedding_create_response.py +41 -0
- digitalocean_genai_sdk/types/indexing_job_create_params.py +14 -0
- digitalocean_genai_sdk/types/indexing_job_create_response.py +12 -0
- digitalocean_genai_sdk/types/indexing_job_list_params.py +15 -0
- digitalocean_genai_sdk/types/indexing_job_list_response.py +18 -0
- digitalocean_genai_sdk/types/indexing_job_retrieve_data_sources_response.py +52 -0
- digitalocean_genai_sdk/types/indexing_job_retrieve_response.py +12 -0
- digitalocean_genai_sdk/types/indexing_job_update_cancel_params.py +14 -0
- digitalocean_genai_sdk/types/indexing_job_update_cancel_response.py +12 -0
- digitalocean_genai_sdk/types/knowledge_base_create_params.py +64 -0
- digitalocean_genai_sdk/types/knowledge_base_create_response.py +12 -0
- digitalocean_genai_sdk/types/knowledge_base_delete_response.py +11 -0
- digitalocean_genai_sdk/types/knowledge_base_list_params.py +15 -0
- digitalocean_genai_sdk/types/knowledge_base_list_response.py +18 -0
- digitalocean_genai_sdk/types/knowledge_base_retrieve_response.py +30 -0
- digitalocean_genai_sdk/types/knowledge_base_update_params.py +27 -0
- digitalocean_genai_sdk/types/knowledge_base_update_response.py +12 -0
- digitalocean_genai_sdk/types/knowledge_bases/__init__.py +16 -0
- digitalocean_genai_sdk/types/knowledge_bases/api_file_upload_data_source.py +15 -0
- digitalocean_genai_sdk/types/knowledge_bases/api_file_upload_data_source_param.py +15 -0
- digitalocean_genai_sdk/types/knowledge_bases/api_knowledge_base_data_source.py +35 -0
- digitalocean_genai_sdk/types/knowledge_bases/api_spaces_data_source.py +15 -0
- digitalocean_genai_sdk/types/knowledge_bases/api_spaces_data_source_param.py +15 -0
- digitalocean_genai_sdk/types/knowledge_bases/api_web_crawler_data_source.py +26 -0
- digitalocean_genai_sdk/types/knowledge_bases/api_web_crawler_data_source_param.py +25 -0
- digitalocean_genai_sdk/types/knowledge_bases/data_source_create_params.py +33 -0
- digitalocean_genai_sdk/types/knowledge_bases/data_source_create_response.py +12 -0
- digitalocean_genai_sdk/types/knowledge_bases/data_source_delete_response.py +13 -0
- digitalocean_genai_sdk/types/knowledge_bases/data_source_list_params.py +15 -0
- digitalocean_genai_sdk/types/knowledge_bases/data_source_list_response.py +18 -0
- digitalocean_genai_sdk/types/model.py +21 -0
- digitalocean_genai_sdk/types/model_list_response.py +15 -0
- digitalocean_genai_sdk/types/providers/__init__.py +3 -0
- digitalocean_genai_sdk/types/providers/anthropic/__init__.py +14 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_create_params.py +13 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_create_response.py +12 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_delete_response.py +12 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_list_agents_params.py +15 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_list_agents_response.py +22 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_list_params.py +15 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_list_response.py +18 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_retrieve_response.py +12 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_update_params.py +17 -0
- digitalocean_genai_sdk/types/providers/anthropic/key_update_response.py +12 -0
- digitalocean_genai_sdk/types/providers/openai/__init__.py +14 -0
- digitalocean_genai_sdk/types/providers/openai/key_create_params.py +13 -0
- digitalocean_genai_sdk/types/providers/openai/key_create_response.py +12 -0
- digitalocean_genai_sdk/types/providers/openai/key_delete_response.py +12 -0
- digitalocean_genai_sdk/types/providers/openai/key_list_params.py +15 -0
- digitalocean_genai_sdk/types/providers/openai/key_list_response.py +18 -0
- digitalocean_genai_sdk/types/providers/openai/key_retrieve_agents_params.py +15 -0
- digitalocean_genai_sdk/types/providers/openai/key_retrieve_agents_response.py +22 -0
- digitalocean_genai_sdk/types/providers/openai/key_retrieve_response.py +12 -0
- digitalocean_genai_sdk/types/providers/openai/key_update_params.py +17 -0
- digitalocean_genai_sdk/types/providers/openai/key_update_response.py +12 -0
- digitalocean_genai_sdk/types/region_list_params.py +15 -0
- digitalocean_genai_sdk/types/region_list_response.py +23 -0
@@ -0,0 +1,219 @@
|
|
1
|
+
from __future__ import annotations
|
2
|
+
|
3
|
+
from typing import TYPE_CHECKING, Any, Union, Generic, TypeVar, Callable, cast, overload
|
4
|
+
from datetime import date, datetime
|
5
|
+
from typing_extensions import Self, Literal
|
6
|
+
|
7
|
+
import pydantic
|
8
|
+
from pydantic.fields import FieldInfo
|
9
|
+
|
10
|
+
from ._types import IncEx, StrBytesIntFloat
|
11
|
+
|
12
|
+
_T = TypeVar("_T")
|
13
|
+
_ModelT = TypeVar("_ModelT", bound=pydantic.BaseModel)
|
14
|
+
|
15
|
+
# --------------- Pydantic v2 compatibility ---------------
|
16
|
+
|
17
|
+
# Pyright incorrectly reports some of our functions as overriding a method when they don't
|
18
|
+
# pyright: reportIncompatibleMethodOverride=false
|
19
|
+
|
20
|
+
PYDANTIC_V2 = pydantic.VERSION.startswith("2.")
|
21
|
+
|
22
|
+
# v1 re-exports
|
23
|
+
if TYPE_CHECKING:
|
24
|
+
|
25
|
+
def parse_date(value: date | StrBytesIntFloat) -> date: # noqa: ARG001
|
26
|
+
...
|
27
|
+
|
28
|
+
def parse_datetime(value: Union[datetime, StrBytesIntFloat]) -> datetime: # noqa: ARG001
|
29
|
+
...
|
30
|
+
|
31
|
+
def get_args(t: type[Any]) -> tuple[Any, ...]: # noqa: ARG001
|
32
|
+
...
|
33
|
+
|
34
|
+
def is_union(tp: type[Any] | None) -> bool: # noqa: ARG001
|
35
|
+
...
|
36
|
+
|
37
|
+
def get_origin(t: type[Any]) -> type[Any] | None: # noqa: ARG001
|
38
|
+
...
|
39
|
+
|
40
|
+
def is_literal_type(type_: type[Any]) -> bool: # noqa: ARG001
|
41
|
+
...
|
42
|
+
|
43
|
+
def is_typeddict(type_: type[Any]) -> bool: # noqa: ARG001
|
44
|
+
...
|
45
|
+
|
46
|
+
else:
|
47
|
+
if PYDANTIC_V2:
|
48
|
+
from pydantic.v1.typing import (
|
49
|
+
get_args as get_args,
|
50
|
+
is_union as is_union,
|
51
|
+
get_origin as get_origin,
|
52
|
+
is_typeddict as is_typeddict,
|
53
|
+
is_literal_type as is_literal_type,
|
54
|
+
)
|
55
|
+
from pydantic.v1.datetime_parse import parse_date as parse_date, parse_datetime as parse_datetime
|
56
|
+
else:
|
57
|
+
from pydantic.typing import (
|
58
|
+
get_args as get_args,
|
59
|
+
is_union as is_union,
|
60
|
+
get_origin as get_origin,
|
61
|
+
is_typeddict as is_typeddict,
|
62
|
+
is_literal_type as is_literal_type,
|
63
|
+
)
|
64
|
+
from pydantic.datetime_parse import parse_date as parse_date, parse_datetime as parse_datetime
|
65
|
+
|
66
|
+
|
67
|
+
# refactored config
|
68
|
+
if TYPE_CHECKING:
|
69
|
+
from pydantic import ConfigDict as ConfigDict
|
70
|
+
else:
|
71
|
+
if PYDANTIC_V2:
|
72
|
+
from pydantic import ConfigDict
|
73
|
+
else:
|
74
|
+
# TODO: provide an error message here?
|
75
|
+
ConfigDict = None
|
76
|
+
|
77
|
+
|
78
|
+
# renamed methods / properties
|
79
|
+
def parse_obj(model: type[_ModelT], value: object) -> _ModelT:
|
80
|
+
if PYDANTIC_V2:
|
81
|
+
return model.model_validate(value)
|
82
|
+
else:
|
83
|
+
return cast(_ModelT, model.parse_obj(value)) # pyright: ignore[reportDeprecated, reportUnnecessaryCast]
|
84
|
+
|
85
|
+
|
86
|
+
def field_is_required(field: FieldInfo) -> bool:
|
87
|
+
if PYDANTIC_V2:
|
88
|
+
return field.is_required()
|
89
|
+
return field.required # type: ignore
|
90
|
+
|
91
|
+
|
92
|
+
def field_get_default(field: FieldInfo) -> Any:
|
93
|
+
value = field.get_default()
|
94
|
+
if PYDANTIC_V2:
|
95
|
+
from pydantic_core import PydanticUndefined
|
96
|
+
|
97
|
+
if value == PydanticUndefined:
|
98
|
+
return None
|
99
|
+
return value
|
100
|
+
return value
|
101
|
+
|
102
|
+
|
103
|
+
def field_outer_type(field: FieldInfo) -> Any:
|
104
|
+
if PYDANTIC_V2:
|
105
|
+
return field.annotation
|
106
|
+
return field.outer_type_ # type: ignore
|
107
|
+
|
108
|
+
|
109
|
+
def get_model_config(model: type[pydantic.BaseModel]) -> Any:
|
110
|
+
if PYDANTIC_V2:
|
111
|
+
return model.model_config
|
112
|
+
return model.__config__ # type: ignore
|
113
|
+
|
114
|
+
|
115
|
+
def get_model_fields(model: type[pydantic.BaseModel]) -> dict[str, FieldInfo]:
|
116
|
+
if PYDANTIC_V2:
|
117
|
+
return model.model_fields
|
118
|
+
return model.__fields__ # type: ignore
|
119
|
+
|
120
|
+
|
121
|
+
def model_copy(model: _ModelT, *, deep: bool = False) -> _ModelT:
|
122
|
+
if PYDANTIC_V2:
|
123
|
+
return model.model_copy(deep=deep)
|
124
|
+
return model.copy(deep=deep) # type: ignore
|
125
|
+
|
126
|
+
|
127
|
+
def model_json(model: pydantic.BaseModel, *, indent: int | None = None) -> str:
|
128
|
+
if PYDANTIC_V2:
|
129
|
+
return model.model_dump_json(indent=indent)
|
130
|
+
return model.json(indent=indent) # type: ignore
|
131
|
+
|
132
|
+
|
133
|
+
def model_dump(
|
134
|
+
model: pydantic.BaseModel,
|
135
|
+
*,
|
136
|
+
exclude: IncEx | None = None,
|
137
|
+
exclude_unset: bool = False,
|
138
|
+
exclude_defaults: bool = False,
|
139
|
+
warnings: bool = True,
|
140
|
+
mode: Literal["json", "python"] = "python",
|
141
|
+
) -> dict[str, Any]:
|
142
|
+
if PYDANTIC_V2 or hasattr(model, "model_dump"):
|
143
|
+
return model.model_dump(
|
144
|
+
mode=mode,
|
145
|
+
exclude=exclude,
|
146
|
+
exclude_unset=exclude_unset,
|
147
|
+
exclude_defaults=exclude_defaults,
|
148
|
+
# warnings are not supported in Pydantic v1
|
149
|
+
warnings=warnings if PYDANTIC_V2 else True,
|
150
|
+
)
|
151
|
+
return cast(
|
152
|
+
"dict[str, Any]",
|
153
|
+
model.dict( # pyright: ignore[reportDeprecated, reportUnnecessaryCast]
|
154
|
+
exclude=exclude,
|
155
|
+
exclude_unset=exclude_unset,
|
156
|
+
exclude_defaults=exclude_defaults,
|
157
|
+
),
|
158
|
+
)
|
159
|
+
|
160
|
+
|
161
|
+
def model_parse(model: type[_ModelT], data: Any) -> _ModelT:
|
162
|
+
if PYDANTIC_V2:
|
163
|
+
return model.model_validate(data)
|
164
|
+
return model.parse_obj(data) # pyright: ignore[reportDeprecated]
|
165
|
+
|
166
|
+
|
167
|
+
# generic models
|
168
|
+
if TYPE_CHECKING:
|
169
|
+
|
170
|
+
class GenericModel(pydantic.BaseModel): ...
|
171
|
+
|
172
|
+
else:
|
173
|
+
if PYDANTIC_V2:
|
174
|
+
# there no longer needs to be a distinction in v2 but
|
175
|
+
# we still have to create our own subclass to avoid
|
176
|
+
# inconsistent MRO ordering errors
|
177
|
+
class GenericModel(pydantic.BaseModel): ...
|
178
|
+
|
179
|
+
else:
|
180
|
+
import pydantic.generics
|
181
|
+
|
182
|
+
class GenericModel(pydantic.generics.GenericModel, pydantic.BaseModel): ...
|
183
|
+
|
184
|
+
|
185
|
+
# cached properties
|
186
|
+
if TYPE_CHECKING:
|
187
|
+
cached_property = property
|
188
|
+
|
189
|
+
# we define a separate type (copied from typeshed)
|
190
|
+
# that represents that `cached_property` is `set`able
|
191
|
+
# at runtime, which differs from `@property`.
|
192
|
+
#
|
193
|
+
# this is a separate type as editors likely special case
|
194
|
+
# `@property` and we don't want to cause issues just to have
|
195
|
+
# more helpful internal types.
|
196
|
+
|
197
|
+
class typed_cached_property(Generic[_T]):
|
198
|
+
func: Callable[[Any], _T]
|
199
|
+
attrname: str | None
|
200
|
+
|
201
|
+
def __init__(self, func: Callable[[Any], _T]) -> None: ...
|
202
|
+
|
203
|
+
@overload
|
204
|
+
def __get__(self, instance: None, owner: type[Any] | None = None) -> Self: ...
|
205
|
+
|
206
|
+
@overload
|
207
|
+
def __get__(self, instance: object, owner: type[Any] | None = None) -> _T: ...
|
208
|
+
|
209
|
+
def __get__(self, instance: object, owner: type[Any] | None = None) -> _T | Self:
|
210
|
+
raise NotImplementedError()
|
211
|
+
|
212
|
+
def __set_name__(self, owner: type[Any], name: str) -> None: ...
|
213
|
+
|
214
|
+
# __set__ is not defined at runtime, but @cached_property is designed to be settable
|
215
|
+
def __set__(self, instance: object, value: _T) -> None: ...
|
216
|
+
else:
|
217
|
+
from functools import cached_property as cached_property
|
218
|
+
|
219
|
+
typed_cached_property = cached_property
|
@@ -0,0 +1,14 @@
|
|
1
|
+
# File generated from our OpenAPI spec by Stainless. See CONTRIBUTING.md for details.
|
2
|
+
|
3
|
+
import httpx
|
4
|
+
|
5
|
+
RAW_RESPONSE_HEADER = "X-Stainless-Raw-Response"
|
6
|
+
OVERRIDE_CAST_TO_HEADER = "____stainless_override_cast_to"
|
7
|
+
|
8
|
+
# default timeout is 1 minute
|
9
|
+
DEFAULT_TIMEOUT = httpx.Timeout(timeout=60, connect=5.0)
|
10
|
+
DEFAULT_MAX_RETRIES = 2
|
11
|
+
DEFAULT_CONNECTION_LIMITS = httpx.Limits(max_connections=100, max_keepalive_connections=20)
|
12
|
+
|
13
|
+
INITIAL_RETRY_DELAY = 0.5
|
14
|
+
MAX_RETRY_DELAY = 8.0
|
@@ -0,0 +1,108 @@
|
|
1
|
+
# File generated from our OpenAPI spec by Stainless. See CONTRIBUTING.md for details.
|
2
|
+
|
3
|
+
from __future__ import annotations
|
4
|
+
|
5
|
+
from typing_extensions import Literal
|
6
|
+
|
7
|
+
import httpx
|
8
|
+
|
9
|
+
__all__ = [
|
10
|
+
"BadRequestError",
|
11
|
+
"AuthenticationError",
|
12
|
+
"PermissionDeniedError",
|
13
|
+
"NotFoundError",
|
14
|
+
"ConflictError",
|
15
|
+
"UnprocessableEntityError",
|
16
|
+
"RateLimitError",
|
17
|
+
"InternalServerError",
|
18
|
+
]
|
19
|
+
|
20
|
+
|
21
|
+
class DigitaloceanGenaiSDKError(Exception):
|
22
|
+
pass
|
23
|
+
|
24
|
+
|
25
|
+
class APIError(DigitaloceanGenaiSDKError):
|
26
|
+
message: str
|
27
|
+
request: httpx.Request
|
28
|
+
|
29
|
+
body: object | None
|
30
|
+
"""The API response body.
|
31
|
+
|
32
|
+
If the API responded with a valid JSON structure then this property will be the
|
33
|
+
decoded result.
|
34
|
+
|
35
|
+
If it isn't a valid JSON structure then this will be the raw response.
|
36
|
+
|
37
|
+
If there was no response associated with this error then it will be `None`.
|
38
|
+
"""
|
39
|
+
|
40
|
+
def __init__(self, message: str, request: httpx.Request, *, body: object | None) -> None: # noqa: ARG002
|
41
|
+
super().__init__(message)
|
42
|
+
self.request = request
|
43
|
+
self.message = message
|
44
|
+
self.body = body
|
45
|
+
|
46
|
+
|
47
|
+
class APIResponseValidationError(APIError):
|
48
|
+
response: httpx.Response
|
49
|
+
status_code: int
|
50
|
+
|
51
|
+
def __init__(self, response: httpx.Response, body: object | None, *, message: str | None = None) -> None:
|
52
|
+
super().__init__(message or "Data returned by API invalid for expected schema.", response.request, body=body)
|
53
|
+
self.response = response
|
54
|
+
self.status_code = response.status_code
|
55
|
+
|
56
|
+
|
57
|
+
class APIStatusError(APIError):
|
58
|
+
"""Raised when an API response has a status code of 4xx or 5xx."""
|
59
|
+
|
60
|
+
response: httpx.Response
|
61
|
+
status_code: int
|
62
|
+
|
63
|
+
def __init__(self, message: str, *, response: httpx.Response, body: object | None) -> None:
|
64
|
+
super().__init__(message, response.request, body=body)
|
65
|
+
self.response = response
|
66
|
+
self.status_code = response.status_code
|
67
|
+
|
68
|
+
|
69
|
+
class APIConnectionError(APIError):
|
70
|
+
def __init__(self, *, message: str = "Connection error.", request: httpx.Request) -> None:
|
71
|
+
super().__init__(message, request, body=None)
|
72
|
+
|
73
|
+
|
74
|
+
class APITimeoutError(APIConnectionError):
|
75
|
+
def __init__(self, request: httpx.Request) -> None:
|
76
|
+
super().__init__(message="Request timed out.", request=request)
|
77
|
+
|
78
|
+
|
79
|
+
class BadRequestError(APIStatusError):
|
80
|
+
status_code: Literal[400] = 400 # pyright: ignore[reportIncompatibleVariableOverride]
|
81
|
+
|
82
|
+
|
83
|
+
class AuthenticationError(APIStatusError):
|
84
|
+
status_code: Literal[401] = 401 # pyright: ignore[reportIncompatibleVariableOverride]
|
85
|
+
|
86
|
+
|
87
|
+
class PermissionDeniedError(APIStatusError):
|
88
|
+
status_code: Literal[403] = 403 # pyright: ignore[reportIncompatibleVariableOverride]
|
89
|
+
|
90
|
+
|
91
|
+
class NotFoundError(APIStatusError):
|
92
|
+
status_code: Literal[404] = 404 # pyright: ignore[reportIncompatibleVariableOverride]
|
93
|
+
|
94
|
+
|
95
|
+
class ConflictError(APIStatusError):
|
96
|
+
status_code: Literal[409] = 409 # pyright: ignore[reportIncompatibleVariableOverride]
|
97
|
+
|
98
|
+
|
99
|
+
class UnprocessableEntityError(APIStatusError):
|
100
|
+
status_code: Literal[422] = 422 # pyright: ignore[reportIncompatibleVariableOverride]
|
101
|
+
|
102
|
+
|
103
|
+
class RateLimitError(APIStatusError):
|
104
|
+
status_code: Literal[429] = 429 # pyright: ignore[reportIncompatibleVariableOverride]
|
105
|
+
|
106
|
+
|
107
|
+
class InternalServerError(APIStatusError):
|
108
|
+
pass
|
@@ -0,0 +1,123 @@
|
|
1
|
+
from __future__ import annotations
|
2
|
+
|
3
|
+
import io
|
4
|
+
import os
|
5
|
+
import pathlib
|
6
|
+
from typing import overload
|
7
|
+
from typing_extensions import TypeGuard
|
8
|
+
|
9
|
+
import anyio
|
10
|
+
|
11
|
+
from ._types import (
|
12
|
+
FileTypes,
|
13
|
+
FileContent,
|
14
|
+
RequestFiles,
|
15
|
+
HttpxFileTypes,
|
16
|
+
Base64FileInput,
|
17
|
+
HttpxFileContent,
|
18
|
+
HttpxRequestFiles,
|
19
|
+
)
|
20
|
+
from ._utils import is_tuple_t, is_mapping_t, is_sequence_t
|
21
|
+
|
22
|
+
|
23
|
+
def is_base64_file_input(obj: object) -> TypeGuard[Base64FileInput]:
|
24
|
+
return isinstance(obj, io.IOBase) or isinstance(obj, os.PathLike)
|
25
|
+
|
26
|
+
|
27
|
+
def is_file_content(obj: object) -> TypeGuard[FileContent]:
|
28
|
+
return (
|
29
|
+
isinstance(obj, bytes) or isinstance(obj, tuple) or isinstance(obj, io.IOBase) or isinstance(obj, os.PathLike)
|
30
|
+
)
|
31
|
+
|
32
|
+
|
33
|
+
def assert_is_file_content(obj: object, *, key: str | None = None) -> None:
|
34
|
+
if not is_file_content(obj):
|
35
|
+
prefix = f"Expected entry at `{key}`" if key is not None else f"Expected file input `{obj!r}`"
|
36
|
+
raise RuntimeError(
|
37
|
+
f"{prefix} to be bytes, an io.IOBase instance, PathLike or a tuple but received {type(obj)} instead."
|
38
|
+
) from None
|
39
|
+
|
40
|
+
|
41
|
+
@overload
|
42
|
+
def to_httpx_files(files: None) -> None: ...
|
43
|
+
|
44
|
+
|
45
|
+
@overload
|
46
|
+
def to_httpx_files(files: RequestFiles) -> HttpxRequestFiles: ...
|
47
|
+
|
48
|
+
|
49
|
+
def to_httpx_files(files: RequestFiles | None) -> HttpxRequestFiles | None:
|
50
|
+
if files is None:
|
51
|
+
return None
|
52
|
+
|
53
|
+
if is_mapping_t(files):
|
54
|
+
files = {key: _transform_file(file) for key, file in files.items()}
|
55
|
+
elif is_sequence_t(files):
|
56
|
+
files = [(key, _transform_file(file)) for key, file in files]
|
57
|
+
else:
|
58
|
+
raise TypeError(f"Unexpected file type input {type(files)}, expected mapping or sequence")
|
59
|
+
|
60
|
+
return files
|
61
|
+
|
62
|
+
|
63
|
+
def _transform_file(file: FileTypes) -> HttpxFileTypes:
|
64
|
+
if is_file_content(file):
|
65
|
+
if isinstance(file, os.PathLike):
|
66
|
+
path = pathlib.Path(file)
|
67
|
+
return (path.name, path.read_bytes())
|
68
|
+
|
69
|
+
return file
|
70
|
+
|
71
|
+
if is_tuple_t(file):
|
72
|
+
return (file[0], _read_file_content(file[1]), *file[2:])
|
73
|
+
|
74
|
+
raise TypeError(f"Expected file types input to be a FileContent type or to be a tuple")
|
75
|
+
|
76
|
+
|
77
|
+
def _read_file_content(file: FileContent) -> HttpxFileContent:
|
78
|
+
if isinstance(file, os.PathLike):
|
79
|
+
return pathlib.Path(file).read_bytes()
|
80
|
+
return file
|
81
|
+
|
82
|
+
|
83
|
+
@overload
|
84
|
+
async def async_to_httpx_files(files: None) -> None: ...
|
85
|
+
|
86
|
+
|
87
|
+
@overload
|
88
|
+
async def async_to_httpx_files(files: RequestFiles) -> HttpxRequestFiles: ...
|
89
|
+
|
90
|
+
|
91
|
+
async def async_to_httpx_files(files: RequestFiles | None) -> HttpxRequestFiles | None:
|
92
|
+
if files is None:
|
93
|
+
return None
|
94
|
+
|
95
|
+
if is_mapping_t(files):
|
96
|
+
files = {key: await _async_transform_file(file) for key, file in files.items()}
|
97
|
+
elif is_sequence_t(files):
|
98
|
+
files = [(key, await _async_transform_file(file)) for key, file in files]
|
99
|
+
else:
|
100
|
+
raise TypeError("Unexpected file type input {type(files)}, expected mapping or sequence")
|
101
|
+
|
102
|
+
return files
|
103
|
+
|
104
|
+
|
105
|
+
async def _async_transform_file(file: FileTypes) -> HttpxFileTypes:
|
106
|
+
if is_file_content(file):
|
107
|
+
if isinstance(file, os.PathLike):
|
108
|
+
path = anyio.Path(file)
|
109
|
+
return (path.name, await path.read_bytes())
|
110
|
+
|
111
|
+
return file
|
112
|
+
|
113
|
+
if is_tuple_t(file):
|
114
|
+
return (file[0], await _async_read_file_content(file[1]), *file[2:])
|
115
|
+
|
116
|
+
raise TypeError(f"Expected file types input to be a FileContent type or to be a tuple")
|
117
|
+
|
118
|
+
|
119
|
+
async def _async_read_file_content(file: FileContent) -> HttpxFileContent:
|
120
|
+
if isinstance(file, os.PathLike):
|
121
|
+
return await anyio.Path(file).read_bytes()
|
122
|
+
|
123
|
+
return file
|